Skip to content

Commit 374dbe2

Browse files
committed
src,lib: expose node.http trace flag by array
Instead call the C++ code every time we make a HTTP request, now we get the C++ pointer to the flag that holds the info if the trace is enabled for node.http, then we expose that flag using BindingData, with this change, no C++ call is made and the access to the info happens in JS side, which has no perf penalty.
1 parent 85ac915 commit 374dbe2

9 files changed

+161
-4
lines changed

lib/internal/http.js

+6-2
Original file line numberDiff line numberDiff line change
@@ -8,7 +8,11 @@ const {
88
} = primordials;
99

1010
const { setUnrefTimeout } = require('internal/timers');
11-
const { trace, isTraceCategoryEnabled } = internalBinding('trace_events');
11+
const { trace } = internalBinding('trace_events');
12+
// The trace variable must not be destructured
13+
// because it might break node snapshot serialization
14+
// ref: https://github.com/nodejs/node/pull/48142#discussion_r1204128367
15+
const tracing = internalBinding('tracing');
1216
const {
1317
CHAR_LOWERCASE_B,
1418
CHAR_LOWERCASE_E,
@@ -38,7 +42,7 @@ function getNextTraceEventId() {
3842
}
3943

4044
function isTraceHTTPEnabled() {
41-
return isTraceCategoryEnabled('node.http');
45+
return tracing.tracingCategories[0] > 0;
4246
}
4347

4448
const traceEventCategory = 'node,node.http';

node.gyp

+2
Original file line numberDiff line numberDiff line change
@@ -128,6 +128,7 @@
128128
'src/node_symbols.cc',
129129
'src/node_task_queue.cc',
130130
'src/node_trace_events.cc',
131+
'src/node_tracing.cc',
131132
'src/node_types.cc',
132133
'src/node_url.cc',
133134
'src/node_util.cc',
@@ -248,6 +249,7 @@
248249
'src/node_sockaddr-inl.h',
249250
'src/node_stat_watcher.h',
250251
'src/node_union_bytes.h',
252+
'src/node_tracing.h',
251253
'src/node_url.h',
252254
'src/node_util.h',
253255
'src/node_version.h',

src/base_object_types.h

+2-1
Original file line numberDiff line numberDiff line change
@@ -16,7 +16,8 @@ namespace node {
1616
V(blob_binding_data, BlobBindingData) \
1717
V(process_binding_data, process::BindingData) \
1818
V(timers_binding_data, timers::BindingData) \
19-
V(url_binding_data, url::BindingData)
19+
V(url_binding_data, url::BindingData) \
20+
V(tracing_binding_data, tracing::BindingData)
2021

2122
#define UNSERIALIZABLE_BINDING_TYPES(V) \
2223
V(http2_binding_data, http2::BindingData) \

src/node_binding.cc

+1
Original file line numberDiff line numberDiff line change
@@ -69,6 +69,7 @@
6969
V(task_queue) \
7070
V(tcp_wrap) \
7171
V(timers) \
72+
V(tracing) \
7273
V(trace_events) \
7374
V(tty_wrap) \
7475
V(types) \

src/node_builtins.cc

+1-1
Original file line numberDiff line numberDiff line change
@@ -113,7 +113,7 @@ BuiltinLoader::BuiltinCategories BuiltinLoader::GetBuiltinCategories() const {
113113
#endif // !HAVE_INSPECTOR
114114

115115
#if !NODE_USE_V8_PLATFORM || !defined(NODE_HAVE_I18N_SUPPORT)
116-
"trace_events",
116+
"tracing", "trace_events",
117117
#endif // !NODE_USE_V8_PLATFORM || !defined(NODE_HAVE_I18N_SUPPORT)
118118

119119
#if !HAVE_OPENSSL

src/node_external_reference.h

+1
Original file line numberDiff line numberDiff line change
@@ -110,6 +110,7 @@ class ExternalReferenceRegistry {
110110
V(string_decoder) \
111111
V(stream_wrap) \
112112
V(signal_wrap) \
113+
V(tracing) \
113114
V(trace_events) \
114115
V(timers) \
115116
V(types) \

src/node_snapshotable.cc

+1
Original file line numberDiff line numberDiff line change
@@ -19,6 +19,7 @@
1919
#include "node_metadata.h"
2020
#include "node_process.h"
2121
#include "node_snapshot_builder.h"
22+
#include "node_tracing.h"
2223
#include "node_url.h"
2324
#include "node_util.h"
2425
#include "node_v8.h"

src/node_tracing.cc

+102
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,102 @@
1+
#include "node_tracing.h"
2+
#include "base_object-inl.h"
3+
#include "env-inl.h"
4+
#include "memory_tracker-inl.h"
5+
#include "node.h"
6+
#include "node_external_reference.h"
7+
#include "node_internals.h"
8+
#include "node_v8_platform-inl.h"
9+
#include "tracing/agent.h"
10+
#include "util-inl.h"
11+
#include "v8.h"
12+
13+
#include <numeric>
14+
#include <set>
15+
#include <string>
16+
17+
namespace node {
18+
namespace tracing {
19+
20+
using v8::Context;
21+
using v8::FunctionTemplate;
22+
using v8::HandleScope;
23+
using v8::Local;
24+
using v8::Object;
25+
using v8::Uint8Array;
26+
using v8::Value;
27+
28+
void BindingData::MemoryInfo(MemoryTracker* tracker) const {}
29+
30+
BindingData::BindingData(Realm* realm, v8::Local<v8::Object> object)
31+
: SnapshotableObject(realm, object, type_int) {
32+
// Get the pointer of the memory for the flag that
33+
// store if trace is enabled for http the pointer will
34+
// always be the same and if the category does not exist
35+
// it creates:
36+
// https://github.com/nodejs/node/blob/6bbf2a57fcf33266c5859497f8cc32e1389a358a/deps/v8/src/libplatform/tracing/tracing-controller.cc#L322-L342
37+
auto p = const_cast<uint8_t*>(
38+
TRACE_EVENT_API_GET_CATEGORY_GROUP_ENABLED("node.http"));
39+
40+
// NO_OP deleter
41+
// the idea of this chunk of code is to construct
42+
auto nop = [](void*, size_t, void*) {};
43+
auto bs = v8::ArrayBuffer::NewBackingStore(
44+
p, 1, nop, nullptr);
45+
auto ab =
46+
v8::ArrayBuffer::New(realm->isolate(), std::move(bs));
47+
v8::Local<Uint8Array> u8 = v8::Uint8Array::New(ab, 0, 1);
48+
49+
object
50+
->Set(realm->context(),
51+
FIXED_ONE_BYTE_STRING(realm->isolate(), "tracingCategories"),
52+
u8)
53+
.Check();
54+
}
55+
56+
bool BindingData::PrepareForSerialization(v8::Local<v8::Context> context,
57+
v8::SnapshotCreator* creator) {
58+
return true;
59+
}
60+
61+
InternalFieldInfoBase* BindingData::Serialize(int index) {
62+
DCHECK_EQ(index, BaseObject::kEmbedderType);
63+
InternalFieldInfo* info =
64+
InternalFieldInfoBase::New<InternalFieldInfo>(type());
65+
return info;
66+
}
67+
68+
void BindingData::Deserialize(v8::Local<v8::Context> context,
69+
v8::Local<v8::Object> holder,
70+
int index,
71+
InternalFieldInfoBase* info) {
72+
DCHECK_EQ(index, BaseObject::kEmbedderType);
73+
v8::HandleScope scope(context->GetIsolate());
74+
Realm* realm = Realm::GetCurrent(context);
75+
BindingData* binding = realm->AddBindingData<BindingData>(context, holder);
76+
CHECK_NOT_NULL(binding);
77+
}
78+
79+
void BindingData::CreatePerIsolateProperties(IsolateData* isolate_data,
80+
Local<FunctionTemplate> ctor) {}
81+
82+
void BindingData::CreatePerContextProperties(Local<Object> target,
83+
Local<Value> unused,
84+
Local<Context> context,
85+
void* priv) {
86+
Realm* realm = Realm::GetCurrent(context);
87+
realm->AddBindingData<BindingData>(context, target);
88+
}
89+
90+
void BindingData::RegisterExternalReferences(
91+
ExternalReferenceRegistry* registry) {}
92+
93+
} // namespace tracing
94+
95+
} // namespace node
96+
97+
NODE_BINDING_CONTEXT_AWARE_INTERNAL(
98+
tracing, node::tracing::BindingData::CreatePerContextProperties)
99+
NODE_BINDING_PER_ISOLATE_INIT(
100+
tracing, node::tracing::BindingData::CreatePerIsolateProperties)
101+
NODE_BINDING_EXTERNAL_REFERENCE(
102+
tracing, node::tracing::BindingData::RegisterExternalReferences)

src/node_tracing.h

+45
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,45 @@
1+
#ifndef SRC_NODE_TRACING_H_
2+
#define SRC_NODE_TRACING_H_
3+
4+
#include <cinttypes>
5+
#include "aliased_buffer.h"
6+
#include "node.h"
7+
#include "node_snapshotable.h"
8+
#include "util.h"
9+
#include "v8-fast-api-calls.h"
10+
#include "v8.h"
11+
12+
#include <string>
13+
14+
namespace node {
15+
class ExternalReferenceRegistry;
16+
17+
namespace tracing {
18+
19+
class BindingData : public SnapshotableObject {
20+
public:
21+
BindingData(Realm* realm, v8::Local<v8::Object> obj);
22+
23+
using InternalFieldInfo = InternalFieldInfoBase;
24+
25+
SERIALIZABLE_OBJECT_METHODS()
26+
SET_BINDING_ID(tracing_binding_data)
27+
28+
void MemoryInfo(MemoryTracker* tracker) const override;
29+
SET_SELF_SIZE(BindingData)
30+
SET_MEMORY_INFO_NAME(BindingData)
31+
32+
static void CreatePerIsolateProperties(IsolateData* isolate_data,
33+
v8::Local<v8::FunctionTemplate> ctor);
34+
static void CreatePerContextProperties(v8::Local<v8::Object> target,
35+
v8::Local<v8::Value> unused,
36+
v8::Local<v8::Context> context,
37+
void* priv);
38+
static void RegisterExternalReferences(ExternalReferenceRegistry* registry);
39+
};
40+
41+
} // namespace tracing
42+
43+
} // namespace node
44+
45+
#endif // SRC_NODE_TRACING_H_

0 commit comments

Comments
 (0)