Skip to content

Commit c3df7da

Browse files
committed
node: count callouts to JS to reemit beforeExit
1 parent 5d7262c commit c3df7da

File tree

5 files changed

+37
-10
lines changed

5 files changed

+37
-10
lines changed

src/async-wrap-inl.h

+1
Original file line numberDiff line numberDiff line change
@@ -40,6 +40,7 @@ inline AsyncWrap::AsyncWrap(Environment* env,
4040
v8::TryCatch try_catch;
4141

4242
v8::Local<v8::Value> n = v8::Int32::New(env->isolate(), provider);
43+
env->inc_callout_count();
4344
env->async_hooks_init_function()->Call(object, 1, &n);
4445

4546
if (try_catch.HasCaught())

src/async-wrap.cc

+6
Original file line numberDiff line numberDiff line change
@@ -185,6 +185,7 @@ Local<Value> AsyncWrap::MakeCallback(const Local<Function> cb,
185185
if (has_domain) {
186186
Local<Value> enter_v = domain->Get(env()->enter_string());
187187
if (enter_v->IsFunction()) {
188+
env()->inc_callout_count();
188189
enter_v.As<Function>()->Call(domain, 0, nullptr);
189190
if (try_catch.HasCaught())
190191
return Undefined(env()->isolate());
@@ -193,12 +194,14 @@ Local<Value> AsyncWrap::MakeCallback(const Local<Function> cb,
193194

194195
if (has_async_queue()) {
195196
try_catch.SetVerbose(false);
197+
env()->inc_callout_count();
196198
env()->async_hooks_pre_function()->Call(context, 0, nullptr);
197199
if (try_catch.HasCaught())
198200
FatalError("node::AsyncWrap::MakeCallback", "pre hook threw");
199201
try_catch.SetVerbose(true);
200202
}
201203

204+
env()->inc_callout_count();
202205
Local<Value> ret = cb->Call(context, argc, argv);
203206

204207
if (try_catch.HasCaught()) {
@@ -207,6 +210,7 @@ Local<Value> AsyncWrap::MakeCallback(const Local<Function> cb,
207210

208211
if (has_async_queue()) {
209212
try_catch.SetVerbose(false);
213+
env()->inc_callout_count();
210214
env()->async_hooks_post_function()->Call(context, 0, nullptr);
211215
if (try_catch.HasCaught())
212216
FatalError("node::AsyncWrap::MakeCallback", "post hook threw");
@@ -216,6 +220,7 @@ Local<Value> AsyncWrap::MakeCallback(const Local<Function> cb,
216220
if (has_domain) {
217221
Local<Value> exit_v = domain->Get(env()->exit_string());
218222
if (exit_v->IsFunction()) {
223+
env()->inc_callout_count();
219224
exit_v.As<Function>()->Call(domain, 0, nullptr);
220225
if (try_catch.HasCaught())
221226
return Undefined(env()->isolate());
@@ -239,6 +244,7 @@ Local<Value> AsyncWrap::MakeCallback(const Local<Function> cb,
239244

240245
tick_info->set_in_tick(true);
241246

247+
env()->inc_callout_count();
242248
env()->tick_callback_function()->Call(process, 0, nullptr);
243249

244250
tick_info->set_in_tick(false);

src/env-inl.h

+15
Original file line numberDiff line numberDiff line change
@@ -387,6 +387,21 @@ inline void Environment::set_http_parser_buffer(char* buffer) {
387387
http_parser_buffer_ = buffer;
388388
}
389389

390+
391+
inline unsigned int Environment::callout_count() const {
392+
return callout_count_;
393+
}
394+
395+
396+
inline void Environment::reset_callout_count() {
397+
callout_count_ = 0;
398+
}
399+
400+
401+
inline void Environment::inc_callout_count() {
402+
callout_count_++;
403+
}
404+
390405
inline Environment* Environment::from_cares_timer_handle(uv_timer_t* handle) {
391406
return ContainerOf(&Environment::cares_timer_handle_, handle);
392407
}

src/env.h

+5
Original file line numberDiff line numberDiff line change
@@ -452,6 +452,10 @@ class Environment {
452452
inline char* http_parser_buffer() const;
453453
inline void set_http_parser_buffer(char* buffer);
454454

455+
inline unsigned int callout_count() const;
456+
inline void reset_callout_count();
457+
inline void inc_callout_count();
458+
455459
inline void ThrowError(const char* errmsg);
456460
inline void ThrowTypeError(const char* errmsg);
457461
inline void ThrowRangeError(const char* errmsg);
@@ -551,6 +555,7 @@ class Environment {
551555
uint32_t* heap_statistics_buffer_ = nullptr;
552556

553557
char* http_parser_buffer_;
558+
unsigned int callout_count_ = 0;
554559

555560
#define V(PropertyName, TypeName) \
556561
v8::Persistent<TypeName> PropertyName ## _;

src/node.cc

+10-10
Original file line numberDiff line numberDiff line change
@@ -1015,6 +1015,7 @@ void PromiseRejectCallback(PromiseRejectMessage message) {
10151015
Local<Value> args[] = { event, promise, value };
10161016
Local<Object> process = env->process_object();
10171017

1018+
env->inc_callout_count();
10181019
callback->Call(process, ARRAY_SIZE(args), args);
10191020
}
10201021

@@ -1068,24 +1069,28 @@ Local<Value> MakeCallback(Environment* env,
10681069
if (has_domain) {
10691070
Local<Value> enter_v = domain->Get(env->enter_string());
10701071
if (enter_v->IsFunction()) {
1071-
enter_v.As<Function>()->Call(domain, 0, nullptr);
1072+
env->inc_callout_count();
1073+
enter_v.As<Function>()->Call(domain, 0, nullptr);
10721074
if (try_catch.HasCaught())
10731075
return Undefined(env->isolate());
10741076
}
10751077
}
10761078

10771079
if (has_async_queue) {
10781080
try_catch.SetVerbose(false);
1081+
env->inc_callout_count();
10791082
env->async_hooks_pre_function()->Call(object, 0, nullptr);
10801083
if (try_catch.HasCaught())
10811084
FatalError("node::MakeCallback", "pre hook threw");
10821085
try_catch.SetVerbose(true);
10831086
}
10841087

1088+
env->inc_callout_count();
10851089
Local<Value> ret = callback->Call(recv, argc, argv);
10861090

10871091
if (has_async_queue) {
10881092
try_catch.SetVerbose(false);
1093+
env->inc_callout_count();
10891094
env->async_hooks_post_function()->Call(object, 0, nullptr);
10901095
if (try_catch.HasCaught())
10911096
FatalError("node::MakeCallback", "post hook threw");
@@ -1095,6 +1100,7 @@ Local<Value> MakeCallback(Environment* env,
10951100
if (has_domain) {
10961101
Local<Value> exit_v = domain->Get(env->exit_string());
10971102
if (exit_v->IsFunction()) {
1103+
env->inc_callout_count();
10981104
exit_v.As<Function>()->Call(domain, 0, nullptr);
10991105
if (try_catch.HasCaught())
11001106
return Undefined(env->isolate());
@@ -3971,17 +3977,11 @@ static void StartNodeInstance(void* arg) {
39713977
v8::platform::PumpMessageLoop(default_platform, isolate);
39723978
EmitBeforeExit(env);
39733979

3974-
// If there are only unrefed handles left, we need to run an
3975-
// extra event loop turn to purge the unrefed handles.
3976-
if (event_loop->active_handles == 0 &&
3977-
event_loop->handle_queue[0] != event_loop->handle_queue[1])
3978-
uv_run(event_loop, UV_RUN_NOWAIT);
3979-
39803980
// Emit `beforeExit` if the loop became alive either after emitting
39813981
// event, or after running some callbacks.
3982-
more = uv_loop_alive(event_loop);
3983-
if (uv_run(event_loop, UV_RUN_NOWAIT) != 0)
3984-
more = true;
3982+
env->reset_callout_count();
3983+
uv_run(event_loop, UV_RUN_NOWAIT);
3984+
more = uv_loop_alive(event_loop) || env->callout_count() != 0;
39853985
}
39863986
} while (more == true);
39873987
}

0 commit comments

Comments
 (0)