2020-07-19 19:49:44 +02:00
// Copyright 2018-2020 the Deno authors. All rights reserved. MIT license.
( ( window ) => {
const core = window . Deno . core ;
const util = window . _ _bootstrap . util ;
// Using an object without a prototype because `Map` was causing GC problems.
const promiseTableMin = Object . create ( null ) ;
2020-08-26 18:20:22 +02:00
const decoder = new TextDecoder ( ) ;
2020-07-19 19:49:44 +02:00
// Note it's important that promiseId starts at 1 instead of 0, because sync
// messages are indicated with promiseId 0. If we ever add wrap around logic for
// overflows, this should be taken into account.
let _nextPromiseId = 1 ;
function nextPromiseId ( ) {
return _nextPromiseId ++ ;
}
function recordFromBufMinimal ( ui8 ) {
2020-08-26 18:20:22 +02:00
const headerLen = 12 ;
const header = ui8 . subarray ( 0 , headerLen ) ;
2020-07-19 19:49:44 +02:00
const buf32 = new Int32Array (
header . buffer ,
header . byteOffset ,
header . byteLength / 4 ,
) ;
const promiseId = buf32 [ 0 ] ;
const arg = buf32 [ 1 ] ;
const result = buf32 [ 2 ] ;
let err ;
if ( arg < 0 ) {
2020-08-26 18:20:22 +02:00
err = {
className : decoder . decode ( ui8 . subarray ( headerLen , headerLen + result ) ) ,
message : decoder . decode ( ui8 . subarray ( headerLen + result ) ) ,
} ;
2020-07-19 19:49:44 +02:00
} else if ( ui8 . length != 12 ) {
2020-08-07 22:47:18 +02:00
throw new TypeError ( "Malformed response message" ) ;
2020-07-19 19:49:44 +02:00
}
return {
promiseId ,
arg ,
result ,
err ,
} ;
}
function unwrapResponse ( res ) {
if ( res . err != null ) {
2020-10-05 20:35:51 +11:00
const ErrorClass = core . getErrorClass ( res . err . className ) ;
if ( ! ErrorClass ) {
throw new Error (
` Unregistered error class: " ${ res . err . className } " \n ${ res . err . message } \n Classes of errors returned from ops should be registered via Deno.core.registerErrorClass(). ` ,
) ;
}
throw new ErrorClass ( res . err . message ) ;
2020-07-19 19:49:44 +02:00
}
return res . result ;
}
const scratch32 = new Int32Array ( 3 ) ;
const scratchBytes = new Uint8Array (
scratch32 . buffer ,
scratch32 . byteOffset ,
scratch32 . byteLength ,
) ;
util . assert ( scratchBytes . byteLength === scratch32 . length * 4 ) ;
function asyncMsgFromRust ( ui8 ) {
const record = recordFromBufMinimal ( ui8 ) ;
const { promiseId } = record ;
const promise = promiseTableMin [ promiseId ] ;
delete promiseTableMin [ promiseId ] ;
util . assert ( promise ) ;
promise . resolve ( record ) ;
}
2020-08-26 00:22:15 +02:00
async function sendAsync ( opName , arg , zeroCopy ) {
2020-07-19 19:49:44 +02:00
const promiseId = nextPromiseId ( ) ; // AKA cmdId
scratch32 [ 0 ] = promiseId ;
scratch32 [ 1 ] = arg ;
scratch32 [ 2 ] = 0 ; // result
const promise = util . createResolvable ( ) ;
const buf = core . dispatchByName ( opName , scratchBytes , zeroCopy ) ;
if ( buf != null ) {
const record = recordFromBufMinimal ( buf ) ;
// Sync result.
promise . resolve ( record ) ;
} else {
// Async result.
promiseTableMin [ promiseId ] = promise ;
}
const res = await promise ;
return unwrapResponse ( res ) ;
}
2020-08-26 00:22:15 +02:00
function sendSync ( opName , arg , zeroCopy ) {
2020-07-19 19:49:44 +02:00
scratch32 [ 0 ] = 0 ; // promiseId 0 indicates sync
scratch32 [ 1 ] = arg ;
const res = core . dispatchByName ( opName , scratchBytes , zeroCopy ) ;
const resRecord = recordFromBufMinimal ( res ) ;
return unwrapResponse ( resRecord ) ;
}
window . _ _bootstrap . dispatchMinimal = {
asyncMsgFromRust ,
sendSync ,
sendAsync ,
} ;
} ) ( this ) ;