rustdoc-search: less new Maps in unifyFunctionType

This is a major source of expense on generic queries,
and this commit reduces them.

Profile output:
https://notriddle.com/rustdoc-html-demo-5/profile-2/index.html
This commit is contained in:
Michael Howell 2023-11-17 13:05:18 -07:00
parent 4d7f952a02
commit 6d59452841

View File

@ -1329,9 +1329,9 @@ function initSearch(rawSearchIndex) {
*/ */
function unifyFunctionTypes(fnTypesIn, queryElems, whereClause, mgensIn, solutionCb) { function unifyFunctionTypes(fnTypesIn, queryElems, whereClause, mgensIn, solutionCb) {
/** /**
* @type Map<integer, integer> * @type Map<integer, integer>|null
*/ */
let mgens = new Map(mgensIn); let mgens = mgensIn === null ? null : new Map(mgensIn);
if (queryElems.length === 0) { if (queryElems.length === 0) {
return !solutionCb || solutionCb(mgens); return !solutionCb || solutionCb(mgens);
} }
@ -1382,12 +1382,14 @@ function initSearch(rawSearchIndex) {
fnTypesOffset, fnTypesOffset,
unbox, unbox,
} = backtracking.pop(); } = backtracking.pop();
mgens = new Map(mgensScratch); mgens = mgensScratch !== null ? new Map(mgensScratch) : null;
const fnType = fnTypesScratch[fnTypesOffset]; const fnType = fnTypesScratch[fnTypesOffset];
const queryElem = queryElems[queryElemsOffset]; const queryElem = queryElems[queryElemsOffset];
if (unbox) { if (unbox) {
if (fnType.id < 0) { if (fnType.id < 0) {
if (mgens.has(fnType.id) && mgens.get(fnType.id) !== 0) { if (mgens === null) {
mgens = new Map();
} else if (mgens.has(fnType.id) && mgens.get(fnType.id) !== 0) {
continue; continue;
} }
mgens.set(fnType.id, 0); mgens.set(fnType.id, 0);
@ -1401,7 +1403,9 @@ function initSearch(rawSearchIndex) {
i = queryElemsOffset - 1; i = queryElemsOffset - 1;
} else { } else {
if (fnType.id < 0) { if (fnType.id < 0) {
if (mgens.has(fnType.id) && mgens.get(fnType.id) !== queryElem.id) { if (mgens === null) {
mgens = new Map();
} else if (mgens.has(fnType.id) && mgens.get(fnType.id) !== queryElem.id) {
continue; continue;
} }
mgens.set(fnType.id, queryElem.id); mgens.set(fnType.id, queryElem.id);
@ -1456,7 +1460,7 @@ function initSearch(rawSearchIndex) {
if (!fnTypesScratch) { if (!fnTypesScratch) {
fnTypesScratch = fnTypes.slice(); fnTypesScratch = fnTypes.slice();
} }
if (!mgensScratch) { if (!mgensScratch && mgens !== null) {
mgensScratch = new Map(mgens); mgensScratch = new Map(mgens);
} }
backtracking.push({ backtracking.push({
@ -1478,11 +1482,20 @@ function initSearch(rawSearchIndex) {
// use the current candidate // use the current candidate
const {fnTypesOffset: candidate, mgensScratch: mgensNew} = matchCandidates.pop(); const {fnTypesOffset: candidate, mgensScratch: mgensNew} = matchCandidates.pop();
if (fnTypes[candidate].id < 0 && queryElems[i].id < 0) { if (fnTypes[candidate].id < 0 && queryElems[i].id < 0) {
if (mgens === null) {
mgens = new Map();
}
mgens.set(fnTypes[candidate].id, queryElems[i].id); mgens.set(fnTypes[candidate].id, queryElems[i].id);
} }
if (mgensNew !== null) {
if (mgens === null) {
mgens = mgensNew;
} else {
for (const [fid, qid] of mgensNew) { for (const [fid, qid] of mgensNew) {
mgens.set(fid, qid); mgens.set(fid, qid);
} }
}
}
// `i` and `j` are paired off // `i` and `j` are paired off
// `queryElems[i]` is left in place // `queryElems[i]` is left in place
// `fnTypes[j]` is swapped with `fnTypes[i]` to pair them off // `fnTypes[j]` is swapped with `fnTypes[i]` to pair them off
@ -1514,6 +1527,7 @@ function initSearch(rawSearchIndex) {
// or, if mgens[fnType.id] = 0, then we've matched this generic with a bare trait // or, if mgens[fnType.id] = 0, then we've matched this generic with a bare trait
// and should make that same decision everywhere it appears // and should make that same decision everywhere it appears
if (fnType.id < 0 && queryElem.id < 0) { if (fnType.id < 0 && queryElem.id < 0) {
if (mgens !== null) {
if (mgens.has(fnType.id) && mgens.get(fnType.id) !== queryElem.id) { if (mgens.has(fnType.id) && mgens.get(fnType.id) !== queryElem.id) {
return false; return false;
} }
@ -1525,6 +1539,7 @@ function initSearch(rawSearchIndex) {
return false; return false;
} }
} }
}
} else { } else {
if (queryElem.id === typeNameIdOfArrayOrSlice && if (queryElem.id === typeNameIdOfArrayOrSlice &&
(fnType.id === typeNameIdOfSlice || fnType.id === typeNameIdOfArray) (fnType.id === typeNameIdOfSlice || fnType.id === typeNameIdOfArray)
@ -1575,7 +1590,7 @@ function initSearch(rawSearchIndex) {
} }
// mgens[fnType.id] === 0 indicates that we committed to unboxing this generic // mgens[fnType.id] === 0 indicates that we committed to unboxing this generic
// mgens[fnType.id] === null indicates that we haven't decided yet // mgens[fnType.id] === null indicates that we haven't decided yet
if (mgens.has(fnType.id) && mgens.get(fnType.id) !== 0) { if (mgens !== null && mgens.has(fnType.id) && mgens.get(fnType.id) !== 0) {
return false; return false;
} }
// This is only a potential unbox if the search query appears in the where clause // This is only a potential unbox if the search query appears in the where clause