Skip to content
Merged
Show file tree
Hide file tree
Changes from 1 commit
Commits
Show all changes
56 commits
Select commit Hold shift + click to select a range
ea02d94
wip
pavelsavara Mar 3, 2023
c1aa6c3
wip
pavelsavara Mar 3, 2023
bf9ef4d
wip
pavelsavara Mar 3, 2023
f883a72
Merge branch 'main' into browser_memory_snapshot3
pavelsavara Mar 3, 2023
3fe9aba
wip
pavelsavara Mar 3, 2023
f4225aa
Merge branch 'main' into browser_memory_snapshot3
pavelsavara Mar 3, 2023
d2e5b49
fix
pavelsavara Mar 3, 2023
c409f51
Merge branch 'main' into browser_memory_snapshot3
pavelsavara Mar 6, 2023
e695c42
feedback
pavelsavara Mar 6, 2023
1189015
lint
pavelsavara Mar 6, 2023
b92c82e
Update src/mono/wasm/memory-snapshot.md
pavelsavara Mar 6, 2023
7ccc8e6
Update src/mono/wasm/memory-snapshot.md
pavelsavara Mar 6, 2023
ec762ac
Update src/mono/wasm/memory-snapshot.md
pavelsavara Mar 6, 2023
91c0211
Update src/mono/wasm/memory-snapshot.md
pavelsavara Mar 6, 2023
20d548b
Update src/mono/wasm/memory-snapshot.md
pavelsavara Mar 6, 2023
8e6865c
more linear memory usage and recovery from failure of snapshot use
pavelsavara Mar 6, 2023
7bf7881
feedback
pavelsavara Mar 6, 2023
aec9b53
Merge branch 'main' into browser_memory_snapshot3
pavelsavara Mar 8, 2023
5651f37
start runtime twice for unit tests, so that the actual tests is runni…
pavelsavara Mar 8, 2023
a16f667
Merge branch 'main' into browser_memory_snapshot3
pavelsavara Mar 8, 2023
31b46ff
wip
pavelsavara Mar 8, 2023
43a73e1
continue loading non-snapshot assets
pavelsavara Mar 8, 2023
f49354a
disable adding methods to __indirect_function_table before we take me…
pavelsavara Mar 8, 2023
53781ab
rename startupMemoryCache
pavelsavara Mar 8, 2023
60b2c5a
try to fix threads
pavelsavara Mar 8, 2023
33b6e3e
Merge branch 'main' into browser_memory_snapshot3
pavelsavara Mar 8, 2023
977b46c
whitespace
pavelsavara Mar 8, 2023
933594b
fix
pavelsavara Mar 9, 2023
c68c9e8
fix storing memory
pavelsavara Mar 9, 2023
fa3e331
assert
pavelsavara Mar 9, 2023
c7f913b
- make snapshot hash more resilient
pavelsavara Mar 9, 2023
9cb1b72
fix
pavelsavara Mar 9, 2023
d1bbb9c
- renamed memorySnapshotSkippedOrDone
pavelsavara Mar 9, 2023
ee4ba76
- rename loadedMemorySnapshot
pavelsavara Mar 9, 2023
83f0032
Merge branch 'main' into browser_memory_snapshot3
pavelsavara Mar 9, 2023
85ac81d
fix
pavelsavara Mar 9, 2023
b7790f2
Merge branch 'main' into browser_memory_snapshot3
pavelsavara Mar 10, 2023
3168c42
- internal config exitAfterSnapshot
pavelsavara Mar 10, 2023
4a25447
measure
pavelsavara Mar 10, 2023
5067cf7
Merge branch 'main' into browser_memory_snapshot3
pavelsavara Mar 11, 2023
1761786
fix
pavelsavara Mar 11, 2023
94e931a
fix
pavelsavara Mar 13, 2023
2a7508e
optimize timezone detection
pavelsavara Mar 13, 2023
9759ad8
feedback
pavelsavara Mar 13, 2023
1f4e4c6
fix
pavelsavara Mar 13, 2023
e2a2cd4
- enable snapshot in blazor
pavelsavara Mar 13, 2023
eaaeaa5
store cache key
pavelsavara Mar 13, 2023
bc86ffe
feedback and cleanup
pavelsavara Mar 13, 2023
59a7a56
fix debugging
pavelsavara Mar 13, 2023
6d9f7ee
fix
pavelsavara Mar 14, 2023
3e961d1
Merge branch 'main' into browser_memory_snapshot3
pavelsavara Mar 14, 2023
f617787
Update src/mono/wasm/memory-snapshot.md
pavelsavara Mar 14, 2023
cfc5acf
Update src/mono/wasm/memory-snapshot.md
pavelsavara Mar 14, 2023
c78ef57
Update src/mono/wasm/runtime/assets.ts
pavelsavara Mar 14, 2023
7309e43
Update src/mono/wasm/runtime/assets.ts
pavelsavara Mar 14, 2023
5f8aeef
Merge branch 'main' into browser_memory_snapshot3
pavelsavara Mar 14, 2023
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
Prev Previous commit
Next Next commit
more linear memory usage and recovery from failure of snapshot use
  • Loading branch information
pavelsavara committed Mar 6, 2023
commit 8e6865cf6ff5db7995c50fef7e139d19078ec192
7 changes: 5 additions & 2 deletions src/mono/wasm/runtime/assets.ts
Original file line number Diff line number Diff line change
Expand Up @@ -8,7 +8,7 @@ import { mono_wasm_load_bytes_into_heap } from "./memory";
import { endMeasure, MeasuredBlock, startMeasure } from "./profiler";
import { createPromiseController, PromiseAndController } from "./promise-controller";
import { delay } from "./promise-utils";
import { abort_startup, beforeOnRuntimeInitialized } from "./startup";
import { abort_startup, beforeOnRuntimeInitialized, memorySnapshotIsResolved } from "./startup";
import { AssetBehaviours, AssetEntry, AssetEntryInternal, LoadingResource, mono_assert, ResourceRequest } from "./types";
import { InstantiateWasmSuccessCallback, VoidPtr } from "./types/emscripten";

Expand Down Expand Up @@ -59,7 +59,10 @@ export function resolve_asset_path(behavior: AssetBehaviours) {
return asset;
}
export async function mono_download_assets(): Promise<void> {
if (runtimeHelpers.memoryIsLoaded) {
// continue after we know if memory snapshot is available or not
await memorySnapshotIsResolved.promise;

if (runtimeHelpers.useMemorySnapshot) {
allAssetsInMemory.promise_control.resolve();
return;
}
Expand Down
55 changes: 37 additions & 18 deletions src/mono/wasm/runtime/startup.ts
Original file line number Diff line number Diff line change
Expand Up @@ -27,7 +27,7 @@ import { mono_wasm_init_diagnostics } from "./diagnostics";
import { preAllocatePThreadWorkerPool, instantiateWasmPThreadWorkerPool } from "./pthreads/browser";
import { export_linker } from "./exports-linker";
import { endMeasure, MeasuredBlock, startMeasure } from "./profiler";
import { getMemory, storeMemory } from "./storage";
import { getMemorySnapshot, storeMemorySnapshot, getMemorySnapshotSize } from "./storage";

// legacy
import { init_legacy_exports } from "./net6-legacy/corebindings";
Expand All @@ -39,6 +39,7 @@ let configLoaded = false;
export const dotnetReady = createPromiseController<any>();
export const afterConfigLoaded = createPromiseController<MonoConfig>();
export const beforeInstantiateWasm = createPromiseController<void>();
export const memorySnapshotIsResolved = createPromiseController<void>();
export const afterInstantiateWasm = createPromiseController<void>();
export const beforePreInit = createPromiseController<void>();
export const afterPreInit = createPromiseController<void>();
Expand Down Expand Up @@ -108,7 +109,9 @@ function instantiateWasm(

const mark = startMeasure();
if (userInstantiateWasm) {
// user wasm doesn't support memory snapshots
beforeInstantiateWasm.promise_control.resolve();
memorySnapshotIsResolved.promise_control.resolve();
const exports = userInstantiateWasm(imports, (instance: WebAssembly.Instance, module: WebAssembly.Module | undefined) => {
endMeasure(mark, MeasuredBlock.instantiateWasm);
afterInstantiateWasm.promise_control.resolve();
Expand Down Expand Up @@ -179,7 +182,6 @@ async function preInitWorkerAsync() {
const mark = startMeasure();
try {
if (runtimeHelpers.diagnosticTracing) console.debug("MONO_WASM: preInitWorker");
beforePreInit.promise_control.resolve();
mono_wasm_pre_init_essential(true);
await init_polyfills_async();
afterPreInit.promise_control.resolve();
Expand Down Expand Up @@ -286,6 +288,7 @@ export function abort_startup(reason: any, should_exit: boolean): void {
if (runtimeHelpers.diagnosticTracing) console.trace("MONO_WASM: abort_startup");
dotnetReady.promise_control.reject(reason);
beforeInstantiateWasm.promise_control.reject(reason);
memorySnapshotIsResolved.promise_control.reject(reason);
afterInstantiateWasm.promise_control.reject(reason);
beforePreInit.promise_control.reject(reason);
afterPreInit.promise_control.reject(reason);
Expand Down Expand Up @@ -334,7 +337,9 @@ async function mono_wasm_pre_init_full(): Promise<void> {
if (runtimeHelpers.diagnosticTracing) console.debug("MONO_WASM: mono_wasm_pre_init_full");
Module.addRunDependency("mono_wasm_pre_init_full");

// continue after wasm is downloaded
await beforeInstantiateWasm.promise;

await mono_download_assets();

Module.removeRunDependency("mono_wasm_pre_init_full");
Expand All @@ -348,8 +353,8 @@ async function mono_wasm_before_user_runtime_initialized(): Promise<void> {
mono_wasm_globalization_init();

if (!runtimeHelpers.mono_wasm_load_runtime_done) mono_wasm_load_runtime("unused", config.debugLevel);
if (config.cacheMemory && !runtimeHelpers.memoryIsLoaded) {
await storeMemory(Module.HEAP8.buffer);
if (config.cacheMemory && !runtimeHelpers.useMemorySnapshot) {
await storeMemorySnapshot(Module.HEAP8.buffer);
}
if (MonoWasmThreads) {
preAllocatePThreadWorkerPool(MONO_PTHREAD_POOL_SIZE, config);
Expand Down Expand Up @@ -460,19 +465,23 @@ async function instantiate_wasm_module(
): Promise<void> {
// this is called so early that even Module exports like addRunDependency don't exist yet
try {
let memoryBytes: ArrayBuffer | undefined = undefined;
let memorySize: number | undefined = undefined;
await mono_wasm_load_config(Module.configSrc);
if (runtimeHelpers.diagnosticTracing) console.debug("MONO_WASM: instantiate_wasm_module");
const assetToLoad = resolve_asset_path("dotnetwasm");
// FIXME: this would not apply re-try (on connection reset during download) for dotnet.wasm because we could not download the buffer before we pass it to instantiate_wasm_asset
await start_asset_download(assetToLoad);
beforeInstantiateWasm.promise_control.resolve();

if (config.cacheMemory && config.assetsHash) {
memoryBytes = await getMemory();
runtimeHelpers.memoryIsLoaded = !!memoryBytes;
memorySize = await getMemorySnapshotSize();
runtimeHelpers.useMemorySnapshot = !!memorySize;
}
if (!runtimeHelpers.useMemorySnapshot) {
// we should start downloading DLLs etc as they are not in the snapshot
memorySnapshotIsResolved.promise_control.resolve();
}
beforeInstantiateWasm.promise_control.resolve();

// FIXME: this would not apply re-try (on connection reset during download) for dotnet.wasm because we could not download the buffer before we pass it to instantiate_wasm_asset
await start_asset_download(assetToLoad);
await beforePreInit.promise;
Module.addRunDependency("instantiate_wasm_module");

Expand All @@ -483,13 +492,23 @@ async function instantiate_wasm_module(
assetToLoad.buffer = null as any; // GC
if (runtimeHelpers.diagnosticTracing) console.debug("MONO_WASM: instantiate_wasm_module done");

if (runtimeHelpers.memoryIsLoaded) {
const wasmMemory = anyModule.asm.memory;
// .grow() takes a delta compared to the previous size
wasmMemory.grow((memoryBytes!.byteLength - wasmMemory.buffer.byteLength + 65535) >>> 16);
runtimeHelpers.updateGlobalBufferAndViews(wasmMemory.buffer);
Module.HEAP8.set(new Int8Array(memoryBytes!), 0);
if (runtimeHelpers.diagnosticTracing) console.debug("MONO_WASM: Loaded memory from cache");
if (runtimeHelpers.useMemorySnapshot) {
try {
const wasmMemory = anyModule.asm.memory;
// .grow() takes a delta compared to the previous size
wasmMemory.grow((memorySize! - wasmMemory.buffer.byteLength + 65535) >>> 16);
runtimeHelpers.updateGlobalBufferAndViews(wasmMemory.buffer);

// get the bytes after we re-sized the memory, so that we don't have too much memory in use at the same time
const memoryBytes = await getMemorySnapshot();
Module.HEAP8.set(new Int8Array(memoryBytes!), 0);
if (runtimeHelpers.diagnosticTracing) console.debug("MONO_WASM: Loaded memory from cache");
} catch (err) {
console.warn("MONO_WASM: failed to load memory snapshot", err);
runtimeHelpers.useMemorySnapshot = false;
}
// now we know if the loading of memory succeeded or not, we can start loading the rest of the assets
memorySnapshotIsResolved.promise_control.resolve();
}
afterInstantiateWasm.promise_control.resolve();
} catch (err) {
Expand Down Expand Up @@ -531,7 +550,7 @@ export function mono_wasm_load_runtime(unused?: string, debugLevel?: number): vo
runtimeHelpers.mono_wasm_load_runtime_done = true;
try {
const mark = startMeasure();
if (!runtimeHelpers.memoryIsLoaded) {
if (!runtimeHelpers.useMemorySnapshot) {
if (debugLevel == undefined) {
debugLevel = 0;
if (config && config.debugLevel) {
Expand Down
38 changes: 30 additions & 8 deletions src/mono/wasm/runtime/storage.ts
Original file line number Diff line number Diff line change
Expand Up @@ -8,7 +8,7 @@ import { runtimeHelpers } from "./imports";
const memoryPrefix = "https://dotnet.generated.invalid/wasm-memory";

// adapted from Blazor's WebAssemblyResourceLoader.ts
async function open(): Promise<Cache | null> {
async function openCache(): Promise<Cache | null> {
// caches will be undefined if we're running on an insecure origin (secure means https or localhost)
if (typeof globalThis.caches === "undefined") {
return null;
Expand Down Expand Up @@ -43,14 +43,34 @@ async function open(): Promise<Cache | null> {
}
}

export async function getMemory(): Promise<ArrayBuffer | undefined> {
export async function getMemorySnapshotSize(): Promise<number | undefined> {
try {
const inputsHash = await getInputsHash();
if (!inputsHash) {
return undefined;
}
const cacheKey = `${memoryPrefix}-${ProductVersion}-${GitHash}-${inputsHash}`;
const cache = await open();
const cache = await openCache();
if (!cache) {
return undefined;
}
const res = await cache.match(cacheKey);
const contentLength = res?.headers.get("content-length");
return contentLength ? parseInt(contentLength) : undefined;
} catch (ex) {
console.warn("MONO_WASM: Failed find memory snapshot in the cache", ex);
return undefined;
}
}

export async function getMemorySnapshot(): Promise<ArrayBuffer | undefined> {
try {
const inputsHash = await getInputsHash();
if (!inputsHash) {
return undefined;
}
const cacheKey = `${memoryPrefix}-${ProductVersion}-${GitHash}-${inputsHash}`;
const cache = await openCache();
if (!cache) {
return undefined;
}
Expand All @@ -60,18 +80,19 @@ export async function getMemory(): Promise<ArrayBuffer | undefined> {
}
return res.arrayBuffer();
} catch (ex) {
console.warn("MONO_WASM: Failed load memory snapshot from the cache", ex);
return undefined;
}
}

export async function storeMemory(memory: ArrayBuffer) {
export async function storeMemorySnapshot(memory: ArrayBuffer) {
try {
const inputsHash = await getInputsHash();
if (!inputsHash) {
return;
}
const cacheKey = `${memoryPrefix}-${ProductVersion}-${GitHash}-${inputsHash}`;
const cache = await open();
const cache = await openCache();
if (!cache) {
return;
}
Expand All @@ -85,15 +106,16 @@ export async function storeMemory(memory: ArrayBuffer) {

await cache.put(cacheKey, responseToCache);

cleanupMemory(cacheKey); // no await
cleanupMemorySnapshots(cacheKey); // no await
} catch (ex) {
console.warn("MONO_WASM: Failed to store memory snapshot in the cache", ex);
return;
}
}

export async function cleanupMemory(protectKey: string) {
export async function cleanupMemorySnapshots(protectKey: string) {
try {
const cache = await open();
const cache = await openCache();
if (!cache) {
return;
}
Expand Down
2 changes: 1 addition & 1 deletion src/mono/wasm/runtime/types.ts
Original file line number Diff line number Diff line change
Expand Up @@ -233,7 +233,7 @@ export type RuntimeHelpers = {
locateFile: (path: string, prefix?: string) => string,
javaScriptExports: JavaScriptExports,
loadedFiles: string[],
memoryIsLoaded: boolean,
useMemorySnapshot: boolean,
subtle: SubtleCrypto | null,
preferredIcuAsset: string | null,
timezone: string | null,
Expand Down