Skip to content

Commit e919224

Browse files
committedFeb 12, 2016
src: add AsyncCallbackScope
Add a scope that will allow MakeCallback to know whether or not it's currently running. This will prevent nextTickQueue and the MicrotaskQueue from being processed recursively. It is also required to wrap the bootloading stage since it doesn't run within a MakeCallback. Ref: nodejs/node-v0.x-archive#9245 PR-URL: nodejs#4507 Reviewed-By: Fedor Indutny <[email protected]>
1 parent 575b84e commit e919224

7 files changed

+45
-17
lines changed
 

‎src/async-wrap.cc

+3-5
Original file line numberDiff line numberDiff line change
@@ -184,6 +184,8 @@ Local<Value> AsyncWrap::MakeCallback(const Local<Function> cb,
184184
Local<Object> domain;
185185
bool has_domain = false;
186186

187+
Environment::AsyncCallbackScope callback_scope(env());
188+
187189
if (env()->using_domains()) {
188190
Local<Value> domain_v = context->Get(env()->domain_string());
189191
has_domain = domain_v->IsObject();
@@ -236,7 +238,7 @@ Local<Value> AsyncWrap::MakeCallback(const Local<Function> cb,
236238

237239
Environment::TickInfo* tick_info = env()->tick_info();
238240

239-
if (tick_info->in_tick()) {
241+
if (callback_scope.in_makecallback()) {
240242
return ret;
241243
}
242244

@@ -249,12 +251,8 @@ Local<Value> AsyncWrap::MakeCallback(const Local<Function> cb,
249251
return ret;
250252
}
251253

252-
tick_info->set_in_tick(true);
253-
254254
env()->tick_callback_function()->Call(process, 0, nullptr);
255255

256-
tick_info->set_in_tick(false);
257-
258256
if (try_catch.HasCaught()) {
259257
tick_info->set_last_threw(true);
260258
return Undefined(env()->isolate());

‎src/env-inl.h

+15
Original file line numberDiff line numberDiff line change
@@ -101,6 +101,20 @@ inline void Environment::AsyncHooks::set_enable_callbacks(uint32_t flag) {
101101
fields_[kEnableCallbacks] = flag;
102102
}
103103

104+
inline Environment::AsyncCallbackScope::AsyncCallbackScope(Environment* env)
105+
: env_(env) {
106+
env_->makecallback_cntr_++;
107+
}
108+
109+
inline Environment::AsyncCallbackScope::~AsyncCallbackScope() {
110+
env_->makecallback_cntr_--;
111+
CHECK_GE(env_->makecallback_cntr_, 0);
112+
}
113+
114+
inline bool Environment::AsyncCallbackScope::in_makecallback() {
115+
return env_->makecallback_cntr_ > 1;
116+
}
117+
104118
inline Environment::DomainFlag::DomainFlag() {
105119
for (int i = 0; i < kFieldsCount; ++i) fields_[i] = 0;
106120
}
@@ -223,6 +237,7 @@ inline Environment::Environment(v8::Local<v8::Context> context,
223237
using_domains_(false),
224238
printed_error_(false),
225239
trace_sync_io_(false),
240+
makecallback_cntr_(0),
226241
async_wrap_uid_(0),
227242
debugger_agent_(this),
228243
http_parser_buffer_(nullptr),

‎src/env.cc

+2-6
Original file line numberDiff line numberDiff line change
@@ -64,10 +64,10 @@ void Environment::PrintSyncTrace() const {
6464
}
6565

6666

67-
bool Environment::KickNextTick() {
67+
bool Environment::KickNextTick(Environment::AsyncCallbackScope* scope) {
6868
TickInfo* info = tick_info();
6969

70-
if (info->in_tick()) {
70+
if (scope->in_makecallback()) {
7171
return true;
7272
}
7373

@@ -80,15 +80,11 @@ bool Environment::KickNextTick() {
8080
return true;
8181
}
8282

83-
info->set_in_tick(true);
84-
8583
// process nextTicks after call
8684
TryCatch try_catch;
8785
try_catch.SetVerbose(true);
8886
tick_callback_function()->Call(process_object(), 0, nullptr);
8987

90-
info->set_in_tick(false);
91-
9288
if (try_catch.HasCaught()) {
9389
info->set_last_threw(true);
9490
return false;

‎src/env.h

+15-1
Original file line numberDiff line numberDiff line change
@@ -314,6 +314,19 @@ class Environment {
314314
DISALLOW_COPY_AND_ASSIGN(AsyncHooks);
315315
};
316316

317+
class AsyncCallbackScope {
318+
public:
319+
explicit AsyncCallbackScope(Environment* env);
320+
~AsyncCallbackScope();
321+
322+
inline bool in_makecallback();
323+
324+
private:
325+
Environment* env_;
326+
327+
DISALLOW_COPY_AND_ASSIGN(AsyncCallbackScope);
328+
};
329+
317330
class DomainFlag {
318331
public:
319332
inline uint32_t* fields();
@@ -466,7 +479,7 @@ class Environment {
466479

467480
inline int64_t get_async_wrap_uid();
468481

469-
bool KickNextTick();
482+
bool KickNextTick(AsyncCallbackScope* scope);
470483

471484
inline uint32_t* heap_statistics_buffer() const;
472485
inline void set_heap_statistics_buffer(uint32_t* pointer);
@@ -569,6 +582,7 @@ class Environment {
569582
bool using_domains_;
570583
bool printed_error_;
571584
bool trace_sync_io_;
585+
size_t makecallback_cntr_;
572586
int64_t async_wrap_uid_;
573587
debugger::Agent debugger_agent_;
574588

‎src/node.cc

+7-2
Original file line numberDiff line numberDiff line change
@@ -1140,6 +1140,8 @@ Local<Value> MakeCallback(Environment* env,
11401140
bool ran_init_callback = false;
11411141
bool has_domain = false;
11421142

1143+
Environment::AsyncCallbackScope callback_scope(env);
1144+
11431145
// TODO(trevnorris): Adding "_asyncQueue" to the "this" in the init callback
11441146
// is a horrible way to detect usage. Rethink how detection should happen.
11451147
if (recv->IsObject()) {
@@ -1200,7 +1202,7 @@ Local<Value> MakeCallback(Environment* env,
12001202
}
12011203
}
12021204

1203-
if (!env->KickNextTick())
1205+
if (!env->KickNextTick(&callback_scope))
12041206
return Undefined(env->isolate());
12051207

12061208
return ret;
@@ -4151,7 +4153,10 @@ static void StartNodeInstance(void* arg) {
41514153
if (instance_data->use_debug_agent())
41524154
StartDebug(env, debug_wait_connect);
41534155

4154-
LoadEnvironment(env);
4156+
{
4157+
Environment::AsyncCallbackScope callback_scope(env);
4158+
LoadEnvironment(env);
4159+
}
41554160

41564161
env->set_trace_sync_io(trace_sync_io);
41574162

‎src/node_http_parser.cc

+3-1
Original file line numberDiff line numberDiff line change
@@ -578,6 +578,8 @@ class Parser : public BaseObject {
578578
if (!cb->IsFunction())
579579
return;
580580

581+
Environment::AsyncCallbackScope callback_scope(parser->env());
582+
581583
// Hooks for GetCurrentBuffer
582584
parser->current_buffer_len_ = nread;
583585
parser->current_buffer_data_ = buf->base;
@@ -587,7 +589,7 @@ class Parser : public BaseObject {
587589
parser->current_buffer_len_ = 0;
588590
parser->current_buffer_data_ = nullptr;
589591

590-
parser->env()->KickNextTick();
592+
parser->env()->KickNextTick(&callback_scope);
591593
}
592594

593595

‎src/node_internals.h

-2
Original file line numberDiff line numberDiff line change
@@ -69,8 +69,6 @@ v8::Local<v8::Value> MakeCallback(Environment* env,
6969
int argc = 0,
7070
v8::Local<v8::Value>* argv = nullptr);
7171

72-
bool KickNextTick();
73-
7472
// Convert a struct sockaddr to a { address: '1.2.3.4', port: 1234 } JS object.
7573
// Sets address and port properties on the info object and returns it.
7674
// If |info| is omitted, a new object is returned.

0 commit comments

Comments
 (0)
Please sign in to comment.