%PDF- <> %âãÏÓ endobj 2 0 obj <> endobj 3 0 obj <>/ExtGState<>/ProcSet[/PDF/Text/ImageB/ImageC/ImageI] >>/Annots[ 28 0 R 29 0 R] /MediaBox[ 0 0 595.5 842.25] /Contents 4 0 R/Group<>/Tabs/S>> endobj ºaâÚÎΞ-ÌE1ÍØÄ÷{òò2ÿ ÛÖ^ÔÀá TÎ{¦?§®¥kuµùÕ5sLOšuY>endobj 2 0 obj<>endobj 2 0 obj<>endobj 2 0 obj<>endobj 2 0 obj<> endobj 2 0 obj<>endobj 2 0 obj<>es 3 0 R>> endobj 2 0 obj<> ox[ 0.000000 0.000000 609.600000 935.600000]/Fi endobj 3 0 obj<> endobj 7 1 obj<>/ProcSet[/PDF/Text/ImageB/ImageC/ImageI]>>/Subtype/Form>> stream
#ifndef SRC_JS_NATIVE_API_V8_H_ #define SRC_JS_NATIVE_API_V8_H_ #include "js_native_api_types.h" #include "js_native_api_v8_internals.h" static napi_status napi_clear_last_error(napi_env env); namespace v8impl { class RefTracker { public: RefTracker() {} virtual ~RefTracker() {} virtual void Finalize(bool isEnvTeardown) {} typedef RefTracker RefList; inline void Link(RefList* list) { prev_ = list; next_ = list->next_; if (next_ != nullptr) { next_->prev_ = this; } list->next_ = this; } inline void Unlink() { if (prev_ != nullptr) { prev_->next_ = next_; } if (next_ != nullptr) { next_->prev_ = prev_; } prev_ = nullptr; next_ = nullptr; } static void FinalizeAll(RefList* list) { while (list->next_ != nullptr) { list->next_->Finalize(true); } } private: RefList* next_ = nullptr; RefList* prev_ = nullptr; }; } // end of namespace v8impl struct napi_env__ { explicit napi_env__(v8::Local<v8::Context> context) : isolate(context->GetIsolate()), context_persistent(isolate, context) { CHECK_EQ(isolate, context->GetIsolate()); napi_clear_last_error(this); } inline v8::Local<v8::Context> context() const { return v8impl::PersistentToLocal::Strong(context_persistent); } inline void Ref() { refs++; } inline void Unref() { if (--refs == 0) DeleteMe(); } virtual bool can_call_into_js() const { return true; } virtual v8::Maybe<bool> mark_arraybuffer_as_untransferable( v8::Local<v8::ArrayBuffer> ab) const { return v8::Just(true); } static inline void HandleThrow(napi_env env, v8::Local<v8::Value> value) { env->isolate->ThrowException(value); } template <typename T, typename U = decltype(HandleThrow)> inline void CallIntoModule(T&& call, U&& handle_exception = HandleThrow) { int open_handle_scopes_before = open_handle_scopes; int open_callback_scopes_before = open_callback_scopes; napi_clear_last_error(this); call(this); CHECK_EQ(open_handle_scopes, open_handle_scopes_before); CHECK_EQ(open_callback_scopes, open_callback_scopes_before); if (!last_exception.IsEmpty()) { handle_exception(this, last_exception.Get(this->isolate)); last_exception.Reset(); } } // This should be overridden to schedule the finalization to a properiate // timing, like next tick of the event loop. virtual void CallFinalizer(napi_finalize cb, void* data, void* hint) { v8::HandleScope handle_scope(isolate); CallIntoModule([&](napi_env env) { cb(env, data, hint); }); } virtual void DeleteMe() { // First we must finalize those references that have `napi_finalizer` // callbacks. The reason is that addons might store other references which // they delete during their `napi_finalizer` callbacks. If we deleted such // references here first, they would be doubly deleted when the // `napi_finalizer` deleted them subsequently. v8impl::RefTracker::FinalizeAll(&finalizing_reflist); v8impl::RefTracker::FinalizeAll(&reflist); delete this; } v8::Isolate* const isolate; // Shortcut for context()->GetIsolate() v8impl::Persistent<v8::Context> context_persistent; v8impl::Persistent<v8::Value> last_exception; // We store references in two different lists, depending on whether they have // `napi_finalizer` callbacks, because we must first finalize the ones that // have such a callback. See `~napi_env__()` above for details. v8impl::RefTracker::RefList reflist; v8impl::RefTracker::RefList finalizing_reflist; napi_extended_error_info last_error; int open_handle_scopes = 0; int open_callback_scopes = 0; int refs = 1; void* instance_data = nullptr; protected: // Should not be deleted directly. Delete with `napi_env__::DeleteMe()` // instead. virtual ~napi_env__() = default; }; // This class is used to keep a napi_env live in a way that // is exception safe versus calling Ref/Unref directly class EnvRefHolder { public: explicit EnvRefHolder(napi_env env) : _env(env) { _env->Ref(); } explicit EnvRefHolder(const EnvRefHolder& other) : _env(other.env()) { _env->Ref(); } EnvRefHolder(EnvRefHolder&& other) { _env = other._env; other._env = nullptr; } ~EnvRefHolder() { if (_env != nullptr) { _env->Unref(); } } napi_env env(void) const { return _env; } private: napi_env _env; }; inline napi_status napi_clear_last_error(napi_env env) { env->last_error.error_code = napi_ok; // TODO(boingoing): Should this be a callback? env->last_error.engine_error_code = 0; env->last_error.engine_reserved = nullptr; env->last_error.error_message = nullptr; return napi_ok; } inline napi_status napi_set_last_error(napi_env env, napi_status error_code, uint32_t engine_error_code = 0, void* engine_reserved = nullptr) { env->last_error.error_code = error_code; env->last_error.engine_error_code = engine_error_code; env->last_error.engine_reserved = engine_reserved; return error_code; } #define RETURN_STATUS_IF_FALSE(env, condition, status) \ do { \ if (!(condition)) { \ return napi_set_last_error((env), (status)); \ } \ } while (0) #define RETURN_STATUS_IF_FALSE_WITH_PREAMBLE(env, condition, status) \ do { \ if (!(condition)) { \ return napi_set_last_error( \ (env), try_catch.HasCaught() ? napi_pending_exception : (status)); \ } \ } while (0) #define CHECK_ENV(env) \ do { \ if ((env) == nullptr) { \ return napi_invalid_arg; \ } \ } while (0) #define CHECK_ARG(env, arg) \ RETURN_STATUS_IF_FALSE((env), ((arg) != nullptr), napi_invalid_arg) #define CHECK_ARG_WITH_PREAMBLE(env, arg) \ RETURN_STATUS_IF_FALSE_WITH_PREAMBLE( \ (env), ((arg) != nullptr), napi_invalid_arg) #define CHECK_MAYBE_EMPTY(env, maybe, status) \ RETURN_STATUS_IF_FALSE((env), !((maybe).IsEmpty()), (status)) #define CHECK_MAYBE_EMPTY_WITH_PREAMBLE(env, maybe, status) \ RETURN_STATUS_IF_FALSE_WITH_PREAMBLE((env), !((maybe).IsEmpty()), (status)) // NAPI_PREAMBLE is not wrapped in do..while: try_catch must have function scope #define NAPI_PREAMBLE(env) \ CHECK_ENV((env)); \ RETURN_STATUS_IF_FALSE( \ (env), \ (env)->last_exception.IsEmpty() && (env)->can_call_into_js(), \ napi_pending_exception); \ napi_clear_last_error((env)); \ v8impl::TryCatch try_catch((env)) #define CHECK_TO_TYPE(env, type, context, result, src, status) \ do { \ CHECK_ARG((env), (src)); \ auto maybe = v8impl::V8LocalValueFromJsValue((src))->To##type((context)); \ CHECK_MAYBE_EMPTY((env), maybe, (status)); \ (result) = maybe.ToLocalChecked(); \ } while (0) #define CHECK_TO_TYPE_WITH_PREAMBLE(env, type, context, result, src, status) \ do { \ CHECK_ARG_WITH_PREAMBLE((env), (src)); \ auto maybe = v8impl::V8LocalValueFromJsValue((src))->To##type((context)); \ CHECK_MAYBE_EMPTY_WITH_PREAMBLE((env), maybe, (status)); \ (result) = maybe.ToLocalChecked(); \ } while (0) #define CHECK_TO_FUNCTION(env, result, src) \ do { \ CHECK_ARG((env), (src)); \ v8::Local<v8::Value> v8value = v8impl::V8LocalValueFromJsValue((src)); \ RETURN_STATUS_IF_FALSE((env), v8value->IsFunction(), napi_invalid_arg); \ (result) = v8value.As<v8::Function>(); \ } while (0) #define CHECK_TO_OBJECT(env, context, result, src) \ CHECK_TO_TYPE((env), Object, (context), (result), (src), napi_object_expected) #define CHECK_TO_OBJECT_WITH_PREAMBLE(env, context, result, src) \ CHECK_TO_TYPE_WITH_PREAMBLE( \ (env), Object, (context), (result), (src), napi_object_expected) #define CHECK_TO_STRING(env, context, result, src) \ CHECK_TO_TYPE((env), String, (context), (result), (src), napi_string_expected) #define GET_RETURN_STATUS(env) \ (!try_catch.HasCaught() \ ? napi_ok \ : napi_set_last_error((env), napi_pending_exception)) #define THROW_RANGE_ERROR_IF_FALSE(env, condition, error, message) \ do { \ if (!(condition)) { \ napi_throw_range_error((env), (error), (message)); \ return napi_set_last_error((env), napi_generic_failure); \ } \ } while (0) #define RETURN_STATUS_IF_FALSE_WITH_PREAMBLE(env, condition, status) \ do { \ if (!(condition)) { \ return napi_set_last_error( \ (env), try_catch.HasCaught() ? napi_pending_exception : (status)); \ } \ } while (0) #define CHECK_MAYBE_EMPTY_WITH_PREAMBLE(env, maybe, status) \ RETURN_STATUS_IF_FALSE_WITH_PREAMBLE((env), !((maybe).IsEmpty()), (status)) #define STATUS_CALL(call) \ do { \ napi_status status = (call); \ if (status != napi_ok) return status; \ } while (0) namespace v8impl { //=== Conversion between V8 Handles and napi_value ======================== // This asserts v8::Local<> will always be implemented with a single // pointer field so that we can pass it around as a void*. static_assert(sizeof(v8::Local<v8::Value>) == sizeof(napi_value), "Cannot convert between v8::Local<v8::Value> and napi_value"); inline napi_value JsValueFromV8LocalValue(v8::Local<v8::Value> local) { return reinterpret_cast<napi_value>(*local); } inline v8::Local<v8::Value> V8LocalValueFromJsValue(napi_value v) { v8::Local<v8::Value> local; memcpy(static_cast<void*>(&local), &v, sizeof(v)); return local; } // Adapter for napi_finalize callbacks. class Finalizer { public: // Some Finalizers are run during shutdown when the napi_env is destroyed, // and some need to keep an explicit reference to the napi_env because they // are run independently. enum EnvReferenceMode { kNoEnvReference, kKeepEnvReference }; protected: Finalizer(napi_env env, napi_finalize finalize_callback, void* finalize_data, void* finalize_hint, EnvReferenceMode refmode = kNoEnvReference) : _env(env), _finalize_callback(finalize_callback), _finalize_data(finalize_data), _finalize_hint(finalize_hint), _has_env_reference(refmode == kKeepEnvReference) { if (_has_env_reference) _env->Ref(); } ~Finalizer() { if (_has_env_reference) _env->Unref(); } public: static Finalizer* New(napi_env env, napi_finalize finalize_callback = nullptr, void* finalize_data = nullptr, void* finalize_hint = nullptr, EnvReferenceMode refmode = kNoEnvReference) { return new Finalizer( env, finalize_callback, finalize_data, finalize_hint, refmode); } static void Delete(Finalizer* finalizer) { delete finalizer; } protected: napi_env _env; napi_finalize _finalize_callback; void* _finalize_data; void* _finalize_hint; bool _finalize_ran = false; bool _has_env_reference = false; }; class TryCatch : public v8::TryCatch { public: explicit TryCatch(napi_env env) : v8::TryCatch(env->isolate), _env(env) {} ~TryCatch() { if (HasCaught()) { _env->last_exception.Reset(_env->isolate, Exception()); } } private: napi_env _env; }; // Wrapper around v8impl::Persistent that implements reference counting. class RefBase : protected Finalizer, RefTracker { protected: RefBase(napi_env env, uint32_t initial_refcount, bool delete_self, napi_finalize finalize_callback, void* finalize_data, void* finalize_hint); public: static RefBase* New(napi_env env, uint32_t initial_refcount, bool delete_self, napi_finalize finalize_callback, void* finalize_data, void* finalize_hint); static inline void Delete(RefBase* reference); virtual ~RefBase(); void* Data(); uint32_t Ref(); uint32_t Unref(); uint32_t RefCount(); protected: void Finalize(bool is_env_teardown = false) override; private: uint32_t _refcount; bool _delete_self; }; class Reference : public RefBase { using SecondPassCallParameterRef = Reference*; protected: template <typename... Args> Reference(napi_env env, v8::Local<v8::Value> value, Args&&... args); public: static Reference* New(napi_env env, v8::Local<v8::Value> value, uint32_t initial_refcount, bool delete_self, napi_finalize finalize_callback = nullptr, void* finalize_data = nullptr, void* finalize_hint = nullptr); virtual ~Reference(); uint32_t Ref(); uint32_t Unref(); v8::Local<v8::Value> Get(); protected: void Finalize(bool is_env_teardown = false) override; private: void ClearWeak(); void SetWeak(); static void FinalizeCallback( const v8::WeakCallbackInfo<SecondPassCallParameterRef>& data); static void SecondPassCallback( const v8::WeakCallbackInfo<SecondPassCallParameterRef>& data); v8impl::Persistent<v8::Value> _persistent; SecondPassCallParameterRef* _secondPassParameter; bool _secondPassScheduled; FRIEND_TEST(JsNativeApiV8Test, Reference); }; } // end of namespace v8impl #endif // SRC_JS_NATIVE_API_V8_H_