#ifndef SRC_ASYNC_WRAP_INL_H_ #define SRC_ASYNC_WRAP_INL_H_ #include "async-wrap.h" #include "base-object.h" #include "base-object-inl.h" #include "env.h" #include "env-inl.h" #include "node_internals.h" #include "util.h" #include "util-inl.h" #include "v8.h" namespace node { inline AsyncWrap::AsyncWrap(Environment* env, v8::Local object, ProviderType provider, AsyncWrap* parent) : BaseObject(env, object), bits_(static_cast(provider) << 1), uid_(env->get_async_wrap_uid()) { CHECK_NE(provider, PROVIDER_NONE); CHECK_GE(object->InternalFieldCount(), 1); // Shift provider value over to prevent id collision. persistent().SetWrapperClassId(NODE_ASYNC_ID_OFFSET + provider); v8::Local init_fn = env->async_hooks_init_function(); // No init callback exists, no reason to go on. if (init_fn.IsEmpty()) return; // If async wrap callbacks are disabled and no parent was passed that has // run the init callback then return. if (!env->async_wrap_callbacks_enabled() && (parent == nullptr || !parent->ran_init_callback())) return; v8::HandleScope scope(env->isolate()); v8::Local argv[] = { v8::Integer::New(env->isolate(), get_uid()), v8::Int32::New(env->isolate(), provider), Null(env->isolate()), Null(env->isolate()) }; if (parent != nullptr) { argv[2] = v8::Integer::New(env->isolate(), parent->get_uid()); argv[3] = parent->object(); } v8::TryCatch try_catch(env->isolate()); v8::MaybeLocal ret = init_fn->Call(env->context(), object, arraysize(argv), argv); if (ret.IsEmpty()) { ClearFatalExceptionHandlers(env); FatalException(env->isolate(), try_catch); } bits_ |= 1; // ran_init_callback() is true now. } inline AsyncWrap::~AsyncWrap() { if (!ran_init_callback()) return; v8::Local fn = env()->async_hooks_destroy_function(); if (!fn.IsEmpty()) { v8::HandleScope scope(env()->isolate()); v8::Local uid = v8::Integer::New(env()->isolate(), get_uid()); v8::TryCatch try_catch(env()->isolate()); v8::MaybeLocal ret = fn->Call(env()->context(), v8::Null(env()->isolate()), 1, &uid); if (ret.IsEmpty()) { ClearFatalExceptionHandlers(env()); FatalException(env()->isolate(), try_catch); } } } inline bool AsyncWrap::ran_init_callback() const { return static_cast(bits_ & 1); } inline AsyncWrap::ProviderType AsyncWrap::provider_type() const { return static_cast(bits_ >> 1); } inline int64_t AsyncWrap::get_uid() const { return uid_; } inline v8::Local AsyncWrap::MakeCallback( const v8::Local symbol, int argc, v8::Local* argv) { v8::Local cb_v = object()->Get(symbol); CHECK(cb_v->IsFunction()); return MakeCallback(cb_v.As(), argc, argv); } inline v8::Local AsyncWrap::MakeCallback( uint32_t index, int argc, v8::Local* argv) { v8::Local cb_v = object()->Get(index); CHECK(cb_v->IsFunction()); return MakeCallback(cb_v.As(), argc, argv); } } // namespace node #endif // SRC_ASYNC_WRAP_INL_H_