Skip to content

Commit 9bad248

Browse files
author
Matheus Marchini
committed
src: add commands to inspect the workqueue
Add two new commands: `v8 getactivehandles` and `v8 getactiverequests`. These comamnds will print all pending handles and requests. The result should be similar to running process._getActiveHandles() and process._getActiveRequests() on the living process. Fixes: #100
1 parent cf25dd3 commit 9bad248

10 files changed

+502
-5
lines changed

README.md

+6
Original file line numberDiff line numberDiff line change
@@ -207,6 +207,12 @@ The following subcommands are supported:
207207
* -n, --name name - all properties with the specified name
208208
* -s, --string string - all properties that refer to the specified JavaScript string value
209209
210+
getactivehandles -- Print all pending handles in the queue. Equivalent to running process._getActiveHandles() on
211+
the living process.
212+
213+
getactiverequests -- Print all pending handles in the queue. Equivalent to running process._getActiveHandles() on
214+
the living process.
215+
210216
inspect -- Print detailed description and contents of the JavaScript value.
211217
212218
Possible flags (all optional):

llnode.gyp.json

+1
Original file line numberDiff line numberDiff line change
@@ -22,6 +22,7 @@
2222
"src/llv8.cc",
2323
"src/llv8-constants.cc",
2424
"src/llscan.cc",
25+
"src/node.cc",
2526
"src/node-constants.cc",
2627
],
2728

src/llnode.cc

+121
Original file line numberDiff line numberDiff line change
@@ -4,13 +4,16 @@
44
#include <string.h>
55

66
#include <cinttypes>
7+
#include <sstream>
8+
#include <string>
79

810
#include <lldb/API/SBExpressionOptions.h>
911

1012
#include "src/error.h"
1113
#include "src/llnode.h"
1214
#include "src/llscan.h"
1315
#include "src/llv8.h"
16+
#include "src/node-inl.h"
1417

1518
namespace llnode {
1619

@@ -20,6 +23,7 @@ using lldb::SBDebugger;
2023
using lldb::SBError;
2124
using lldb::SBExpressionOptions;
2225
using lldb::SBFrame;
26+
using lldb::SBProcess;
2327
using lldb::SBStream;
2428
using lldb::SBSymbol;
2529
using lldb::SBTarget;
@@ -300,6 +304,112 @@ bool ListCmd::DoExecute(SBDebugger d, char** cmd,
300304
return true;
301305
}
302306

307+
bool WorkqueueCmd::DoExecute(SBDebugger d, char** cmd,
308+
SBCommandReturnObject& result) {
309+
SBTarget target = d.GetSelectedTarget();
310+
SBProcess process = target.GetProcess();
311+
SBThread thread = process.GetSelectedThread();
312+
std::string result_message;
313+
Error err;
314+
315+
llv8_->Load(target);
316+
node_->Load(target);
317+
318+
if (!thread.IsValid()) {
319+
result.SetError("No valid process, please start something\n");
320+
return false;
321+
}
322+
323+
node::Environment env = node::Environment::GetCurrent(node_, err);
324+
if (err.Fail()) {
325+
result.SetError(err.GetMessage());
326+
return false;
327+
}
328+
329+
result_message = GetResultMessage(&env, err);
330+
331+
if (err.Fail()) {
332+
result.SetError(err.GetMessage());
333+
return false;
334+
}
335+
336+
result.Printf("%s", result_message.c_str());
337+
return true;
338+
}
339+
340+
std::string GetActiveHandlesCmd::GetResultMessage(node::Environment* env,
341+
Error& err) {
342+
int active_handles = 0;
343+
v8::Value::InspectOptions inspect_options;
344+
inspect_options.detailed = true;
345+
std::ostringstream result_message;
346+
347+
for (auto w : env->handle_wrap_queue()) {
348+
addr_t persistent_addr = w.persistent_addr(err);
349+
if (err.Fail()) {
350+
break;
351+
}
352+
if (persistent_addr == 0) {
353+
continue;
354+
}
355+
356+
addr_t v8_object_addr = w.v8_object_addr(err);
357+
if (err.Fail()) {
358+
break;
359+
}
360+
v8::JSObject v8_object(llv8(), v8_object_addr);
361+
std::string res = v8_object.Inspect(&inspect_options, err);
362+
if (err.Fail()) {
363+
Error::PrintInDebugMode("Failed to load object at address %" PRIx64,
364+
v8_object_addr);
365+
break;
366+
}
367+
368+
active_handles++;
369+
result_message << res.c_str() << std::endl;
370+
}
371+
372+
result_message << "Total: " << active_handles << std::endl;
373+
return result_message.str();
374+
}
375+
376+
377+
std::string GetActiveRequestsCmd::GetResultMessage(node::Environment* env,
378+
Error& err) {
379+
int active_handles = 0;
380+
v8::Value::InspectOptions inspect_options;
381+
inspect_options.detailed = true;
382+
std::ostringstream result_message;
383+
384+
for (auto w : env->req_wrap_queue()) {
385+
addr_t persistent_addr = w.persistent_addr(err);
386+
if (err.Fail()) {
387+
break;
388+
}
389+
if (persistent_addr == 0) {
390+
continue;
391+
}
392+
393+
addr_t v8_object_addr = w.v8_object_addr(err);
394+
if (err.Fail()) {
395+
break;
396+
}
397+
v8::JSObject v8_object(llv8(), v8_object_addr);
398+
std::string res = v8_object.Inspect(&inspect_options, err);
399+
if (err.Fail()) {
400+
Error::PrintInDebugMode("Failed to load object at address %" PRIx64,
401+
v8_object_addr);
402+
break;
403+
}
404+
405+
active_handles++;
406+
result_message << res.c_str() << std::endl;
407+
}
408+
409+
result_message << "Total: " << active_handles << std::endl;
410+
return result_message.str();
411+
}
412+
303413

304414
void InitDebugMode() {
305415
bool is_debug_mode = false;
@@ -319,6 +429,7 @@ bool PluginInitialize(SBDebugger d) {
319429
llnode::InitDebugMode();
320430

321431
static llnode::v8::LLV8 llv8;
432+
static llnode::node::Node node(&llv8);
322433
static llnode::LLScan llscan = llnode::LLScan(&llv8);
323434

324435
SBCommandInterpreter interpreter = d.GetCommandInterpreter();
@@ -404,6 +515,16 @@ bool PluginInitialize(SBDebugger d) {
404515
"JavaScript string value\n"
405516
"\n");
406517

518+
v8.AddCommand("getactivehandles",
519+
new llnode::GetActiveHandlesCmd(&llv8, &node),
520+
"Print all pending handles in the queue. Equivalent to running "
521+
"process._getActiveHandles() on the living process.\n");
522+
523+
v8.AddCommand(
524+
"getactiverequests", new llnode::GetActiveRequestsCmd(&llv8, &node),
525+
"Print all pending requests in the queue. Equivalent to "
526+
"running process._getActiveRequests() on the living process.\n");
527+
407528
return true;
408529
}
409530

src/llnode.h

+38
Original file line numberDiff line numberDiff line change
@@ -6,6 +6,7 @@
66
#include <lldb/API/LLDB.h>
77

88
#include "src/llv8.h"
9+
#include "src/node.h"
910

1011
namespace llnode {
1112

@@ -52,6 +53,43 @@ class ListCmd : public CommandBase {
5253
v8::LLV8* llv8_;
5354
};
5455

56+
class WorkqueueCmd : public CommandBase {
57+
public:
58+
WorkqueueCmd(v8::LLV8* llv8, node::Node* node) : llv8_(llv8), node_(node) {}
59+
~WorkqueueCmd() override {}
60+
61+
inline v8::LLV8* llv8() { return llv8_; };
62+
inline node::Node* node() { return node_; };
63+
64+
bool DoExecute(lldb::SBDebugger d, char** cmd,
65+
lldb::SBCommandReturnObject& result) override;
66+
67+
virtual std::string GetResultMessage(node::Environment* env, Error& err) {
68+
return std::string();
69+
};
70+
71+
private:
72+
v8::LLV8* llv8_;
73+
node::Node* node_;
74+
};
75+
76+
class GetActiveHandlesCmd : public WorkqueueCmd {
77+
public:
78+
GetActiveHandlesCmd(v8::LLV8* llv8, node::Node* node)
79+
: WorkqueueCmd(llv8, node) {}
80+
81+
std::string GetResultMessage(node::Environment* env, Error& err) override;
82+
};
83+
84+
class GetActiveRequestsCmd : public WorkqueueCmd {
85+
public:
86+
GetActiveRequestsCmd(v8::LLV8* llv8, node::Node* node)
87+
: WorkqueueCmd(llv8, node) {}
88+
89+
std::string GetResultMessage(node::Environment* env, Error& err) override;
90+
};
91+
92+
5593
} // namespace llnode
5694

5795
#endif // SRC_LLNODE_H_

src/node-inl.h

+42
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,42 @@
1+
#include "node.h"
2+
3+
namespace llnode {
4+
namespace node {
5+
6+
template <typename T, typename C>
7+
T Queue<T, C>::Iterator::operator*() const {
8+
return T::FromListNode(node_, current_);
9+
}
10+
11+
template <typename T, typename C>
12+
const typename Queue<T, C>::Iterator Queue<T, C>::Iterator::operator++() {
13+
lldb::SBError sberr;
14+
15+
addr_t current = current_ + constants_->kNextOffset;
16+
current = node_->process().ReadPointerFromMemory(current, sberr);
17+
current_ = current;
18+
return Iterator(node_, current, constants_);
19+
}
20+
21+
template <typename T, typename C>
22+
bool Queue<T, C>::Iterator::operator!=(const Iterator& that) const {
23+
return current_ != that.current_;
24+
}
25+
26+
template <typename T, typename C>
27+
typename Queue<T, C>::Iterator Queue<T, C>::begin() const {
28+
lldb::SBError sberr;
29+
addr_t currentNode = raw_ + constants_->kHeadOffset;
30+
31+
currentNode = currentNode + constants_->kNextOffset;
32+
currentNode = node_->process().ReadPointerFromMemory(currentNode, sberr);
33+
34+
return Iterator(node_, currentNode, constants_);
35+
}
36+
37+
template <typename T, typename C>
38+
typename Queue<T, C>::Iterator Queue<T, C>::end() const {
39+
return Iterator(node_, raw_ + constants_->kHeadOffset, constants_);
40+
}
41+
} // namespace node
42+
} // namespace llnode

src/node.cc

+77
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,77 @@
1+
#include "node.h"
2+
3+
namespace llnode {
4+
namespace node {
5+
6+
addr_t BaseObject::persistent_addr(Error& err) {
7+
lldb::SBError sberr;
8+
9+
addr_t persistentHandlePtr =
10+
raw_ + node_->base_object()->kPersistentHandleOffset;
11+
addr_t persistentHandle =
12+
node_->process().ReadPointerFromMemory(persistentHandlePtr, sberr);
13+
if (sberr.Fail()) {
14+
err = Error::Failure("Failed to load persistent handle");
15+
return 0;
16+
}
17+
return persistentHandle;
18+
}
19+
20+
addr_t BaseObject::v8_object_addr(Error& err) {
21+
lldb::SBError sberr;
22+
23+
addr_t persistentHandle = persistent_addr(err);
24+
addr_t obj = node_->process().ReadPointerFromMemory(persistentHandle, sberr);
25+
if (sberr.Fail()) {
26+
err = Error::Failure("Failed to load object from persistent handle");
27+
return 0;
28+
}
29+
return obj;
30+
}
31+
32+
HandleWrap HandleWrap::FromListNode(Node* node, addr_t list_node_addr) {
33+
return HandleWrap(node,
34+
list_node_addr - node->handle_wrap()->kListNodeOffset);
35+
}
36+
37+
ReqWrap ReqWrap::FromListNode(Node* node, addr_t list_node_addr) {
38+
return ReqWrap(node, list_node_addr - node->req_wrap()->kListNodeOffset);
39+
}
40+
41+
Environment Environment::GetCurrent(Node* node, Error& err) {
42+
addr_t envAddr = node->env()->kCurrentEnvironment;
43+
if (envAddr == 0) {
44+
err = Error::Failure("Couldn't get node's Environment");
45+
}
46+
47+
return Environment(node, envAddr);
48+
}
49+
50+
HandleWrapQueue Environment::handle_wrap_queue() const {
51+
return HandleWrapQueue(node_, raw_ + node_->env()->kHandleWrapQueueOffset,
52+
node_->handle_wrap_queue());
53+
}
54+
55+
ReqWrapQueue Environment::req_wrap_queue() const {
56+
return ReqWrapQueue(node_, raw_ + node_->env()->kReqWrapQueueOffset,
57+
node_->req_wrap_queue());
58+
}
59+
60+
void Node::Load(SBTarget target) {
61+
// Reload process anyway
62+
process_ = target.GetProcess();
63+
64+
// No need to reload
65+
if (target_ == target) return;
66+
67+
target_ = target;
68+
69+
env.Assign(target);
70+
req_wrap_queue.Assign(target);
71+
req_wrap.Assign(target);
72+
handle_wrap_queue.Assign(target);
73+
handle_wrap.Assign(target);
74+
base_object.Assign(target);
75+
}
76+
} // namespace node
77+
} // namespace llnode

0 commit comments

Comments
 (0)