2023-07-31 07:52:04 +00:00
|
|
|
|
|
|
|
{.pragma: exported, exportc, cdecl, raises: [].}
|
|
|
|
{.pragma: callback, cdecl, raises: [], gcsafe.}
|
|
|
|
{.passc: "-fPIC".}
|
|
|
|
|
|
|
|
import
|
|
|
|
std/[json,sequtils,times,strformat,options,atomics,strutils,os]
|
|
|
|
import
|
|
|
|
chronicles,
|
|
|
|
chronos,
|
|
|
|
stew/results,
|
|
|
|
stew/shims/net
|
|
|
|
import
|
2023-08-09 17:11:50 +00:00
|
|
|
../../../waku/node/waku_node,
|
2023-07-31 07:52:04 +00:00
|
|
|
../events/[json_error_event,json_message_event,json_base_event],
|
|
|
|
./inter_thread_communication/request
|
|
|
|
|
|
|
|
type
|
|
|
|
Context* = object
|
|
|
|
thread: Thread[(ptr Context)]
|
|
|
|
reqChannel: Channel[InterThreadRequest]
|
|
|
|
respChannel: Channel[Result[string, string]]
|
|
|
|
node: WakuNode
|
|
|
|
|
|
|
|
var ctx {.threadvar.}: ptr Context
|
|
|
|
|
|
|
|
# To control when the thread is running
|
|
|
|
var running: Atomic[bool]
|
|
|
|
|
|
|
|
# Every Nim library must have this function called - the name is derived from
|
|
|
|
# the `--nimMainPrefix` command line option
|
|
|
|
proc NimMain() {.importc.}
|
|
|
|
var initialized: Atomic[bool]
|
|
|
|
|
|
|
|
proc waku_init() =
|
|
|
|
if not initialized.exchange(true):
|
|
|
|
NimMain() # Every Nim library needs to call `NimMain` once exactly
|
|
|
|
when declared(setupForeignThreadGc): setupForeignThreadGc()
|
|
|
|
when declared(nimGC_setStackBottom):
|
|
|
|
var locals {.volatile, noinit.}: pointer
|
|
|
|
locals = addr(locals)
|
|
|
|
nimGC_setStackBottom(locals)
|
|
|
|
|
|
|
|
proc run(ctx: ptr Context) {.thread.} =
|
|
|
|
## This is the worker thread body. This thread runs the Waku node
|
|
|
|
## and attends library user requests (stop, connect_to, etc.)
|
|
|
|
|
2023-09-01 06:37:02 +00:00
|
|
|
var node: WakuNode
|
|
|
|
|
2023-07-31 07:52:04 +00:00
|
|
|
while running.load == true:
|
|
|
|
## Trying to get a request from the libwaku main thread
|
|
|
|
let req = ctx.reqChannel.tryRecv()
|
|
|
|
if req[0] == true:
|
2023-09-01 06:37:02 +00:00
|
|
|
let response = waitFor req[1].process(addr node)
|
2023-07-31 07:52:04 +00:00
|
|
|
ctx.respChannel.send( response )
|
|
|
|
|
|
|
|
poll()
|
|
|
|
|
|
|
|
tearDownForeignThreadGc()
|
|
|
|
|
2023-09-01 06:37:02 +00:00
|
|
|
proc createWakuThread*(): Result[void, string] =
|
2023-07-31 07:52:04 +00:00
|
|
|
## This proc is called from the main thread and it creates
|
|
|
|
## the Waku working thread.
|
|
|
|
|
|
|
|
waku_init()
|
|
|
|
|
|
|
|
ctx = createShared(Context, 1)
|
|
|
|
ctx.reqChannel.open()
|
|
|
|
ctx.respChannel.open()
|
|
|
|
|
|
|
|
running.store(true)
|
|
|
|
|
|
|
|
try:
|
|
|
|
createThread(ctx.thread, run, ctx)
|
2023-09-01 06:37:02 +00:00
|
|
|
except ValueError, ResourceExhaustedError:
|
2023-07-31 07:52:04 +00:00
|
|
|
# and freeShared for typed allocations!
|
|
|
|
freeShared(ctx)
|
|
|
|
|
2023-09-01 06:37:02 +00:00
|
|
|
return err("failed to create the Waku thread: " & getCurrentExceptionMsg())
|
2023-07-31 07:52:04 +00:00
|
|
|
|
|
|
|
return ok()
|
|
|
|
|
|
|
|
proc stopWakuNodeThread*() =
|
|
|
|
running.store(false)
|
|
|
|
joinThread(ctx.thread)
|
|
|
|
|
|
|
|
ctx.reqChannel.close()
|
|
|
|
ctx.respChannel.close()
|
|
|
|
|
|
|
|
freeShared(ctx)
|
|
|
|
|
|
|
|
proc sendRequestToWakuThread*(req: InterThreadRequest): Result[string, string] =
|
|
|
|
|
|
|
|
ctx.reqChannel.send(req)
|
|
|
|
|
|
|
|
var resp = ctx.respChannel.tryRecv()
|
|
|
|
while resp[0] == false:
|
|
|
|
resp = ctx.respChannel.tryRecv()
|
|
|
|
os.sleep(1)
|
|
|
|
|
|
|
|
return resp[1]
|
|
|
|
|