rquickjs-core 0.11.0

High level bindings to the QuickJS JavaScript engine
Documentation
use core::ptr::NonNull;

use crate::qjs;

#[cfg(feature = "parallel")]
use std::sync::Arc;

/// Trait to specify how to drop a context once it goes out of scope.
/// Implemented on Runtime and AsyncRuntime.
pub(crate) trait DropContext: Clone {
    unsafe fn drop_context(&self, ctx: NonNull<qjs::JSContext>);
}

#[cfg(feature = "parallel")]
unsafe impl<R: Send + DropContext> Send for ContextOwner<R> {}

/// Struct in charge of dropping contexts when they go out of scope
pub(crate) struct ContextOwner<R: DropContext> {
    #[cfg(not(feature = "parallel"))]
    ctx: NonNull<qjs::JSContext>,
    #[cfg(feature = "parallel")]
    pub(crate) ctx: Arc<NonNull<qjs::JSContext>>,
    pub(crate) rt: R,
}

impl<R: DropContext> ContextOwner<R> {
    #[cfg(not(feature = "parallel"))]
    pub(crate) unsafe fn new(ctx: NonNull<qjs::JSContext>, rt: R) -> Self {
        Self { ctx, rt }
    }
    #[cfg(feature = "parallel")]
    pub(crate) unsafe fn new(ctx: NonNull<qjs::JSContext>, rt: R) -> Self {
        Self {
            ctx: Arc::new(ctx),
            rt,
        }
    }

    #[cfg(not(feature = "parallel"))]
    pub(crate) fn ctx(&self) -> NonNull<qjs::JSContext> {
        self.ctx
    }

    #[cfg(feature = "parallel")]
    pub(crate) fn ctx(&self) -> NonNull<qjs::JSContext> {
        *self.ctx
    }

    pub(crate) fn rt(&self) -> &R {
        &self.rt
    }
}

#[cfg(not(feature = "parallel"))]
impl<R: DropContext> Clone for ContextOwner<R> {
    fn clone(&self) -> Self {
        let ctx = unsafe { NonNull::new_unchecked(qjs::JS_DupContext(self.ctx.as_ptr())) };
        let rt = self.rt.clone();
        Self { ctx, rt }
    }
}

#[cfg(feature = "parallel")]
impl<R: DropContext> Clone for ContextOwner<R> {
    fn clone(&self) -> Self {
        Self {
            ctx: self.ctx.clone(),
            rt: self.rt.clone(),
        }
    }
}

#[cfg(not(feature = "parallel"))]
impl<R: DropContext> Drop for ContextOwner<R> {
    fn drop(&mut self) {
        unsafe { self.rt.drop_context(self.ctx()) }
    }
}

#[cfg(feature = "parallel")]
impl<R: DropContext> Drop for ContextOwner<R> {
    fn drop(&mut self) {
        if Arc::strong_count(&self.ctx) == 1 {
            unsafe { self.rt.drop_context(self.ctx()) }
        }
    }
}