blob: c62e633a164cb13545c0f2c5d1f670e872a3a293 [file] [log] [blame]
/**
* @license
* Copyright 2018 The Emscripten Authors
* SPDX-License-Identifier: MIT
*/
var LibraryWebSocket = {
$webSockets__deps: ['$HandleAllocator'],
$webSockets: "new HandleAllocator();",
$WS__deps: ['$webSockets', 'malloc'],
$WS: {
socketEvent: null,
getSocket(socketId) {
if (!webSockets.has(socketId)) {
return 0;
}
return webSockets.get(socketId);
},
getSocketEvent(socketId) {
// Singleton event pointer. Use EmscriptenWebSocketCloseEvent, which is
// the largest event struct
this.socketEvent ||= _malloc({{{ C_STRUCTS.EmscriptenWebSocketCloseEvent.__size__ }}});
{{{ makeSetValue('this.socketEvent', 0, 'socketId', 'u32') }}};
return this.socketEvent;
},
},
emscripten_websocket_get_ready_state__deps: ['$WS'],
emscripten_websocket_get_ready_state__proxy: 'sync',
emscripten_websocket_get_ready_state: (socketId, readyState) => {
var socket = WS.getSocket(socketId);
if (!socket) {
#if WEBSOCKET_DEBUG
dbg(`emscripten_websocket_get_ready_state(): Invalid socket ID ${socketId} specified!`);
#endif
return {{{ cDefs.EMSCRIPTEN_RESULT_INVALID_TARGET }}};
}
{{{ makeSetValue('readyState', '0', 'socket.readyState', 'i16') }}};
return {{{ cDefs.EMSCRIPTEN_RESULT_SUCCESS }}};
},
emscripten_websocket_get_buffered_amount__deps: ['$WS'],
emscripten_websocket_get_buffered_amount__proxy: 'sync',
emscripten_websocket_get_buffered_amount: (socketId, bufferedAmount) => {
var socket = WS.getSocket(socketId);
if (!socket) {
#if WEBSOCKET_DEBUG
dbg(`emscripten_websocket_get_buffered_amount(): Invalid socket ID ${socketId} specified!`);
#endif
return {{{ cDefs.EMSCRIPTEN_RESULT_INVALID_TARGET }}};
}
{{{ makeSetValue('bufferedAmount', '0', 'socket.bufferedAmount', '*') }}};
return {{{ cDefs.EMSCRIPTEN_RESULT_SUCCESS }}};
},
emscripten_websocket_get_extensions__deps: ['$WS', '$stringToUTF8'],
emscripten_websocket_get_extensions__proxy: 'sync',
emscripten_websocket_get_extensions: (socketId, extensions, extensionsLength) => {
var socket = WS.getSocket(socketId);
if (!socket) {
#if WEBSOCKET_DEBUG
dbg(`emscripten_websocket_get_extensions(): Invalid socket ID ${socketId} specified!`);
#endif
return {{{ cDefs.EMSCRIPTEN_RESULT_INVALID_TARGET }}};
}
if (!extensions) return {{{ cDefs.EMSCRIPTEN_RESULT_INVALID_PARAM }}};
stringToUTF8(socket.extensions, extensions, extensionsLength);
return {{{ cDefs.EMSCRIPTEN_RESULT_SUCCESS }}};
},
emscripten_websocket_get_extensions_length__deps: ['$WS'],
emscripten_websocket_get_extensions_length__proxy: 'sync',
emscripten_websocket_get_extensions_length: (socketId, extensionsLength) => {
var socket = WS.getSocket(socketId);
if (!socket) {
#if WEBSOCKET_DEBUG
dbg(`emscripten_websocket_get_extensions_length(): Invalid socket ID ${socketId} specified!`);
#endif
return {{{ cDefs.EMSCRIPTEN_RESULT_INVALID_TARGET }}};
}
if (!extensionsLength) return {{{ cDefs.EMSCRIPTEN_RESULT_INVALID_PARAM }}};
{{{ makeSetValue('extensionsLength', '0', 'lengthBytesUTF8(socket.extensions)+1', 'i32') }}};
return {{{ cDefs.EMSCRIPTEN_RESULT_SUCCESS }}};
},
emscripten_websocket_get_protocol__deps: ['$WS', '$stringToUTF8'],
emscripten_websocket_get_protocol__proxy: 'sync',
emscripten_websocket_get_protocol: (socketId, protocol, protocolLength) => {
var socket = WS.getSocket(socketId);
if (!socket) {
#if WEBSOCKET_DEBUG
dbg(`emscripten_websocket_get_protocol(): Invalid socket ID ${socketId} specified!`);
#endif
return {{{ cDefs.EMSCRIPTEN_RESULT_INVALID_TARGET }}};
}
if (!protocol) return {{{ cDefs.EMSCRIPTEN_RESULT_INVALID_PARAM }}};
stringToUTF8(socket.protocol, protocol, protocolLength);
return {{{ cDefs.EMSCRIPTEN_RESULT_SUCCESS }}};
},
emscripten_websocket_get_protocol_length__deps: ['$WS'],
emscripten_websocket_get_protocol_length__proxy: 'sync',
emscripten_websocket_get_protocol_length: (socketId, protocolLength) => {
var socket = WS.getSocket(socketId);
if (!socket) {
#if WEBSOCKET_DEBUG
dbg(`emscripten_websocket_get_protocol_length(): Invalid socket ID ${socketId} specified!`);
#endif
return {{{ cDefs.EMSCRIPTEN_RESULT_INVALID_TARGET }}};
}
if (!protocolLength) return {{{ cDefs.EMSCRIPTEN_RESULT_INVALID_PARAM }}};
{{{ makeSetValue('protocolLength', '0', 'lengthBytesUTF8(socket.protocol)+1', 'i32') }}};
return {{{ cDefs.EMSCRIPTEN_RESULT_SUCCESS }}};
},
emscripten_websocket_get_url__deps: ['$WS', '$stringToUTF8'],
emscripten_websocket_get_url__proxy: 'sync',
emscripten_websocket_get_url: (socketId, url, urlLength) => {
var socket = WS.getSocket(socketId);
if (!socket) {
#if WEBSOCKET_DEBUG
dbg(`emscripten_websocket_get_url(): Invalid socket ID ${socketId} specified!`);
#endif
return {{{ cDefs.EMSCRIPTEN_RESULT_INVALID_TARGET }}};
}
if (!url) return {{{ cDefs.EMSCRIPTEN_RESULT_INVALID_PARAM }}};
stringToUTF8(socket.url, url, urlLength);
return {{{ cDefs.EMSCRIPTEN_RESULT_SUCCESS }}};
},
emscripten_websocket_get_url_length__deps: ['$WS'],
emscripten_websocket_get_url_length__proxy: 'sync',
emscripten_websocket_get_url_length: (socketId, urlLength) => {
var socket = WS.getSocket(socketId);
if (!socket) {
#if WEBSOCKET_DEBUG
dbg(`emscripten_websocket_get_url_length(): Invalid socket ID ${socketId} specified!`);
#endif
return {{{ cDefs.EMSCRIPTEN_RESULT_INVALID_TARGET }}};
}
if (!urlLength) return {{{ cDefs.EMSCRIPTEN_RESULT_INVALID_PARAM }}};
{{{ makeSetValue('urlLength', '0', 'lengthBytesUTF8(socket.url)+1', 'i32') }}};
return {{{ cDefs.EMSCRIPTEN_RESULT_SUCCESS }}};
},
emscripten_websocket_set_onopen_callback_on_thread__deps: ['$WS'],
emscripten_websocket_set_onopen_callback_on_thread__proxy: 'sync',
emscripten_websocket_set_onopen_callback_on_thread: (socketId, userData, callbackFunc, thread) => {
// TODO:
// if (thread == {{{ cDefs.EM_CALLBACK_THREAD_CONTEXT_CALLING_THREAD }}} ||
// (thread == _pthread_self()) return emscripten_websocket_set_onopen_callback_on_calling_thread(socketId, userData, callbackFunc);
var socket = WS.getSocket(socketId);
if (!socket) {
#if WEBSOCKET_DEBUG
dbg(`emscripten_websocket_set_onopen_callback(): Invalid socket ID ${socketId} specified!`);
#endif
return {{{ cDefs.EMSCRIPTEN_RESULT_INVALID_TARGET }}};
}
#if WEBSOCKET_DEBUG
dbg(`emscripten_websocket_set_onopen_callback(socketId=${socketId},userData=${userData},callbackFunc='+callbackFunc})`);
#endif
socket.onopen = function(e) {
#if WEBSOCKET_DEBUG
dbg(`websocket event "open": socketId=${socketId},userData=${userData},callbackFunc=${callbackFunc})`);
#endif
var eventPtr = WS.getSocketEvent(socketId);
{{{ makeDynCall('iipp', 'callbackFunc') }}}(0/*TODO*/, eventPtr, userData);
}
return {{{ cDefs.EMSCRIPTEN_RESULT_SUCCESS }}};
},
emscripten_websocket_set_onerror_callback_on_thread__deps: ['$WS'],
emscripten_websocket_set_onerror_callback_on_thread__proxy: 'sync',
emscripten_websocket_set_onerror_callback_on_thread: (socketId, userData, callbackFunc, thread) => {
var socket = WS.getSocket(socketId);
if (!socket) {
#if WEBSOCKET_DEBUG
dbg(`emscripten_websocket_set_onerror_callback(): Invalid socket ID ${socketId} specified!`);
#endif
return {{{ cDefs.EMSCRIPTEN_RESULT_INVALID_TARGET }}};
}
#if WEBSOCKET_DEBUG
dbg(`emscripten_websocket_set_onerror_callback(socketId=${socketId},userData=${userData},callbackFunc=${callbackFunc})`);
#endif
socket.onerror = function(e) {
#if WEBSOCKET_DEBUG
dbg(`websocket event "error": socketId=${socketId},userData=${userData},callbackFunc=${callbackFunc})`);
#endif
var eventPtr = WS.getSocketEvent(socketId);
{{{ makeDynCall('iipp', 'callbackFunc') }}}(0/*TODO*/, eventPtr, userData);
}
return {{{ cDefs.EMSCRIPTEN_RESULT_SUCCESS }}};
},
emscripten_websocket_set_onclose_callback_on_thread__deps: ['$WS', '$stringToUTF8'],
emscripten_websocket_set_onclose_callback_on_thread__proxy: 'sync',
emscripten_websocket_set_onclose_callback_on_thread: (socketId, userData, callbackFunc, thread) => {
var socket = WS.getSocket(socketId);
if (!socket) {
#if WEBSOCKET_DEBUG
dbg(`emscripten_websocket_set_onclose_callback(): Invalid socket ID ${socketId} specified!`);
#endif
return {{{ cDefs.EMSCRIPTEN_RESULT_INVALID_TARGET }}};
}
#if WEBSOCKET_DEBUG
dbg(`emscripten_websocket_set_onclose_callback(socketId=${socketId},userData=${userData},callbackFunc=${callbackFunc})`);
#endif
socket.onclose = function(e) {
#if WEBSOCKET_DEBUG
dbg(`websocket event "close": socketId=${socketId},userData=${userData},callbackFunc=${callbackFunc})`);
#endif
var eventPtr = WS.getSocketEvent(socketId);
{{{ makeSetValue('eventPtr', C_STRUCTS.EmscriptenWebSocketCloseEvent.wasClean, 'e.wasClean', 'i8') }}},
{{{ makeSetValue('eventPtr', C_STRUCTS.EmscriptenWebSocketCloseEvent.code, 'e.code', 'i16') }}},
stringToUTF8(e.reason, eventPtr + {{{ C_STRUCTS.EmscriptenWebSocketCloseEvent.reason }}}, 512);
{{{ makeDynCall('iipp', 'callbackFunc') }}}(0/*TODO*/, eventPtr, userData);
}
return {{{ cDefs.EMSCRIPTEN_RESULT_SUCCESS }}};
},
emscripten_websocket_set_onmessage_callback_on_thread__deps: ['$WS', '$stringToNewUTF8', 'malloc', 'free'],
emscripten_websocket_set_onmessage_callback_on_thread__proxy: 'sync',
emscripten_websocket_set_onmessage_callback_on_thread: (socketId, userData, callbackFunc, thread) => {
var socket = WS.getSocket(socketId);
if (!socket) {
#if WEBSOCKET_DEBUG
dbg(`emscripten_websocket_set_onmessage_callback(): Invalid socket ID ${socketId} specified!`);
#endif
return {{{ cDefs.EMSCRIPTEN_RESULT_INVALID_TARGET }}};
}
#if WEBSOCKET_DEBUG
dbg(`emscripten_websocket_set_onmessage_callback(socketId=${socketId},userData=${userData},callbackFunc=${callbackFunc})`);
#endif
socket.onmessage = function(e) {
#if WEBSOCKET_DEBUG == 2
dbg(`websocket event "message": socketId=${socketId},userData=${userData},callbackFunc=${callbackFunc})`);
#endif
var isText = typeof e.data == 'string';
if (isText) {
var buf = stringToNewUTF8(e.data);
var len = lengthBytesUTF8(e.data)+1;
#if WEBSOCKET_DEBUG
var s = (e.data.length < 256) ? e.data : (e.data.slice(0, 256) + ` (${e.data.length-256} more characters)`);
dbg(`WebSocket onmessage, received data: "${e.data}", ${e.data.length} chars, ${len} bytes encoded as UTF-8: "${s}"`);
#endif
} else {
var len = e.data.byteLength;
var buf = _malloc(len);
HEAP8.set(new Uint8Array(e.data), buf);
#if WEBSOCKET_DEBUG
var s = `WebSocket onmessage, received data: ${len} bytes of binary:`;
for (var i = 0; i < Math.min(len, 256); ++i) s += ' ' + HEAPU8[buf+i].toString(16);
s += ', "';
for (var i = 0; i < Math.min(len, 256); ++i) s += (HEAPU8[buf+i] >= 32 && HEAPU8[buf+i] <= 127) ? String.fromCharCode(HEAPU8[buf+i]) : '\uFFFD';
s += '"';
if (len > 256) s + ` ... (${len - 256} more bytes)`;
dbg(s);
#endif
}
var eventPtr = WS.getSocketEvent(socketId);
{{{ makeSetValue('eventPtr', C_STRUCTS.EmscriptenWebSocketMessageEvent.data, 'buf', '*') }}},
{{{ makeSetValue('eventPtr', C_STRUCTS.EmscriptenWebSocketMessageEvent.numBytes, 'len', 'i32') }}},
{{{ makeSetValue('eventPtr', C_STRUCTS.EmscriptenWebSocketMessageEvent.isText, 'isText', 'i8') }}},
{{{ makeDynCall('iipp', 'callbackFunc') }}}(0/*TODO*/, eventPtr, userData);
_free(buf);
}
return {{{ cDefs.EMSCRIPTEN_RESULT_SUCCESS }}};
},
emscripten_websocket_new__proxy: 'sync',
emscripten_websocket_new: (createAttributes) => {
if (typeof WebSocket == 'undefined') {
#if WEBSOCKET_DEBUG
dbg('emscripten_websocket_new(): WebSocket API is not supported by current browser)');
#endif
return {{{ cDefs.EMSCRIPTEN_RESULT_NOT_SUPPORTED }}};
}
if (!createAttributes) {
#if WEBSOCKET_DEBUG
dbg('emscripten_websocket_new(): Missing required "createAttributes" function parameter!');
#endif
return {{{ cDefs.EMSCRIPTEN_RESULT_INVALID_PARAM }}};
}
var url = UTF8ToString({{{ makeGetValue('createAttributes', 0, '*') }}});
var protocols = {{{ makeGetValue('createAttributes', C_STRUCTS.EmscriptenWebSocketCreateAttributes.protocols, '*') }}}
// TODO: Add support for createOnMainThread==false; currently all WebSocket connections are created on the main thread.
// var createOnMainThread = HEAP8[createAttributes+2];
var socket = protocols ? new WebSocket(url, UTF8ToString(protocols).split(',')) : new WebSocket(url);
// We always marshal received WebSocket data back to Wasm, so enable receiving the data as arraybuffers for easy marshalling.
socket.binaryType = 'arraybuffer';
// TODO: While strictly not necessary, this ID would be good to be unique across all threads to avoid confusion.
var socketId = webSockets.allocate(socket);
#if WEBSOCKET_DEBUG
dbg(`emscripten_websocket_new(url=${url}, protocols=${protocols ? UTF8ToString(protocols).split(',') : 'null'}): created socket ID ${socketId})`);
#endif
return socketId;
},
emscripten_websocket_send_utf8_text__deps: ['$WS'],
emscripten_websocket_send_utf8_text__proxy: 'sync',
emscripten_websocket_send_utf8_text: (socketId, textData) => {
var socket = WS.getSocket(socketId);
if (!socket) {
#if WEBSOCKET_DEBUG
dbg(`emscripten_websocket_send_utf8_text(): Invalid socket ID ${socketId} specified!`);
#endif
return {{{ cDefs.EMSCRIPTEN_RESULT_INVALID_TARGET }}};
}
var str = UTF8ToString(textData);
#if WEBSOCKET_DEBUG == 2
dbg(`emscripten_websocket_send_utf8_text(socketId=${socketId},textData=${str.length} chars, "${str}")`);
#else
#if WEBSOCKET_DEBUG
dbg(`emscripten_websocket_send_utf8_text(socketId=${socketId},textData=${str.length} ' chars, "${(str.length > 8) ? (str.substring(0,8) + '...') : str}")`);
#endif
#endif
socket.send(str);
return {{{ cDefs.EMSCRIPTEN_RESULT_SUCCESS }}};
},
emscripten_websocket_send_binary__deps: ['$WS'],
emscripten_websocket_send_binary__proxy: 'sync',
emscripten_websocket_send_binary: (socketId, binaryData, dataLength) => {
var socket = WS.getSocket(socketId);
if (!socket) {
#if WEBSOCKET_DEBUG
dbg(`emscripten_websocket_send_binary(): Invalid socket ID ${socketId} specified!`);
#endif
return {{{ cDefs.EMSCRIPTEN_RESULT_INVALID_TARGET }}};
}
#if WEBSOCKET_DEBUG
var s = `data: ${dataLength} bytes of binary:`;
for (var i = 0; i < Math.min(dataLength, 256); ++i) s += ' '+ HEAPU8[binaryData+i].toString(16);
s += ', "';
for (var i = 0; i < Math.min(dataLength, 256); ++i) s += (HEAPU8[binaryData+i] >= 32 && HEAPU8[binaryData+i] <= 127) ? String.fromCharCode(HEAPU8[binaryData+i]) : '\uFFFD';
s += '"';
if (dataLength > 256) s + ` ... (${dataLength - 256} more bytes)`;
dbg(`emscripten_websocket_send_binary(socketId=${socketId},binaryData=${binaryData},dataLength=${dataLength}), ${s}`);
#endif
#if SHARED_MEMORY
// TODO: This is temporary to cast a shared Uint8Array to a non-shared Uint8Array. This could be removed if WebSocket API is improved
// to allow passing in views to SharedArrayBuffers
socket.send(new Uint8Array({{{ makeHEAPView('U8', 'binaryData', 'binaryData+dataLength') }}}));
#else
socket.send({{{ makeHEAPView('U8', 'binaryData', 'binaryData+dataLength') }}});
#endif
return {{{ cDefs.EMSCRIPTEN_RESULT_SUCCESS }}};
},
emscripten_websocket_close__deps: ['$WS'],
emscripten_websocket_close__proxy: 'sync',
emscripten_websocket_close: (socketId, code, reason) => {
var socket = WS.getSocket(socketId);
if (!socket) {
#if WEBSOCKET_DEBUG
dbg(`emscripten_websocket_close(): Invalid socket ID ${socketId} specified!`);
#endif
return {{{ cDefs.EMSCRIPTEN_RESULT_INVALID_TARGET }}};
}
var reasonStr = reason ? UTF8ToString(reason) : undefined;
#if WEBSOCKET_DEBUG
dbg(`emscripten_websocket_close(socketId=${socketId},code=${code},reason=${reasonStr})`);
#endif
// According to WebSocket specification, only close codes that are recognized have integer values
// 1000-4999, with 3000-3999 and 4000-4999 denoting user-specified close codes:
// https://developer.mozilla.org/en-US/docs/Web/API/CloseEvent#Status_codes
// Therefore be careful to call the .close() function with exact number and types of parameters.
// Coerce code==0 to undefined, since Wasm->JS call can only marshal integers, and 0 is not allowed.
if (reason) socket.close(code || undefined, UTF8ToString(reason));
else if (code) socket.close(code);
else socket.close();
return {{{ cDefs.EMSCRIPTEN_RESULT_SUCCESS }}};
},
emscripten_websocket_delete__deps: ['$WS'],
emscripten_websocket_delete__proxy: 'sync',
emscripten_websocket_delete: (socketId) => {
var socket = WS.getSocket(socketId);
if (!socket) {
#if WEBSOCKET_DEBUG
dbg(`emscripten_websocket_delete(): Invalid socket ID ${socketId} specified!`);
#endif
return {{{ cDefs.EMSCRIPTEN_RESULT_INVALID_TARGET }}};
}
#if WEBSOCKET_DEBUG
dbg(`emscripten_websocket_delete(socketId=${socketId})`);
#endif
socket.onopen = socket.onerror = socket.onclose = socket.onmessage = null;
webSockets.free(socketId);
return {{{ cDefs.EMSCRIPTEN_RESULT_SUCCESS }}};
},
emscripten_websocket_is_supported__proxy: 'sync',
emscripten_websocket_is_supported: () => typeof WebSocket != 'undefined',
emscripten_websocket_deinitialize__deps: ['$WS'],
emscripten_websocket_deinitialize__proxy: 'sync',
emscripten_websocket_deinitialize__deps: ['emscripten_websocket_delete'],
emscripten_websocket_deinitialize: () => {
#if WEBSOCKET_DEBUG
dbg('emscripten_websocket_deinitialize()');
#endif
for (var i in WS.sockets) {
var socket = WS.sockets[i];
if (socket) {
socket.close();
_emscripten_websocket_delete(i);
}
}
WS.sockets = [];
}
}
addToLibrary(LibraryWebSocket);