{.pragma: exported, exportc, cdecl, raises: [].} {.pragma: callback, cdecl, raises: [], gcsafe.} {.passc: "-fPIC".} when defined(linux): {.passl: "-Wl,-soname,libsds.so".} import std/[locks, typetraits, tables, atomics] # Added tables import chronos import results import ./sds_thread/sds_thread, ./alloc, ./ffi_types, ./sds_thread/inter_thread_communication/sds_thread_request, ./sds_thread/inter_thread_communication/requests/ [sds_lifecycle_request, sds_message_request], ../src/[reliability, reliability_utils, message] ################################################################################ ### Wrapper around the reliability manager ################################################################################ ################################################################################ ### Not-exported components template checkLibsdsParams*( ctx: ptr SdsContext, callback: SdsCallBack, userData: pointer ) = ctx[].userData = userData if isNil(callback): return RET_MISSING_CALLBACK template callEventCallback(ctx: ptr SdsContext, eventName: string, body: untyped) = if isNil(ctx[].eventCallback): error eventName & " - eventCallback is nil" return if isNil(ctx[].eventUserData): error eventName & " - eventUserData is nil" return foreignThreadGc: try: let event = body cast[SdsCallBack](ctx[].eventCallback)( RET_OK, unsafeAddr event[0], cast[csize_t](len(event)), ctx[].eventUserData ) except Exception, CatchableError: let msg = "Exception " & eventName & " when calling 'eventCallBack': " & getCurrentExceptionMsg() cast[SdsCallBack](ctx[].eventCallback)( RET_ERR, unsafeAddr msg[0], cast[csize_t](len(msg)), ctx[].eventUserData ) proc handleRequest( ctx: ptr SdsContext, requestType: RequestType, content: pointer, callback: SdsCallBack, userData: pointer, ): cint = sds_thread.sendRequestToSdsThread(ctx, requestType, content, callback, userData).isOkOr: let msg = "libsds error: " & $error callback(RET_ERR, unsafeAddr msg[0], cast[csize_t](len(msg)), userData) return RET_ERR return RET_OK ### End of not-exported components ################################################################################ ################################################################################ ### Library setup # Every Nim library must have this function called - the name is derived from # the `--nimMainPrefix` command line option proc libsdsNimMain() {.importc.} # To control when the library has been initialized var initialized: Atomic[bool] if defined(android): # Redirect chronicles to Android System logs when compiles(defaultChroniclesStream.outputs[0].writer): defaultChroniclesStream.outputs[0].writer = proc( logLevel: LogLevel, msg: LogOutputStr ) {.raises: [].} = echo logLevel, msg proc initializeLibrary() {.exported.} = if not initialized.exchange(true): ## Every Nim library needs to call `NimMain` once exactly, to initialize the Nim runtime. ## Being `` the value given in the optional compilation flag --nimMainPrefix:yourprefix libsdsNimMain() when declared(setupForeignThreadGc): setupForeignThreadGc() when declared(nimGC_setStackBottom): var locals {.volatile, noinit.}: pointer locals = addr(locals) nimGC_setStackBottom(locals) ### End of library setup ################################################################################ ################################################################################ ### Exported procs proc NewReliabilityManager( channelId: cstring, callback: SdsCallBack, userData: pointer ): pointer {.dynlib, exportc, cdecl.} = initializeLibrary() ## Creates a new instance of the Reliability Manager. if isNil(callback): echo "error: missing callback in NewReliabilityManager" return nil ## Create the SDS thread that will keep waiting for req from the main thread. var ctx = sds_thread.createSdsThread().valueOr: let msg = "Error in createSdsThread: " & $error callback(RET_ERR, unsafeAddr msg[0], cast[csize_t](len(msg)), userData) return nil ctx.userData = userData let retCode = handleRequest( ctx, RequestType.LIFECYCLE, SdsLifecycleRequest.createShared( SdsLifecycleMsgType.CREATE_RELIABILITY_MANAGER, channelId ), callback, userData, ) if retCode == RET_ERR: return nil return ctx proc SetEventCallback( ctx: ptr SdsContext, callback: SdsCallBack, userData: pointer ) {.dynlib, exportc.} = initializeLibrary() ctx[].eventCallback = cast[pointer](callback) ctx[].eventUserData = userData proc CleanupReliabilityManager( ctx: ptr SdsContext, callback: SdsCallBack, userData: pointer ): cint {.dynlib, exportc.} = initializeLibrary() checkLibsdsParams(ctx, callback, userData) sds_thread.destroySdsThread(ctx).isOkOr: let msg = "libsds error: " & $error callback(RET_ERR, unsafeAddr msg[0], cast[csize_t](len(msg)), userData) return RET_ERR ## always need to invoke the callback although we don't retrieve value to the caller callback(RET_OK, nil, 0, userData) return RET_OK proc ResetReliabilityManager( ctx: ptr SdsContext, callback: SdsCallBack, userData: pointer ): cint {.dynlib, exportc.} = checkLibsdsParams(ctx, callback, userData) handleRequest( ctx, RequestType.LIFECYCLE, SdsLifecycleRequest.createShared(SdsLifecycleMsgType.RESET_RELIABILITY_MANAGER), callback, userData, ) proc WrapOutgoingMessage( ctx: ptr SdsContext, message: pointer, messageLen: csize_t, messageId: cstring, callback: SdsCallBack, userData: pointer, ): cint {.dynlib, exportc.} = initializeLibrary() checkLibsdsParams(ctx, callback, userData) if message == nil and messageLen > 0: let msg = "libsds error: " & "message pointer is NULL but length > 0" callback(RET_ERR, unsafeAddr msg[0], cast[csize_t](len(msg)), userData) return RET_ERR if messageId == nil: let msg = "libsds error: " & "message ID pointer is NULL" callback(RET_ERR, unsafeAddr msg[0], cast[csize_t](len(msg)), userData) return RET_ERR var msg = allocSharedSeqFromCArray(cast[ptr byte](message), messageLen.int) let msgId = messageId.alloc() defer: deallocSharedSeq(msg) deallocShared(msgId) handleRequest( ctx, RequestType.MESSAGE, SdsMessageRequest.createShared( SdsMessageMsgType.WRAP_MESSAGE, msg, messageLen, msgId ), callback, userData, ) proc UnwrapReceivedMessage( ctx: ptr SdsContext, message: pointer, messageLen: csize_t, callback: SdsCallBack, userData: pointer, ): cint {.dynlib, exportc.} = initializeLibrary() checkLibsdsParams(ctx, callback, userData) if message == nil and messageLen > 0: let msg = "libsds error: " & "message pointer is NULL but length > 0" callback(RET_ERR, unsafeAddr msg[0], cast[csize_t](len(msg)), userData) return RET_ERR var msg = allocSharedSeqFromCArray(cast[ptr byte](message), messageLen.int) defer: deallocSharedSeq(msg) handleRequest( ctx, RequestType.MESSAGE, SdsMessageRequest.createShared(SdsMessageMsgType.UNWRAP_MESSAGE, msg, messageLen), callback, userData, ) ### End of exported procs ################################################################################