@@ -118,6 +118,7 @@ import {
118
118
createPrinterWithRemoveCommentsNeverAsciiEscape,
119
119
createPrinterWithRemoveCommentsOmitTrailingSemicolon,
120
120
createPropertyNameNodeForIdentifierOrLiteral,
121
+ createStackSet,
121
122
createSymbolTable,
122
123
createTextWriter,
123
124
Debug,
@@ -962,6 +963,7 @@ import {
962
963
SourceFile,
963
964
SpreadAssignment,
964
965
SpreadElement,
966
+ StackSet,
965
967
startsWith,
966
968
Statement,
967
969
stringContains,
@@ -20348,10 +20350,9 @@ export function createTypeChecker(host: TypeCheckerHost): TypeChecker {
20348
20350
20349
20351
let errorInfo: DiagnosticMessageChain | undefined;
20350
20352
let relatedInfo: [DiagnosticRelatedInformation, ...DiagnosticRelatedInformation[]] | undefined;
20351
- let maybeKeys: string[] ;
20353
+ let maybeKeys: StackSet< string> ;
20352
20354
let sourceStack: Type[];
20353
20355
let targetStack: Type[];
20354
- let maybeCount = 0;
20355
20356
let sourceDepth = 0;
20356
20357
let targetDepth = 0;
20357
20358
let expandingFlags = ExpandingFlags.None;
@@ -21224,29 +21225,31 @@ export function createTypeChecker(host: TypeCheckerHost): TypeChecker {
21224
21225
}
21225
21226
}
21226
21227
if (!maybeKeys) {
21227
- maybeKeys = [] ;
21228
+ maybeKeys = createStackSet() ;
21228
21229
sourceStack = [];
21229
21230
targetStack = [];
21230
21231
}
21231
21232
else {
21233
+ // If source and target are already being compared, consider them related with assumptions
21234
+ if (maybeKeys.has(id)) {
21235
+ return Ternary.Maybe;
21236
+ }
21237
+
21232
21238
// A key that starts with "*" is an indication that we have type references that reference constrained
21233
21239
// type parameters. For such keys we also check against the key we would have gotten if all type parameters
21234
21240
// were unconstrained.
21235
21241
const broadestEquivalentId = id.startsWith("*") ? getRelationKey(source, target, intersectionState, relation, /*ignoreConstraints*/ true) : undefined;
21236
- for (let i = 0; i < maybeCount; i++) {
21237
- // If source and target are already being compared, consider them related with assumptions
21238
- if (id === maybeKeys[i] || broadestEquivalentId && broadestEquivalentId === maybeKeys[i]) {
21239
- return Ternary.Maybe;
21240
- }
21242
+ if (broadestEquivalentId && maybeKeys.has(broadestEquivalentId)) {
21243
+ return Ternary.Maybe;
21241
21244
}
21245
+
21242
21246
if (sourceDepth === 100 || targetDepth === 100) {
21243
21247
overflow = true;
21244
21248
return Ternary.False;
21245
21249
}
21246
21250
}
21247
- const maybeStart = maybeCount;
21248
- maybeKeys[maybeCount] = id;
21249
- maybeCount++;
21251
+ const maybeStart = maybeKeys.size;
21252
+ maybeKeys.push(id);
21250
21253
const saveExpandingFlags = expandingFlags;
21251
21254
if (recursionFlags & RecursionFlags.Source) {
21252
21255
sourceStack[sourceDepth] = source;
@@ -21301,18 +21304,28 @@ export function createTypeChecker(host: TypeCheckerHost): TypeChecker {
21301
21304
if (result === Ternary.True || result === Ternary.Maybe) {
21302
21305
// If result is definitely true, record all maybe keys as having succeeded. Also, record Ternary.Maybe
21303
21306
// results as having succeeded once we reach depth 0, but never record Ternary.Unknown results.
21304
- for (let i = maybeStart; i < maybeCount; i++) {
21305
- relation.set(maybeKeys[i], RelationComparisonResult.Succeeded | propagatingVarianceFlags);
21307
+ while (maybeKeys.size > maybeStart) {
21308
+ const id = maybeKeys.pop();
21309
+ relation.set(id, RelationComparisonResult.Succeeded | propagatingVarianceFlags);
21310
+ }
21311
+ }
21312
+ else {
21313
+ while (maybeKeys.size > maybeStart) {
21314
+ maybeKeys.pop();
21306
21315
}
21307
21316
}
21308
- maybeCount = maybeStart;
21309
21317
}
21318
+ // Note: it's intentional that we don't pop in the else case;
21319
+ // we leave them on the stack such that when we hit depth zero
21320
+ // above, we can report all of them as successful.
21310
21321
}
21311
21322
else {
21312
21323
// A false result goes straight into global cache (when something is false under
21313
21324
// assumptions it will also be false without assumptions)
21314
21325
relation.set(id, (reportErrors ? RelationComparisonResult.Reported : 0) | RelationComparisonResult.Failed | propagatingVarianceFlags);
21315
- maybeCount = maybeStart;
21326
+ while (maybeKeys.size > maybeStart) {
21327
+ maybeKeys.pop();
21328
+ }
21316
21329
}
21317
21330
return result;
21318
21331
}
0 commit comments