mirror of
https://github.com/denoland/deno.git
synced 2025-01-13 17:39:18 -05:00
tools(bench): rebootstrap (#12281)
Enable deno devs to bench/profile/test JS code changes without doing a full --release rebuild. Incremental release builds take ~4mn on M1s, often more on other machines ...
This commit is contained in:
parent
96530df7fb
commit
871dbb9f30
4 changed files with 85 additions and 0 deletions
33
tools/bench/README.md
Normal file
33
tools/bench/README.md
Normal file
|
@ -0,0 +1,33 @@
|
||||||
|
## Re-bootstrapping
|
||||||
|
|
||||||
|
Re-bootstrapping allows deno devs to bench/profile/test JS-side changes without
|
||||||
|
doing a full `cargo build --release --bin deno` which takes roughly ~4mn on M1s
|
||||||
|
more on other machines which significantly slows down iteration &
|
||||||
|
experimentation.
|
||||||
|
|
||||||
|
## Example
|
||||||
|
|
||||||
|
```js
|
||||||
|
import { benchSync, rebootstrap } from "./tools/bench/mod.js";
|
||||||
|
|
||||||
|
const bootstrap = rebootstrap([
|
||||||
|
"webidl",
|
||||||
|
"console",
|
||||||
|
"url",
|
||||||
|
"web",
|
||||||
|
"fetch",
|
||||||
|
]);
|
||||||
|
|
||||||
|
benchSync("resp_w_h", 1e6, () =>
|
||||||
|
new bootstrap.fetch.Response("yolo", {
|
||||||
|
status: 200,
|
||||||
|
headers: {
|
||||||
|
server: "deno",
|
||||||
|
"content-type": "text/plain",
|
||||||
|
},
|
||||||
|
}));
|
||||||
|
```
|
||||||
|
|
||||||
|
This code can then benched and profiled (using Chrome's DevTools) similar to
|
||||||
|
regular userland code and the original source files appear in the DevTools as
|
||||||
|
you would expect.
|
2
tools/bench/mod.js
Normal file
2
tools/bench/mod.js
Normal file
|
@ -0,0 +1,2 @@
|
||||||
|
export * from "./rebench.js";
|
||||||
|
export * from "./rebootstrap.js";
|
25
tools/bench/rebench.js
Normal file
25
tools/bench/rebench.js
Normal file
|
@ -0,0 +1,25 @@
|
||||||
|
export function benchSync(name, n, innerLoop) {
|
||||||
|
const t1 = Date.now();
|
||||||
|
for (let i = 0; i < n; i++) {
|
||||||
|
innerLoop(i);
|
||||||
|
}
|
||||||
|
const t2 = Date.now();
|
||||||
|
console.log(benchStats(name, n, t1, t2));
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function benchAsync(name, n, innerLoop) {
|
||||||
|
const t1 = Date.now();
|
||||||
|
for (let i = 0; i < n; i++) {
|
||||||
|
await innerLoop(i);
|
||||||
|
}
|
||||||
|
const t2 = Date.now();
|
||||||
|
console.log(benchStats(name, n, t1, t2));
|
||||||
|
}
|
||||||
|
|
||||||
|
function benchStats(name, n, t1, t2) {
|
||||||
|
const dt = (t2 - t1) / 1e3;
|
||||||
|
const r = n / dt;
|
||||||
|
const ns = Math.floor(dt / n * 1e9);
|
||||||
|
return `${name}:${" ".repeat(20 - name.length)}\t` +
|
||||||
|
`n = ${n}, dt = ${dt.toFixed(3)}s, r = ${r.toFixed(0)}/s, t = ${ns}ns/op`;
|
||||||
|
}
|
25
tools/bench/rebootstrap.js
Normal file
25
tools/bench/rebootstrap.js
Normal file
|
@ -0,0 +1,25 @@
|
||||||
|
import { dirname, fromFileUrl, join } from "https://deno.land/std/path/mod.ts";
|
||||||
|
import { expandGlobSync } from "https://deno.land/std/fs/mod.ts";
|
||||||
|
|
||||||
|
const ROOT_DIR = join(dirname(fromFileUrl(import.meta.url)), "..", "..");
|
||||||
|
|
||||||
|
export function rebootstrap(exts) {
|
||||||
|
[
|
||||||
|
"core/00_primordials.js",
|
||||||
|
...exts.map((e) => `ext/${e}/*.js`),
|
||||||
|
]
|
||||||
|
.map((pattern) => join(ROOT_DIR, pattern))
|
||||||
|
.map((pattern) => [...expandGlobSync(pattern)])
|
||||||
|
.flat()
|
||||||
|
.map((entry) => entry.path)
|
||||||
|
.forEach((file) => {
|
||||||
|
Deno.core.evalContext(Deno.readTextFileSync(file), file);
|
||||||
|
});
|
||||||
|
const bootstrap = globalThis.__bootstrap;
|
||||||
|
delete globalThis.__bootstrap;
|
||||||
|
// Patch dispatchEvent so we don't crash when MainWorker exits via:
|
||||||
|
// `window.dispatchEvent(new Event('unload'))`
|
||||||
|
// which fails since symbols are mangled during rebootstrap
|
||||||
|
globalThis.dispatchEvent = () => {};
|
||||||
|
return bootstrap;
|
||||||
|
}
|
Loading…
Reference in a new issue