Skip to content

Commit a703f48

Browse files
committed
Maybe improve perf
1 parent 0009999 commit a703f48

File tree

2 files changed

+34
-25
lines changed

2 files changed

+34
-25
lines changed

src/compiler/checker.ts

+17-8
Original file line numberDiff line numberDiff line change
@@ -21248,7 +21248,7 @@ export function createTypeChecker(host: TypeCheckerHost): TypeChecker {
2124821248
return Ternary.False;
2124921249
}
2125021250
}
21251-
const maybeStart = maybeKeys.size;
21251+
// const maybeStart = maybeKeys.size;
2125221252
maybeKeys.push(id);
2125321253
const saveExpandingFlags = expandingFlags;
2125421254
if (recursionFlags & RecursionFlags.Source) {
@@ -21304,14 +21304,20 @@ export function createTypeChecker(host: TypeCheckerHost): TypeChecker {
2130421304
if (result === Ternary.True || result === Ternary.Maybe) {
2130521305
// If result is definitely true, record all maybe keys as having succeeded. Also, record Ternary.Maybe
2130621306
// results as having succeeded once we reach depth 0, but never record Ternary.Unknown results.
21307-
while (maybeKeys.size > maybeStart) {
21308-
const id = maybeKeys.pop();
21309-
relation.set(id, RelationComparisonResult.Succeeded | propagatingVarianceFlags);
21307+
while (true) {
21308+
const popped = maybeKeys.pop();
21309+
relation.set(popped, RelationComparisonResult.Succeeded | propagatingVarianceFlags);
21310+
if (popped === id) {
21311+
break;
21312+
}
2131021313
}
2131121314
}
2131221315
else {
21313-
while (maybeKeys.size > maybeStart) {
21314-
maybeKeys.pop();
21316+
while (true) {
21317+
const popped = maybeKeys.pop();
21318+
if (popped === id) {
21319+
break;
21320+
}
2131521321
}
2131621322
}
2131721323
}
@@ -21323,8 +21329,11 @@ export function createTypeChecker(host: TypeCheckerHost): TypeChecker {
2132321329
// A false result goes straight into global cache (when something is false under
2132421330
// assumptions it will also be false without assumptions)
2132521331
relation.set(id, (reportErrors ? RelationComparisonResult.Reported : 0) | RelationComparisonResult.Failed | propagatingVarianceFlags);
21326-
while (maybeKeys.size > maybeStart) {
21327-
maybeKeys.pop();
21332+
while (true) {
21333+
const popped = maybeKeys.pop();
21334+
if (popped === id) {
21335+
break;
21336+
}
2132821337
}
2132921338
}
2133021339
return result;

src/compiler/core.ts

+17-17
Original file line numberDiff line numberDiff line change
@@ -2889,38 +2889,38 @@ export interface StackSet<T extends {}> {
28892889
has(value: T): boolean;
28902890
push(value: T): void;
28912891
pop(): T;
2892-
get size(): number;
2892+
// get size(): number;
28932893
}
28942894

28952895
/** @internal */
28962896
export function createStackSet<T extends {}>(): StackSet<T> {
2897-
const refs = new Map<T, number>();
2898-
const stack: T[] = [];
2899-
let end = 0;
2897+
// Why var? It avoids TDZ checks in the runtime which can be costly.
2898+
// See: https://github.com/microsoft/TypeScript/issues/52924
2899+
/* eslint-disable no-var */
2900+
var set = new Set<T>();
2901+
var stack: T[] = [];
2902+
var end = 0;
2903+
/* eslint-enable no-var */
2904+
29002905
return {
29012906
has(value) {
2902-
return refs.has(value);
2907+
return set.has(value);
29032908
},
29042909
push(value) {
2905-
refs.set(value, (refs.get(value) ?? 0) + 1);
2910+
// Debug.assert(!set.has(value), "Value already pushed");
2911+
set.add(value);
29062912
stack[end] = value;
29072913
end++;
29082914
},
29092915
pop() {
29102916
end--;
2911-
Debug.assertGreaterThanOrEqual(end, 0);
2917+
// Debug.assertGreaterThanOrEqual(end, 0);
29122918
const value = stack[end];
2913-
const refCount = refs.get(value)! - 1;
2914-
if (refCount === 0) {
2915-
refs.delete(value);
2916-
}
2917-
else {
2918-
refs.set(value, refCount);
2919-
}
2919+
set.delete(value);
29202920
return value;
29212921
},
2922-
get size() {
2923-
return end;
2924-
},
2922+
// get size() {
2923+
// return end;
2924+
// },
29252925
};
29262926
}

0 commit comments

Comments
 (0)