mirror of
https://github.com/saymrwulf/onnxruntime.git
synced 2026-05-18 21:21:17 +00:00
### Description
This PR revises the backend registration.
The following describes the expected behavior after this change:
(**bolded are changed behavior**)
- (ort.min.js - built without webgpu support)
- loading: do not register 'webgpu' backend
- creating session without EP list: use default EP list ['webnn', 'cpu',
'wasm']
- creating session with ['webgpu'] as EP list: should fail with backend
not available
- (ort.webgpu.min.js - built with webgpu support)
- loading: **always register 'webgpu' backend**
( previous behavior: only register 'webgpu' backend when `navigator.gpu`
is available)
- creating session without EP list: use default EP list ['webgpu',
'webnn', 'cpu', 'wasm']
- when WebGPU is available (win): use WebGPU backend
- when WebGPU is unavailable (android): **should fail backend init,**
and try to use next backend in the list, 'webnn'
(previous behavior: does not fail backend init, but fail in JSEP init,
which was too late to switch to next backend)
- creating session with ['webgpu'] as EP list
- when WebGPU is available (win): use WebGPU backend
- when WebGPU is unavailable (android): **should fail backend init, and
because no more EP listed, fail.
related PRs: #18190 #18144
128 lines
4.5 KiB
TypeScript
128 lines
4.5 KiB
TypeScript
// Copyright (c) Microsoft Corporation. All rights reserved.
|
|
// Licensed under the MIT License.
|
|
|
|
import {readFile} from 'node:fs/promises';
|
|
import {InferenceSession, InferenceSessionHandler, SessionHandler, Tensor} from 'onnxruntime-common';
|
|
|
|
import {SerializableInternalBuffer, TensorMetadata} from './proxy-messages';
|
|
import {copyFromExternalBuffer, createSession, endProfiling, releaseSession, run} from './proxy-wrapper';
|
|
import {isGpuBufferSupportedType} from './wasm-common';
|
|
|
|
export const encodeTensorMetadata = (tensor: Tensor, getName: () => string): TensorMetadata => {
|
|
switch (tensor.location) {
|
|
case 'cpu':
|
|
return [tensor.type, tensor.dims, tensor.data, 'cpu'];
|
|
case 'gpu-buffer':
|
|
return [tensor.type, tensor.dims, {gpuBuffer: tensor.gpuBuffer}, 'gpu-buffer'];
|
|
default:
|
|
throw new Error(`invalid data location: ${tensor.location} for ${getName()}`);
|
|
}
|
|
};
|
|
|
|
export const decodeTensorMetadata = (tensor: TensorMetadata): Tensor => {
|
|
switch (tensor[3]) {
|
|
case 'cpu':
|
|
return new Tensor(tensor[0], tensor[2], tensor[1]);
|
|
case 'gpu-buffer': {
|
|
const dataType = tensor[0];
|
|
if (!isGpuBufferSupportedType(dataType)) {
|
|
throw new Error(`not supported data type: ${dataType} for deserializing GPU tensor`);
|
|
}
|
|
const {gpuBuffer, download, dispose} = tensor[2];
|
|
return Tensor.fromGpuBuffer(gpuBuffer, {dataType, dims: tensor[1], download, dispose});
|
|
}
|
|
default:
|
|
throw new Error(`invalid data location: ${tensor[3]}`);
|
|
}
|
|
};
|
|
|
|
export class OnnxruntimeWebAssemblySessionHandler implements InferenceSessionHandler {
|
|
private sessionId: number;
|
|
|
|
inputNames: string[];
|
|
outputNames: string[];
|
|
|
|
async fetchModelAndCopyToWasmMemory(path: string): Promise<SerializableInternalBuffer> {
|
|
// fetch model from url and move to wasm heap. The arraybufffer that held the http
|
|
// response is freed once we return
|
|
const response = await fetch(path);
|
|
if (response.status !== 200) {
|
|
throw new Error(`failed to load model: ${path}`);
|
|
}
|
|
const arrayBuffer = await response.arrayBuffer();
|
|
return copyFromExternalBuffer(new Uint8Array(arrayBuffer));
|
|
}
|
|
|
|
async loadModel(pathOrBuffer: string|Uint8Array, options?: InferenceSession.SessionOptions): Promise<void> {
|
|
let model: Parameters<typeof createSession>[0];
|
|
|
|
if (typeof pathOrBuffer === 'string') {
|
|
if (typeof process !== 'undefined' && process.versions && process.versions.node) {
|
|
// node
|
|
model = await readFile(pathOrBuffer);
|
|
} else {
|
|
// browser
|
|
// fetch model and copy to wasm heap.
|
|
model = await this.fetchModelAndCopyToWasmMemory(pathOrBuffer);
|
|
}
|
|
} else {
|
|
model = pathOrBuffer;
|
|
}
|
|
|
|
[this.sessionId, this.inputNames, this.outputNames] = await createSession(model, options);
|
|
}
|
|
|
|
async dispose(): Promise<void> {
|
|
return releaseSession(this.sessionId);
|
|
}
|
|
|
|
async run(feeds: SessionHandler.FeedsType, fetches: SessionHandler.FetchesType, options: InferenceSession.RunOptions):
|
|
Promise<SessionHandler.ReturnType> {
|
|
const inputArray: Tensor[] = [];
|
|
const inputIndices: number[] = [];
|
|
Object.entries(feeds).forEach(kvp => {
|
|
const name = kvp[0];
|
|
const tensor = kvp[1];
|
|
const index = this.inputNames.indexOf(name);
|
|
if (index === -1) {
|
|
throw new Error(`invalid input '${name}'`);
|
|
}
|
|
inputArray.push(tensor);
|
|
inputIndices.push(index);
|
|
});
|
|
|
|
const outputArray: Array<Tensor|null> = [];
|
|
const outputIndices: number[] = [];
|
|
Object.entries(fetches).forEach(kvp => {
|
|
const name = kvp[0];
|
|
const tensor = kvp[1];
|
|
const index = this.outputNames.indexOf(name);
|
|
if (index === -1) {
|
|
throw new Error(`invalid output '${name}'`);
|
|
}
|
|
outputArray.push(tensor);
|
|
outputIndices.push(index);
|
|
});
|
|
|
|
const inputs =
|
|
inputArray.map((t, i) => encodeTensorMetadata(t, () => `input "${this.inputNames[inputIndices[i]]}"`));
|
|
const outputs = outputArray.map(
|
|
(t, i) => t ? encodeTensorMetadata(t, () => `output "${this.outputNames[outputIndices[i]]}"`) : null);
|
|
|
|
const results = await run(this.sessionId, inputIndices, inputs, outputIndices, outputs, options);
|
|
|
|
const resultMap: SessionHandler.ReturnType = {};
|
|
for (let i = 0; i < results.length; i++) {
|
|
resultMap[this.outputNames[outputIndices[i]]] = outputArray[i] ?? decodeTensorMetadata(results[i]);
|
|
}
|
|
return resultMap;
|
|
}
|
|
|
|
startProfiling(): void {
|
|
// TODO: implement profiling
|
|
}
|
|
|
|
endProfiling(): void {
|
|
void endProfiling(this.sessionId);
|
|
}
|
|
}
|