mirror of
https://gitlab.com/freepascal.org/fpc/pas2js.git
synced 2025-04-05 11:17:45 +02:00
* Async websocket
This commit is contained in:
parent
0e60c9464f
commit
6bba60ec8a
@ -25,18 +25,18 @@ uses
|
||||
SysUtils, js, wasienv, web,weborworker, wasm.websocket.shared;
|
||||
|
||||
Type
|
||||
TWasmWebSocketAPI = Class;
|
||||
TWasmBaseWebSocketAPI = Class;
|
||||
|
||||
{ TWasmWebsocket }
|
||||
|
||||
TWasmWebsocket = class
|
||||
private
|
||||
FAPI : TWasmWebSocketAPI;
|
||||
FAPI : TWasmBaseWebSocketAPI;
|
||||
FWebsocketID : TWasmWebsocketID;
|
||||
FWS : TJSWebSocket;
|
||||
FUserData: TWasmPointer;
|
||||
Public
|
||||
Constructor Create(aAPI : TWasmWebSocketAPI; aID : TWasmWebsocketID; aUserData : TWasmPointer;const aURL : String; const aProtocols : String = ''); virtual;
|
||||
Constructor Create(aAPI : TWasmBaseWebSocketAPI; aID : TWasmWebsocketID; aUserData : TWasmPointer;const aURL : String; const aProtocols : String = ''); virtual;
|
||||
destructor Destroy; override;
|
||||
Procedure Close(aCode : Integer; aReason : String);
|
||||
procedure SendText(aData: String); virtual;
|
||||
@ -51,14 +51,15 @@ Type
|
||||
end;
|
||||
TWasmWebsocketClass = Class of TWasmWebsocket;
|
||||
|
||||
{ TWasmWebSocketAPI }
|
||||
{ TWasmBaseWebSocketAPI }
|
||||
TWasmWebSocketErrorHandler = Function (aWebsocketID : TWasmWebSocketID; aUserData : TWasmPointer) : TWebsocketCallBackResult;
|
||||
TWasmWebSocketMessageHandler = Function (aWebsocketID : TWasmWebSocketID; aUserData : TWasmPointer; aMessageType : TWasmWebSocketMessageType; aMessage : TWasmPointer; aMessageLen : Integer) : TWebsocketCallBackResult;
|
||||
TWasmWebSocketOpenHandler = Function (aWebsocketID : TWasmWebSocketID; aUserData : TWasmPointer) : TWebsocketCallBackResult;
|
||||
TWasmWebSocketCloseHandler = Function (aWebsocketID : TWasmWebSocketID; aUserData : TWasmPointer; aCode: Longint; aReason : PByte; aReasonLen : Longint; aClean : Longint) : TWebsocketCallBackResult;
|
||||
TWasmWebsocketAllocateBuffer = Function (aWebsocketID : TWasmWebSocketID; aUserData : TWasmPointer; aBufferLen : Longint) : TWasmPointer;
|
||||
|
||||
TWasmWebSocketAPI = class(TImportExtension)
|
||||
|
||||
TWasmBaseWebSocketAPI = class(TImportExtension)
|
||||
FNextID : TWasmWebsocketID;
|
||||
FSockets : TJSObject;
|
||||
FEncoder : TJSTextEncoder;
|
||||
@ -75,16 +76,19 @@ Type
|
||||
Function GetNextID : TWasmWebsocketID;
|
||||
Function GetWebsocket(aID : TWasmWebSocketID) : TWasmWebSocket;
|
||||
function GetWebSocketClass: TWasmWebsocketClass; virtual;
|
||||
function FreeWebSocket(aID: TWasmWebSocketID) : boolean;
|
||||
Procedure HandleOpen(aSocket : TWasmWebSocket);
|
||||
Procedure HandleClose(aSocket : TWasmWebSocket; aCode : Integer; aReason : String; aWasClean : Boolean);
|
||||
Procedure HandleError(aSocket : TWasmWebSocket);
|
||||
Procedure HandleBinaryMessage(aSocket : TWasmWebSocket; aMessage : TJSArrayBuffer);
|
||||
Procedure HandleStringMessage(aSocket : TWasmWebSocket; aMessage : String);
|
||||
function WebsocketAllocate(aURL : PByte; aUrlLen : Longint; aProtocols : PByte; aProtocolLen : Longint; aUserData : TWasmPointer; aWebsocketID : PWasmWebSocketID) : TWasmWebsocketResult; virtual;
|
||||
function WebsocketDeAllocate(aWebsocketID : TWasmWebSocketID) : TWasmWebsocketResult; virtual;
|
||||
function WebsocketClose(aWebsocketID : TWasmWebSocketID; aCode : Longint; aReason : PByte; aReasonLen : Longint) : TWasmWebsocketResult; virtual;
|
||||
function WebsocketSend(aWebsocketID : TWasmWebSocketID; aData : PByte; aDataLen : Longint; aType : Longint) : TWasmWebsocketResult; virtual;
|
||||
public
|
||||
function WebsocketAllocate(aURL : PByte; aUrlLen : Longint; aProtocols : PByte; aProtocolLen : Longint; aUserData : TWasmPointer; aWebsocketID : PWasmWebSocketID) : TWasmWebsocketResult; virtual; abstract;
|
||||
function WebsocketDeAllocate(aWebsocketID : TWasmWebSocketID) : TWasmWebsocketResult; virtual; abstract;
|
||||
function WebsocketClose(aWebsocketID : TWasmWebSocketID; aCode : Longint; aReason : PByte; aReasonLen : Longint) : TWasmWebsocketResult; virtual; abstract;
|
||||
function WebsocketSend(aWebsocketID : TWasmWebSocketID; aData : PByte; aDataLen : Longint; aType : Longint) : TWasmWebsocketResult; virtual; abstract;
|
||||
property Encoder : TJSTextEncoder Read FEncoder;
|
||||
property Decoder : TJSTextDecoder Read FDecoder;
|
||||
public
|
||||
constructor Create(aEnv: TPas2JSWASIEnvironment); override;
|
||||
procedure FillImportObject(aObject: TJSObject); override;
|
||||
function AllocateBuffer(aSocket: TWasmWebSocket; aLen: Longint): TWasmPointer;
|
||||
@ -92,15 +96,51 @@ Type
|
||||
property LogAPICalls : Boolean Read FLogAPICalls Write FLogAPICalls;
|
||||
end;
|
||||
|
||||
{ TWasmWebSocketAPI }
|
||||
// This API handles everything locally.
|
||||
// When using this, the javascript must be able to handle the main event loop,
|
||||
// Meaning that the websockets
|
||||
TWasmWebSocketAPI = class(TWasmBaseWebSocketAPI)
|
||||
private
|
||||
Protected
|
||||
function CreateWebSocket(aID: Integer; aUserData: TWasmPointer; aUrl, aProtocols: string): TWasmWebSocket;
|
||||
function WebsocketAllocate(aURL : PByte; aUrlLen : Longint; aProtocols : PByte; aProtocolLen : Longint; aUserData : TWasmPointer; aWebsocketID : PWasmWebSocketID) : TWasmWebsocketResult; override;
|
||||
function WebsocketDeAllocate(aWebsocketID : TWasmWebSocketID) : TWasmWebsocketResult; override;
|
||||
function WebsocketClose(aWebsocketID : TWasmWebSocketID; aCode : Longint; aReason : PByte; aReasonLen : Longint) : TWasmWebsocketResult; override;
|
||||
function WebsocketSend(aWebsocketID : TWasmWebSocketID; aData : PByte; aDataLen : Longint; aType : Longint) : TWasmWebsocketResult; override;
|
||||
end;
|
||||
|
||||
{ TWorkerWebSocketAPI }
|
||||
|
||||
TWorkerWebSocketAPI = class(TWasmBaseWebSocketAPI)
|
||||
private
|
||||
FSharedMem: TJSSharedArrayBuffer;
|
||||
FArray : TJSUint8Array;
|
||||
FView : TJSDataView;
|
||||
procedure SetSharedMem(AValue: TJSSharedArrayBuffer);
|
||||
protected
|
||||
function AwaitResult: TWasmWebsocketResult;
|
||||
Public
|
||||
function LockMem : boolean;
|
||||
procedure UnlockMem;
|
||||
function WebsocketAllocate(aURL : PByte; aUrlLen : Longint; aProtocols : PByte; aProtocolLen : Longint; aUserData : TWasmPointer; aWebsocketID : PWasmWebSocketID) : TWasmWebsocketResult; override;
|
||||
function WebsocketDeAllocate(aWebsocketID : TWasmWebSocketID) : TWasmWebsocketResult; override;
|
||||
function WebsocketClose(aWebsocketID : TWasmWebSocketID; aCode : Longint; aReason : PByte; aReasonLen : Longint) : TWasmWebsocketResult; override;
|
||||
function WebsocketSend(aWebsocketID : TWasmWebSocketID; aData : PByte; aDataLen : Longint; aType : Longint) : TWasmWebsocketResult; override;
|
||||
property SharedMem : TJSSharedArrayBuffer Read FSharedMem Write SetSharedMem;
|
||||
end;
|
||||
|
||||
|
||||
|
||||
implementation
|
||||
|
||||
{ ---------------------------------------------------------------------
|
||||
TWasmWebSocketAPI
|
||||
TWasmBaseWebSocketAPI
|
||||
---------------------------------------------------------------------}
|
||||
|
||||
// Auxiliary calls
|
||||
|
||||
procedure TWasmWebSocketAPI.LogCall(const Msg: String);
|
||||
procedure TWasmBaseWebSocketAPI.LogCall(const Msg: String);
|
||||
begin
|
||||
{$IFNDEF NOLOGAPICALLS}
|
||||
If not LogAPICalls then exit;
|
||||
@ -109,7 +149,7 @@ begin
|
||||
end;
|
||||
|
||||
|
||||
procedure TWasmWebSocketAPI.LogCall(const Fmt: String; const Args: array of const);
|
||||
procedure TWasmBaseWebSocketAPI.LogCall(const Fmt: String; const Args: array of const);
|
||||
|
||||
begin
|
||||
{$IFNDEF NOLOGAPICALLS}
|
||||
@ -119,26 +159,26 @@ begin
|
||||
end;
|
||||
|
||||
|
||||
function TWasmWebSocketAPI.GetNextID: TWasmWebsocketID;
|
||||
function TWasmBaseWebSocketAPI.GetNextID: TWasmWebsocketID;
|
||||
begin
|
||||
Inc(FNextID);
|
||||
Result:=FNextID;
|
||||
end;
|
||||
|
||||
|
||||
procedure TWasmWebSocketAPI.DoError(const Msg: String);
|
||||
procedure TWasmBaseWebSocketAPI.DoError(const Msg: String);
|
||||
begin
|
||||
Console.Error(Msg);
|
||||
end;
|
||||
|
||||
|
||||
procedure TWasmWebSocketAPI.DoError(const Fmt: String; const Args: array of const);
|
||||
procedure TWasmBaseWebSocketAPI.DoError(const Fmt: String; const Args: array of const);
|
||||
begin
|
||||
Console.Error(Format(Fmt,Args));
|
||||
end;
|
||||
|
||||
|
||||
function TWasmWebSocketAPI.GetWebsocket(aID: TWasmWebSocketID): TWasmWebSocket;
|
||||
function TWasmBaseWebSocketAPI.GetWebsocket(aID: TWasmWebSocketID): TWasmWebSocket;
|
||||
|
||||
var
|
||||
Value : JSValue;
|
||||
@ -152,13 +192,28 @@ begin
|
||||
end;
|
||||
|
||||
|
||||
function TWasmWebSocketAPI.GetWebSocketClass: TWasmWebsocketClass;
|
||||
function TWasmBaseWebSocketAPI.GetWebSocketClass: TWasmWebsocketClass;
|
||||
begin
|
||||
Result:=TWasmWebsocket;
|
||||
end;
|
||||
|
||||
function TWasmBaseWebSocketAPI.FreeWebSocket(aID: TWasmWebSocketID): boolean;
|
||||
|
||||
function TWasmWebSocketAPI.CheckCallbackRes(Res : TWebsocketCallBackResult; const aOperation : string) : Boolean;
|
||||
var
|
||||
lSocket : TWasmWebsocket;
|
||||
|
||||
begin
|
||||
lSocket:=GetWebsocket(aID);
|
||||
Result:=lSocket<>Nil;
|
||||
if Result then
|
||||
begin
|
||||
lSocket.Destroy;
|
||||
FSockets[IntToStr(aID)]:=undefined;
|
||||
end;
|
||||
end;
|
||||
|
||||
|
||||
function TWasmBaseWebSocketAPI.CheckCallbackRes(Res : TWebsocketCallBackResult; const aOperation : string) : Boolean;
|
||||
begin
|
||||
Result:=(Res=WASMWS_CALLBACK_SUCCESS);
|
||||
if not Result then
|
||||
@ -168,7 +223,7 @@ end;
|
||||
|
||||
// Callbacks for TWasmWebSocket, calls exported routines from webassembly module.
|
||||
|
||||
function TWasmWebSocketAPI.AllocateBuffer(aSocket: TWasmWebSocket; aLen : Longint) : TWasmPointer;
|
||||
function TWasmBaseWebSocketAPI.AllocateBuffer(aSocket: TWasmWebSocket; aLen : Longint) : TWasmPointer;
|
||||
|
||||
var
|
||||
aValue : JSValue;
|
||||
@ -184,7 +239,7 @@ begin
|
||||
end;
|
||||
|
||||
|
||||
procedure TWasmWebSocketAPI.HandleOpen(aSocket: TWasmWebSocket);
|
||||
procedure TWasmBaseWebSocketAPI.HandleOpen(aSocket: TWasmWebSocket);
|
||||
|
||||
var
|
||||
value : JSValue;
|
||||
@ -203,7 +258,7 @@ begin
|
||||
end;
|
||||
|
||||
|
||||
procedure TWasmWebSocketAPI.HandleClose(aSocket: TWasmWebSocket; aCode: Integer; aReason: String; aWasClean: Boolean);
|
||||
procedure TWasmBaseWebSocketAPI.HandleClose(aSocket: TWasmWebSocket; aCode: Integer; aReason: String; aWasClean: Boolean);
|
||||
|
||||
var
|
||||
aValue : JSValue;
|
||||
@ -250,7 +305,7 @@ begin
|
||||
end;
|
||||
|
||||
|
||||
procedure TWasmWebSocketAPI.HandleError(aSocket: TWasmWebSocket);
|
||||
procedure TWasmBaseWebSocketAPI.HandleError(aSocket: TWasmWebSocket);
|
||||
var
|
||||
Callback : JSValue;
|
||||
|
||||
@ -262,7 +317,7 @@ begin
|
||||
end;
|
||||
|
||||
|
||||
procedure TWasmWebSocketAPI.HandleSendMessage(aSocket: TWasmWebSocket; aMessage: TJSUInt8Array; aType : TWasmWebSocketMessageType);
|
||||
procedure TWasmBaseWebSocketAPI.HandleSendMessage(aSocket: TWasmWebSocket; aMessage: TJSUInt8Array; aType : TWasmWebSocketMessageType);
|
||||
|
||||
//begin
|
||||
// TWasmWebSocketMessageHandler = Function (aWebsocketID : TWasmWebSocketID; aUserData : Pointer; aMessageType : TWasmWebSocketMessageType; aMessage : Pointer; aMessageLen : Integer) : TWebsocketCallBackResult;
|
||||
@ -299,7 +354,7 @@ begin
|
||||
end;
|
||||
|
||||
|
||||
procedure TWasmWebSocketAPI.HandleBinaryMessage(aSocket: TWasmWebSocket; aMessage: TJSArrayBuffer);
|
||||
procedure TWasmBaseWebSocketAPI.HandleBinaryMessage(aSocket: TWasmWebSocket; aMessage: TJSArrayBuffer);
|
||||
|
||||
var
|
||||
lMessage : TJSUint8array;
|
||||
@ -310,7 +365,7 @@ begin
|
||||
end;
|
||||
|
||||
|
||||
procedure TWasmWebSocketAPI.HandleStringMessage(aSocket: TWasmWebSocket; aMessage: String);
|
||||
procedure TWasmBaseWebSocketAPI.HandleStringMessage(aSocket: TWasmWebSocket; aMessage: String);
|
||||
var
|
||||
lMessage : TJSUint8array;
|
||||
|
||||
@ -320,6 +375,12 @@ begin
|
||||
end;
|
||||
|
||||
// API methods called from within webassembly
|
||||
function TWasmWebSocketAPI.CreateWebSocket(aID : Integer; aUserData : TWasmPointer; aUrl,aProtocols : string) :TWasmWebSocket;
|
||||
|
||||
begin
|
||||
Result:=GetWebSocketClass.Create(Self,aID,aUserData,aURL,aProtocols);
|
||||
FSockets[IntToStr(aID)]:=Result;
|
||||
end;
|
||||
|
||||
function TWasmWebSocketAPI.WebsocketAllocate(aURL: PByte; aUrlLen: Longint; aProtocols: PByte; aProtocolLen: Longint;
|
||||
aUserData: TWasmPointer; aWebsocketID: PWasmWebSocketID): TWasmWebsocketResult;
|
||||
@ -339,10 +400,15 @@ begin
|
||||
if (lUrl='') then
|
||||
Exit(WASMWS_RESULT_NO_URL);
|
||||
lID:=GetNextID;
|
||||
lSocket:=GetWebSocketClass.Create(Self,lID,aUserData,lURL,lProtocols);
|
||||
FSockets[IntToStr(lID)]:=lSocket;
|
||||
env.SetMemInfoInt32(aWebSocketID,lID);
|
||||
Result:=WASMWS_RESULT_SUCCESS;
|
||||
lSocket:=CreateWebSocket(lID,aUserData,lURL,lProtocols);
|
||||
if Assigned(lSocket) then
|
||||
begin
|
||||
env.SetMemInfoInt32(aWebSocketID,lID);
|
||||
Result:=WASMWS_RESULT_SUCCESS;
|
||||
end
|
||||
else
|
||||
Result:=WASMWS_RESULT_ERROR;
|
||||
|
||||
{$IFNDEF NOLOGAPICALLS}
|
||||
If LogAPICalls then
|
||||
LogCall('HTTP.WebSocketAllocate("%s","%s",%d,[%x]) => %d',[lURL,lProtocols,aUserData,aWebSocketID,lID]);
|
||||
@ -352,20 +418,15 @@ end;
|
||||
|
||||
function TWasmWebSocketAPI.WebsocketDeAllocate(aWebsocketID: TWasmWebSocketID): TWasmWebsocketResult;
|
||||
|
||||
var
|
||||
lSocket : TWasmWebSocket;
|
||||
|
||||
begin
|
||||
{$IFNDEF NOLOGAPICALLS}
|
||||
If LogAPICalls then
|
||||
LogCall('HTTP.WebSocketDeAllocate(%d)',[aWebSocketID]);
|
||||
{$ENDIF}
|
||||
lSocket:=GetWebsocket(aWebSocketID);
|
||||
if lSocket=Nil then
|
||||
Exit(WASMWS_RESULT_INVALIDID);
|
||||
lSocket.Destroy;
|
||||
FSockets[IntToStr(aWebSocketID)]:=undefined;
|
||||
Result:=WASMWS_RESULT_SUCCESS;
|
||||
if FreeWebSocket(aWebSocketID) then
|
||||
Result:=WASMWS_RESULT_SUCCESS
|
||||
else
|
||||
Result:=WASMWS_RESULT_INVALIDID;
|
||||
end;
|
||||
|
||||
|
||||
@ -415,8 +476,167 @@ begin
|
||||
Result:=WASMWS_RESULT_SUCCESS;
|
||||
end;
|
||||
|
||||
{ TWorkerWebSocketAPI }
|
||||
|
||||
constructor TWasmWebSocketAPI.Create(aEnv: TPas2JSWASIEnvironment);
|
||||
procedure TWorkerWebSocketAPI.SetSharedMem(AValue: TJSSharedArrayBuffer);
|
||||
begin
|
||||
if FSharedMem=AValue then Exit;
|
||||
FSharedMem:=AValue;
|
||||
if Assigned(aValue) then
|
||||
begin
|
||||
FArray:=TJSUint8Array.New(FSharedMem);
|
||||
FView:=TJSDataView.New(FSharedMem);
|
||||
end
|
||||
else
|
||||
begin
|
||||
FArray:=Nil;
|
||||
FView:=Nil;
|
||||
end
|
||||
end;
|
||||
|
||||
function TWorkerWebSocketAPI.LockMem: boolean;
|
||||
|
||||
|
||||
begin
|
||||
// Wait while it is set.
|
||||
Result:=Assigned(FView);
|
||||
if Result then
|
||||
TJSAtomics.wait(FArray,WASM_SHMSG_SEMAPHORE,WASM_SEM_SET);
|
||||
// Now, when here we definitely have value WASM_SEM_NOT_SET
|
||||
end;
|
||||
|
||||
function TWorkerWebSocketAPI.AwaitResult : TWasmWebsocketResult;
|
||||
|
||||
var
|
||||
S : String;
|
||||
|
||||
begin
|
||||
if not Assigned(FView) then
|
||||
Result:=WASMWS_RESULT_FAILEDLOCK
|
||||
else
|
||||
begin
|
||||
S:=TJSAtomics.wait(FArray,WASM_SHMSG_SEMAPHORE,WASM_SEM_SET);
|
||||
if s='ok' then
|
||||
Result:=TJSAtomics.load(FArray,WASM_SHMSG_RESULT) // get a result
|
||||
else // no result
|
||||
Result:=WASMWS_RESULT_FAILEDLOCK;
|
||||
end;
|
||||
end;
|
||||
|
||||
|
||||
procedure TWorkerWebSocketAPI.UnlockMem;
|
||||
begin
|
||||
// Set and notify.
|
||||
if not Assigned(FView) then
|
||||
exit;
|
||||
TJSAtomics.store(FArray, WASM_SHMSG_SEMAPHORE, WASM_SEM_SET);
|
||||
TJSAtomics.notify(FArray, WASM_SHMSG_SEMAPHORE, 1);
|
||||
end;
|
||||
|
||||
function TWorkerWebSocketAPI.WebsocketAllocate(aURL: PByte; aUrlLen: Longint; aProtocols: PByte; aProtocolLen: Longint;
|
||||
aUserData: TWasmPointer; aWebsocketID: PWasmWebSocketID): TWasmWebsocketResult;
|
||||
|
||||
|
||||
var
|
||||
lID : TWasmWebsocketID;
|
||||
lTmp : TJSUint8Array;
|
||||
lProtocolOffset : Longint;
|
||||
|
||||
begin
|
||||
lID:=GetNextID;
|
||||
if (aURLLen+aProtocolLen)>(FArray.byteLength-WASM_SHMSG_FIXED_LEN) then
|
||||
Exit(WASMWS_RESULT_INVALIDSIZE);
|
||||
if (aURLLen<=0) then
|
||||
Exit(WASMWS_RESULT_INVALIDSIZE);
|
||||
if (aProtocolLen<0) then
|
||||
Exit(WASMWS_RESULT_INVALIDSIZE);
|
||||
if Not LockMem then
|
||||
Exit(WASMWS_RESULT_FAILEDLOCK);
|
||||
try
|
||||
FView.setInt32(WASM_SHMSG_WEBSOCKETID,lID,Env.IsLittleEndian);
|
||||
FView.setInt8(WASM_SHMSG_OPERATION,WASM_WSOPERATION_CREATE);
|
||||
FView.setInt32(WASM_SHMSG_CREATE_USERDATA,aUserData,Env.IsLittleEndian);
|
||||
FView.setInt32(WASM_SHMSG_CREATE_URL_LENGTH,aUrlLen,Env.IsLittleEndian);
|
||||
FView.setInt32(WASM_SHMSG_CREATE_PROTOCOL_LENGTH,aProtocolLen,Env.IsLittleEndian);
|
||||
// Write URL to shared buffer (it may no longer exist when the message is treated)
|
||||
lTmp:=TJSUInt8Array.New(FSharedMem,aURL,aUrlLen);
|
||||
FArray._set(lTmp,WASM_SHMSG_CREATE_URL_DATA);
|
||||
// Write protocols if they are present.
|
||||
if aProtocolLen>0 then
|
||||
begin
|
||||
lTmp:=TJSUInt8Array.New(FSharedMem,aProtocols,aProtocolLen);
|
||||
lProtocolOffset:=WASM_SHMSG_CREATE_PROTOCOL_DATA_OFFSET+aURLLen;
|
||||
FArray._set(lTmp,lProtocolOffset);
|
||||
end;
|
||||
// Result:=AwaitResult;
|
||||
finally
|
||||
UnlockMem;
|
||||
end;
|
||||
getModuleMemoryDataView.setInt32(aWebsocketID,lID);
|
||||
Result:=WASMWS_RESULT_SUCCESS;
|
||||
end;
|
||||
|
||||
function TWorkerWebSocketAPI.WebsocketDeAllocate(aWebsocketID: TWasmWebSocketID): TWasmWebsocketResult;
|
||||
|
||||
begin
|
||||
if Not LockMem then
|
||||
Exit(WASMWS_RESULT_FAILEDLOCK);
|
||||
try
|
||||
FView.setInt32(WASM_SHMSG_WEBSOCKETID,aWebsocketID,Env.IsLittleEndian);
|
||||
FView.setInt8(WASM_SHMSG_OPERATION,WASM_WSOPERATION_FREE);
|
||||
// Result:=AwaitResult;
|
||||
finally
|
||||
UnlockMem;
|
||||
end;
|
||||
Result:=WASMWS_RESULT_SUCCESS;
|
||||
end;
|
||||
|
||||
function TWorkerWebSocketAPI.WebsocketClose(aWebsocketID: TWasmWebSocketID; aCode: Longint; aReason: PByte; aReasonLen: Longint): TWasmWebsocketResult;
|
||||
|
||||
var
|
||||
lTmp : TJSUint8Array;
|
||||
|
||||
begin
|
||||
if Not LockMem then
|
||||
Exit(WASMWS_RESULT_FAILEDLOCK);
|
||||
try
|
||||
FView.setInt32(WASM_SHMSG_WEBSOCKETID,aWebsocketID,Env.IsLittleEndian);
|
||||
FView.setInt8(WASM_SHMSG_OPERATION,WASM_WSOPERATION_CLOSE);
|
||||
FView.setInt32(WASM_SHMSG_CLOSE_CODE,aCode,Env.IsLittleEndian);
|
||||
FView.setInt32(WASM_SHMSG_CLOSE_REASON_LENGTH,aReasonLen,Env.IsLittleEndian);
|
||||
if aReasonLen>0 then
|
||||
begin
|
||||
lTmp:=TJSUInt8Array.New(FSharedMem,aReason,aReasonLen);
|
||||
FArray._set(lTmp,WASM_SHMSG_CLOSE_REASON_DATA);
|
||||
end;
|
||||
// Result:=AwaitResult;
|
||||
finally
|
||||
UnlockMem;
|
||||
end;
|
||||
Result:=WASMWS_RESULT_SUCCESS;
|
||||
end;
|
||||
|
||||
|
||||
function TWorkerWebSocketAPI.WebsocketSend(aWebsocketID: TWasmWebSocketID; aData: PByte; aDataLen: Longint; aType: Longint
|
||||
): TWasmWebsocketResult;
|
||||
|
||||
begin
|
||||
if Not LockMem then
|
||||
Exit(WASMWS_RESULT_FAILEDLOCK);
|
||||
try
|
||||
FView.setInt32(WASM_SHMSG_WEBSOCKETID,aWebsocketID,Env.IsLittleEndian);
|
||||
FView.setInt8(WASM_SHMSG_OPERATION,WASM_WSOPERATION_SEND);
|
||||
FView.setInt32(WASM_SHMSG_SEND_DATA_LENGTH,aDataLen,Env.IsLittleEndian);
|
||||
FView.setInt32(WASM_SHMSG_SEND_DATA_TYPE,aType);
|
||||
FView.setInt32(WASM_SHMSG_SEND_DATA_ADDRESS,aData,Env.IsLittleEndian);
|
||||
// Result:=AwaitResult;
|
||||
finally
|
||||
UnlockMem;
|
||||
end;
|
||||
end;
|
||||
|
||||
|
||||
constructor TWasmBaseWebSocketAPI.Create(aEnv: TPas2JSWASIEnvironment);
|
||||
begin
|
||||
inherited Create(aEnv);
|
||||
FNextID:=0;
|
||||
@ -426,7 +646,7 @@ begin
|
||||
end;
|
||||
|
||||
|
||||
procedure TWasmWebSocketAPI.FillImportObject(aObject: TJSObject);
|
||||
procedure TWasmBaseWebSocketAPI.FillImportObject(aObject: TJSObject);
|
||||
begin
|
||||
aObject[websocketFN_Allocate]:=@WebsocketAllocate;
|
||||
aObject[websocketFN_DeAllocate]:=@WebsocketDeAllocate;
|
||||
@ -435,7 +655,7 @@ begin
|
||||
end;
|
||||
|
||||
|
||||
function TWasmWebSocketAPI.ImportName: String;
|
||||
function TWasmBaseWebSocketAPI.ImportName: String;
|
||||
begin
|
||||
Result:=websocketExportName;
|
||||
end;
|
||||
@ -499,7 +719,7 @@ begin
|
||||
end;
|
||||
|
||||
|
||||
constructor TWasmWebsocket.Create(aAPI: TWasmWebSocketAPI; aID: TWasmWebsocketID; aUserData : TWasmPointer;const aURL: String; const aProtocols: String);
|
||||
constructor TWasmWebsocket.Create(aAPI: TWasmBaseWebSocketAPI; aID: TWasmWebsocketID; aUserData : TWasmPointer;const aURL: String; const aProtocols: String);
|
||||
begin
|
||||
FAPI:=aAPI;
|
||||
FWebsocketID:=aID;
|
||||
|
@ -41,10 +41,12 @@ Type
|
||||
{$endif}
|
||||
|
||||
Const
|
||||
WASMWS_RESULT_SUCCESS = 0;
|
||||
WASMWS_RESULT_ERROR = -1;
|
||||
WASMWS_RESULT_NO_URL = -2;
|
||||
WASMWS_RESULT_INVALIDID = -3;
|
||||
WASMWS_RESULT_SUCCESS = 0;
|
||||
WASMWS_RESULT_ERROR = -1;
|
||||
WASMWS_RESULT_NO_URL = -2;
|
||||
WASMWS_RESULT_INVALIDID = -3;
|
||||
WASMWS_RESULT_FAILEDLOCK = -4;
|
||||
WASMWS_RESULT_INVALIDSIZE = -5;
|
||||
|
||||
WASMWS_CALLBACK_SUCCESS = 0;
|
||||
WASMWS_CALLBACK_NOHANDLER = -1;
|
||||
@ -60,6 +62,68 @@ const
|
||||
websocketFN_close = 'close_websocket';
|
||||
websocketFN_send = 'send_websocket';
|
||||
|
||||
const
|
||||
{
|
||||
Worker websockets use a dedicated worker to be able to handle callbacks.
|
||||
Communication with this worker happens through shared memory.
|
||||
|
||||
The shared memory is at least 1024 bytes large, and has the following layout:
|
||||
|
||||
Index 0 : Semaphore (4 bytes)
|
||||
Index 4 : ID of websocket (4 bytes)
|
||||
Index 8 : Operation (1 byte)
|
||||
0 : Create
|
||||
1 : Send
|
||||
2 : Close
|
||||
Index 9 : Unused
|
||||
Depending on operation:
|
||||
create:
|
||||
Index 10 : User data (4 bytes)
|
||||
Index 14 : Length of URL (4 bytes)
|
||||
Index 18 : Length of protocol (4 bytes)
|
||||
Index 22 : URL data (URL length bytes)
|
||||
Index 22+URL length : Protocol data (protocol length bytes)
|
||||
send:
|
||||
Index 10 : Length of data (4 bytes)
|
||||
Index 14 : Address of data (4 bytes)
|
||||
close:
|
||||
Index 10 : Close code (4 bytes)
|
||||
Index 14 : Reason length (4 bytes)
|
||||
Index 18 : Reason data (reason length bytes)
|
||||
note that this means that the length of URL+Protocol is limited to shared memory length minus 22 bytes.
|
||||
}
|
||||
|
||||
// Common
|
||||
WASM_SHMSG_SEMAPHORE = 0;
|
||||
WASM_SHMSG_RESULT = 4;
|
||||
WASM_SHMSG_WEBSOCKETID = 8;
|
||||
WASM_SHMSG_OPERATION = 12;
|
||||
// Create
|
||||
WASM_SHMSG_CREATE_USERDATA = 14;
|
||||
WASM_SHMSG_CREATE_URL_LENGTH = 18;
|
||||
WASM_SHMSG_CREATE_PROTOCOL_LENGTH = 22;
|
||||
WASM_SHMSG_CREATE_URL_DATA = 26;
|
||||
WASM_SHMSG_CREATE_PROTOCOL_DATA_OFFSET = WASM_SHMSG_CREATE_URL_DATA;
|
||||
// Send
|
||||
WASM_SHMSG_SEND_DATA_LENGTH = 14;
|
||||
WASM_SHMSG_SEND_DATA_TYPE = 18;
|
||||
WASM_SHMSG_SEND_DATA_ADDRESS = 22;
|
||||
// Close
|
||||
WASM_SHMSG_CLOSE_CODE = 14;
|
||||
WASM_SHMSG_CLOSE_REASON_LENGTH = 18;
|
||||
WASM_SHMSG_CLOSE_REASON_DATA = 22;
|
||||
|
||||
WASM_SEM_NOT_SET = 0;
|
||||
WASM_SEM_SET = 1;
|
||||
|
||||
// Operation (goes in WASM_SHMSG_OPERATION);
|
||||
WASM_WSOPERATION_NONE = 0;
|
||||
WASM_WSOPERATION_CREATE = 1;
|
||||
WASM_WSOPERATION_FREE = 2;
|
||||
WASM_WSOPERATION_SEND = 3;
|
||||
WASM_WSOPERATION_CLOSE = 4;
|
||||
|
||||
WASM_SHMSG_FIXED_LEN = WASM_SHMSG_CREATE_URL_DATA+4;
|
||||
|
||||
implementation
|
||||
|
||||
|
Loading…
Reference in New Issue
Block a user