@@ -28,7 +28,6 @@ using v8::Maybe;
28
28
using v8::MaybeLocal;
29
29
using v8::Nothing;
30
30
using v8::Object;
31
- using v8::ObjectTemplate;
32
31
using v8::SharedArrayBuffer;
33
32
using v8::String;
34
33
using v8::Symbol;
@@ -169,6 +168,20 @@ uint32_t Message::AddWASMModule(WasmModuleObject::TransferrableModule&& mod) {
169
168
170
169
namespace {
171
170
171
+ MaybeLocal<Function> GetEmitMessageFunction (Local<Context> context) {
172
+ Isolate* isolate = context->GetIsolate ();
173
+ Local<Object> per_context_bindings;
174
+ Local<Value> emit_message_val;
175
+ if (!GetPerContextExports (context).ToLocal (&per_context_bindings) ||
176
+ !per_context_bindings->Get (context,
177
+ FIXED_ONE_BYTE_STRING (isolate, " emitMessage" ))
178
+ .ToLocal (&emit_message_val)) {
179
+ return MaybeLocal<Function>();
180
+ }
181
+ CHECK (emit_message_val->IsFunction ());
182
+ return emit_message_val.As <Function>();
183
+ }
184
+
172
185
MaybeLocal<Function> GetDOMException (Local<Context> context) {
173
186
Isolate* isolate = context->GetIsolate ();
174
187
Local<Object> per_context_bindings;
@@ -470,20 +483,31 @@ MessagePort::MessagePort(Environment* env,
470
483
MessagePort* channel = ContainerOf (&MessagePort::async_, handle);
471
484
channel->OnMessage ();
472
485
};
486
+
473
487
CHECK_EQ (uv_async_init (env->event_loop (),
474
488
&async_,
475
489
onmessage), 0 );
476
- async_.data = static_cast <void *>(this );
490
+ async_.data = nullptr ; // Reset later to indicate success of the constructor.
491
+ auto cleanup = OnScopeLeave ([&]() {
492
+ if (async_.data == nullptr ) Close ();
493
+ });
477
494
478
495
Local<Value> fn;
479
496
if (!wrap->Get (context, env->oninit_symbol ()).ToLocal (&fn))
480
497
return ;
481
498
482
499
if (fn->IsFunction ()) {
483
500
Local<Function> init = fn.As <Function>();
484
- USE (init->Call (context, wrap, 0 , nullptr ));
501
+ if (init->Call (context, wrap, 0 , nullptr ).IsEmpty ())
502
+ return ;
485
503
}
486
504
505
+ Local<Function> emit_message_fn;
506
+ if (!GetEmitMessageFunction (context).ToLocal (&emit_message_fn))
507
+ return ;
508
+ emit_message_fn_.Reset (env->isolate (), emit_message_fn);
509
+
510
+ async_.data = static_cast <void *>(this );
487
511
Debug (this , " Created message port" );
488
512
}
489
513
@@ -531,6 +555,11 @@ MessagePort* MessagePort::New(
531
555
return nullptr ;
532
556
MessagePort* port = new MessagePort (env, context, instance);
533
557
CHECK_NOT_NULL (port);
558
+ if (port->IsHandleClosing ()) {
559
+ // Construction failed with an exception.
560
+ return nullptr ;
561
+ }
562
+
534
563
if (data) {
535
564
port->Detach ();
536
565
port->data_ = std::move (data);
@@ -623,16 +652,8 @@ void MessagePort::OnMessage() {
623
652
continue ;
624
653
}
625
654
626
- Local<Object> event;
627
- Local<Value> cb_args[1 ];
628
- if (!env ()->message_event_object_template ()->NewInstance (context)
629
- .ToLocal (&event) ||
630
- event->Set (context, env ()->data_string (), payload).IsNothing () ||
631
- event->Set (context, env ()->target_string (), object ()).IsNothing () ||
632
- (cb_args[0 ] = event, false ) ||
633
- MakeCallback (env ()->onmessage_string (),
634
- arraysize (cb_args),
635
- cb_args).IsEmpty ()) {
655
+ Local<Function> emit_message = PersistentToLocal::Strong (emit_message_fn_);
656
+ if (MakeCallback (emit_message, 1 , &payload).IsEmpty ()) {
636
657
// Re-schedule OnMessage() execution in case of failure.
637
658
if (data_)
638
659
TriggerAsync ();
@@ -901,6 +922,7 @@ void MessagePort::Entangle(MessagePort* a, MessagePortData* b) {
901
922
902
923
void MessagePort::MemoryInfo (MemoryTracker* tracker) const {
903
924
tracker->TrackField (" data" , data_);
925
+ tracker->TrackField (" emit_message_fn" , emit_message_fn_);
904
926
}
905
927
906
928
Local<FunctionTemplate> GetMessagePortConstructorTemplate (Environment* env) {
@@ -910,8 +932,6 @@ Local<FunctionTemplate> GetMessagePortConstructorTemplate(Environment* env) {
910
932
if (!templ.IsEmpty ())
911
933
return templ;
912
934
913
- Isolate* isolate = env->isolate ();
914
-
915
935
{
916
936
Local<FunctionTemplate> m = env->NewFunctionTemplate (MessagePort::New);
917
937
m->SetClassName (env->message_port_constructor_string ());
@@ -922,13 +942,6 @@ Local<FunctionTemplate> GetMessagePortConstructorTemplate(Environment* env) {
922
942
env->SetProtoMethod (m, " start" , MessagePort::Start);
923
943
924
944
env->set_message_port_constructor_template (m);
925
-
926
- Local<FunctionTemplate> event_ctor = FunctionTemplate::New (isolate);
927
- event_ctor->SetClassName (FIXED_ONE_BYTE_STRING (isolate, " MessageEvent" ));
928
- Local<ObjectTemplate> e = event_ctor->InstanceTemplate ();
929
- e->Set (env->data_string (), Null (isolate));
930
- e->Set (env->target_string (), Null (isolate));
931
- env->set_message_event_object_template (e);
932
945
}
933
946
934
947
return GetMessagePortConstructorTemplate (env);
@@ -947,7 +960,13 @@ static void MessageChannel(const FunctionCallbackInfo<Value>& args) {
947
960
Context::Scope context_scope (context);
948
961
949
962
MessagePort* port1 = MessagePort::New (env, context);
963
+ if (port1 == nullptr ) return ;
950
964
MessagePort* port2 = MessagePort::New (env, context);
965
+ if (port2 == nullptr ) {
966
+ port1->Close ();
967
+ return ;
968
+ }
969
+
951
970
MessagePort::Entangle (port1, port2);
952
971
953
972
args.This ()->Set (context, env->port1_string (), port1->object ())
0 commit comments