Node.js  v8.x
Node.js is a JavaScript runtime built on Chrome's V8 JavaScript engine
node_win32_etw_provider-inl.h
Go to the documentation of this file.
1 // Copyright Joyent, Inc. and other Node contributors.
2 //
3 // Permission is hereby granted, free of charge, to any person obtaining a
4 // copy of this software and associated documentation files (the
5 // "Software"), to deal in the Software without restriction, including
6 // without limitation the rights to use, copy, modify, merge, publish,
7 // distribute, sublicense, and/or sell copies of the Software, and to permit
8 // persons to whom the Software is furnished to do so, subject to the
9 // following conditions:
10 //
11 // The above copyright notice and this permission notice shall be included
12 // in all copies or substantial portions of the Software.
13 //
14 // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS
15 // OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
16 // MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN
17 // NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM,
18 // DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR
19 // OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE
20 // USE OR OTHER DEALINGS IN THE SOFTWARE.
21 
22 #ifndef SRC_NODE_WIN32_ETW_PROVIDER_INL_H_
23 #define SRC_NODE_WIN32_ETW_PROVIDER_INL_H_
24 
25 #if defined(NODE_WANT_INTERNALS) && NODE_WANT_INTERNALS
26 
28 #include "node_etw_provider.h"
29 
30 #if defined(_WIN64)
31 # define ETW_WRITE_INTPTR_DATA ETW_WRITE_INT64_DATA
32 #else
33 # define ETW_WRITE_INTPTR_DATA ETW_WRITE_INT32_DATA
34 #endif
35 
36 namespace node {
37 
38 // From node_win32_etw_provider.cc
39 extern REGHANDLE node_provider;
40 extern EventWriteFunc event_write;
41 extern int events_enabled;
42 
43 #define ETW_WRITE_STRING_DATA(data_descriptor, data) \
44  EventDataDescCreate(data_descriptor, \
45  data, \
46  (strlen(data) + 1) * sizeof(*data));
47 
48 #define ETW_WRITE_INT32_DATA(data_descriptor, data) \
49  EventDataDescCreate(data_descriptor, data, sizeof(int32_t));
50 
51 #define ETW_WRITE_INT64_DATA(data_descriptor, data) \
52  EventDataDescCreate(data_descriptor, data, sizeof(int64_t));
53 
54 #define ETW_WRITE_ADDRESS_DATA(data_descriptor, data) \
55  EventDataDescCreate(data_descriptor, data, sizeof(intptr_t));
56 
57 #define ETW_WRITE_INT16_DATA(data_descriptor, data) \
58  EventDataDescCreate(data_descriptor, data, sizeof(int16_t));
59 
60 #define ETW_WRITE_WSTRING_DATA_LENGTH(data_descriptor, data, data_len_bytes) \
61  EventDataDescCreate(data_descriptor, \
62  data, \
63  data_len_bytes);
64 
65 #define ETW_WRITE_NET_CONNECTION(descriptors, conn) \
66  ETW_WRITE_INT32_DATA(descriptors, &conn->fd); \
67  ETW_WRITE_INT32_DATA(descriptors + 1, &conn->port); \
68  ETW_WRITE_STRING_DATA(descriptors + 2, conn->remote); \
69  ETW_WRITE_INT32_DATA(descriptors + 3, &conn->buffered);
70 
71 #define ETW_WRITE_HTTP_SERVER_REQUEST(descriptors, req) \
72  ETW_WRITE_STRING_DATA(descriptors, req->url); \
73  ETW_WRITE_STRING_DATA(descriptors + 1, req->method); \
74  ETW_WRITE_STRING_DATA(descriptors + 2, req->forwardedFor);
75 
76 #define ETW_WRITE_HTTP_CLIENT_REQUEST(descriptors, req) \
77  ETW_WRITE_STRING_DATA(descriptors, req->url); \
78  ETW_WRITE_STRING_DATA(descriptors + 1, req->method);
79 
80 #define ETW_WRITE_GC(descriptors, type, flags) \
81  ETW_WRITE_INT32_DATA(descriptors, &type); \
82  ETW_WRITE_INT32_DATA(descriptors + 1, &flags);
83 
84 #define ETW_WRITE_V8ADDRESSCHANGE(descriptors, addr1, addr2) \
85  ETW_WRITE_ADDRESS_DATA(descriptors, &addr1); \
86  ETW_WRITE_ADDRESS_DATA(descriptors + 1, &addr2);
87 
88 #define ETW_WRITE_JSMETHOD_LOADUNLOAD(descriptors, \
89  context, \
90  startAddr, \
91  size, \
92  id, \
93  flags, \
94  rangeId, \
95  sourceId, \
96  line, \
97  col, \
98  name, \
99  name_len_bytes) \
100  ETW_WRITE_ADDRESS_DATA(descriptors, &context); \
101  ETW_WRITE_ADDRESS_DATA(descriptors + 1, &startAddr); \
102  ETW_WRITE_INT64_DATA(descriptors + 2, &size); \
103  ETW_WRITE_INTPTR_DATA(descriptors + 3, &id); \
104  ETW_WRITE_INT16_DATA(descriptors + 4, &flags); \
105  ETW_WRITE_INT16_DATA(descriptors + 5, &rangeId); \
106  ETW_WRITE_INT64_DATA(descriptors + 6, &sourceId); \
107  ETW_WRITE_INT32_DATA(descriptors + 7, &line); \
108  ETW_WRITE_INT32_DATA(descriptors + 8, &col); \
109  ETW_WRITE_WSTRING_DATA_LENGTH(descriptors + 9, name, name_len_bytes);
110 
111 
112 #define ETW_WRITE_EVENT(eventDescriptor, dataDescriptors) \
113  DWORD status = event_write(node_provider, \
114  &eventDescriptor, \
115  sizeof(dataDescriptors) / \
116  sizeof(*dataDescriptors), \
117  dataDescriptors); \
118  CHECK_EQ(status, ERROR_SUCCESS);
119 
120 #define ETW_WRITE_EMPTY_EVENT(eventDescriptor) \
121  DWORD status = event_write(node_provider, \
122  &eventDescriptor, \
123  0, \
124  NULL); \
125  CHECK_EQ(status, ERROR_SUCCESS);
126 
127 
129  node_dtrace_connection_t* conn, const char *remote, int port,
130  const char *method, const char *url, int fd) {
131  EVENT_DATA_DESCRIPTOR descriptors[7];
132  ETW_WRITE_HTTP_SERVER_REQUEST(descriptors, req);
133  ETW_WRITE_NET_CONNECTION(descriptors + 3, conn);
134  ETW_WRITE_EVENT(NODE_HTTP_SERVER_REQUEST_EVENT, descriptors);
135 }
136 
137 
139  const char *remote, int port, int fd) {
140  EVENT_DATA_DESCRIPTOR descriptors[4];
141  ETW_WRITE_NET_CONNECTION(descriptors, conn);
142  ETW_WRITE_EVENT(NODE_HTTP_SERVER_RESPONSE_EVENT, descriptors);
143 }
144 
145 
147  node_dtrace_connection_t* conn, const char *remote, int port,
148  const char *method, const char *url, int fd) {
149  EVENT_DATA_DESCRIPTOR descriptors[6];
150  ETW_WRITE_HTTP_CLIENT_REQUEST(descriptors, req);
151  ETW_WRITE_NET_CONNECTION(descriptors + 2, conn);
152  ETW_WRITE_EVENT(NODE_HTTP_CLIENT_REQUEST_EVENT, descriptors);
153 }
154 
155 
157  const char *remote, int port, int fd) {
158  EVENT_DATA_DESCRIPTOR descriptors[4];
159  ETW_WRITE_NET_CONNECTION(descriptors, conn);
160  ETW_WRITE_EVENT(NODE_HTTP_CLIENT_RESPONSE_EVENT, descriptors);
161 }
162 
163 
165  const char *remote, int port, int fd) {
166  EVENT_DATA_DESCRIPTOR descriptors[4];
167  ETW_WRITE_NET_CONNECTION(descriptors, conn);
168  ETW_WRITE_EVENT(NODE_NET_SERVER_CONNECTION_EVENT, descriptors);
169 }
170 
171 
173  const char *remote, int port, int fd) {
174  EVENT_DATA_DESCRIPTOR descriptors[4];
175  ETW_WRITE_NET_CONNECTION(descriptors, conn);
176  ETW_WRITE_EVENT(NODE_NET_STREAM_END_EVENT, descriptors);
177 }
178 
179 
180 void NODE_GC_START(v8::GCType type,
181  v8::GCCallbackFlags flags,
182  v8::Isolate* isolate) {
183  if (events_enabled > 0) {
184  EVENT_DATA_DESCRIPTOR descriptors[2];
185  ETW_WRITE_GC(descriptors, type, flags);
186  ETW_WRITE_EVENT(NODE_GC_START_EVENT, descriptors);
187  }
188 }
189 
190 
191 void NODE_GC_DONE(v8::GCType type,
192  v8::GCCallbackFlags flags,
193  v8::Isolate* isolate) {
194  if (events_enabled > 0) {
195  EVENT_DATA_DESCRIPTOR descriptors[2];
196  ETW_WRITE_GC(descriptors, type, flags);
197  ETW_WRITE_EVENT(NODE_GC_DONE_EVENT, descriptors);
198  }
199 }
200 
201 
202 void NODE_V8SYMBOL_REMOVE(const void* addr1, const void* addr2) {
203  if (events_enabled > 0) {
204  EVENT_DATA_DESCRIPTOR descriptors[2];
205  ETW_WRITE_V8ADDRESSCHANGE(descriptors, addr1, addr2);
206  ETW_WRITE_EVENT(NODE_V8SYMBOL_REMOVE_EVENT, descriptors);
207  }
208 }
209 
210 
211 void NODE_V8SYMBOL_MOVE(const void* addr1, const void* addr2) {
212  if (events_enabled > 0) {
213  EVENT_DATA_DESCRIPTOR descriptors[2];
214  ETW_WRITE_V8ADDRESSCHANGE(descriptors, addr1, addr2);
215  ETW_WRITE_EVENT(NODE_V8SYMBOL_MOVE_EVENT, descriptors);
216  }
217 }
218 
219 
220 void NODE_V8SYMBOL_RESET() {
221  if (events_enabled > 0) {
222  ETW_WRITE_EMPTY_EVENT(NODE_V8SYMBOL_RESET_EVENT);
223  }
224 }
225 
226 #define SETSYMBUF(s) \
227  wcscpy(symbuf, s); \
228  symbol_len = arraysize(s) - 1;
229 
230 void NODE_V8SYMBOL_ADD(LPCSTR symbol,
231  int symbol_len,
232  const void* addr1,
233  int len) {
234  if (events_enabled > 0) {
235  wchar_t symbuf[128];
236  if (symbol == nullptr) {
237  SETSYMBUF(L"nullptr");
238  } else {
239  symbol_len = MultiByteToWideChar(CP_ACP,
240  0,
241  symbol,
242  symbol_len,
243  symbuf,
244  128);
245  if (symbol_len == 0) {
246  SETSYMBUF(L"Invalid");
247  } else {
248  if (symbol_len > 127) {
249  symbol_len = 127;
250  }
251  symbuf[symbol_len] = L'\0';
252  }
253  }
254  void* context = nullptr;
255  INT64 size = (INT64)len;
256  INT_PTR id = (INT_PTR)addr1;
257  INT16 flags = 0;
258  INT16 rangeid = 1;
259  INT32 col = 1;
260  INT32 line = 1;
261  INT64 sourceid = 0;
262  EVENT_DATA_DESCRIPTOR descriptors[10];
263  ETW_WRITE_JSMETHOD_LOADUNLOAD(descriptors,
264  context,
265  addr1,
266  size,
267  id,
268  flags,
269  rangeid,
270  sourceid,
271  line,
272  col,
273  symbuf,
274  (symbol_len + 1) * sizeof(symbuf[0]));
275  ETW_WRITE_EVENT(MethodLoad, descriptors);
276  }
277 }
278 #undef SETSYMBUF
279 
280 
281 bool NODE_HTTP_SERVER_REQUEST_ENABLED() { return events_enabled > 0; }
282 bool NODE_HTTP_SERVER_RESPONSE_ENABLED() { return events_enabled > 0; }
283 bool NODE_HTTP_CLIENT_REQUEST_ENABLED() { return events_enabled > 0; }
284 bool NODE_HTTP_CLIENT_RESPONSE_ENABLED() { return events_enabled > 0; }
285 bool NODE_NET_SERVER_CONNECTION_ENABLED() { return events_enabled > 0; }
286 bool NODE_NET_STREAM_END_ENABLED() { return events_enabled > 0; }
287 bool NODE_V8SYMBOL_ENABLED() { return events_enabled > 0; }
288 
289 } // namespace node
290 
291 #endif // defined(NODE_WANT_INTERNALS) && NODE_WANT_INTERNALS
292 
293 #endif // SRC_NODE_WIN32_ETW_PROVIDER_INL_H_
#define NODE_HTTP_CLIENT_RESPONSE(arg0)
Definition: node_dtrace.cc:36
EventWriteFunc event_write
#define NODE_NET_STREAM_END_ENABLED()
Definition: node_dtrace.cc:41
int len
Definition: cares_wrap.cc:485
#define NODE_NET_STREAM_END(arg0)
Definition: node_dtrace.cc:40
#define NODE_GC_DONE(arg0, arg1, arg2)
Definition: node_dtrace.cc:43
#define NODE_HTTP_CLIENT_REQUEST(arg0, arg1)
Definition: node_dtrace.cc:34
#define NODE_HTTP_SERVER_REQUEST_ENABLED()
Definition: node_dtrace.cc:31
#define NODE_NET_SERVER_CONNECTION_ENABLED()
Definition: node_dtrace.cc:39
#define NODE_HTTP_CLIENT_REQUEST_ENABLED()
Definition: node_dtrace.cc:35
#define NODE_HTTP_SERVER_RESPONSE_ENABLED()
Definition: node_dtrace.cc:33
#define NODE_HTTP_CLIENT_RESPONSE_ENABLED()
Definition: node_dtrace.cc:37
uv_fs_t req
Definition: node_file.cc:374
REGHANDLE node_provider
method
Definition: node.d:195
#define NODE_NET_SERVER_CONNECTION(arg0)
Definition: node_dtrace.cc:38
#define NODE_GC_START(arg0, arg1, arg2)
Definition: node_dtrace.cc:42
#define NODE_HTTP_SERVER_RESPONSE(arg0)
Definition: node_dtrace.cc:32
#define NODE_HTTP_SERVER_REQUEST(arg0, arg1)
Definition: node_dtrace.cc:30