Skip to content

Commit 2084b2b

Browse files
author
Matheus Marchini
committed
Merge remote-tracking branch 'origin/master' into workqueue_cmd
2 parents 71c965d + f85f8fc commit 2084b2b

12 files changed

+118
-34
lines changed

README.md

+6-6
Original file line numberDiff line numberDiff line change
@@ -212,7 +212,7 @@ v8 bt command. See the [Commands](#commands) section below for more commands.
212212
### Commands
213213

214214
```
215-
(lldb) v8 help
215+
(llnode) v8 help
216216
Node.js helpers
217217
218218
Syntax: v8
@@ -226,7 +226,7 @@ The following subcommands are supported:
226226
Syntax: v8 bt [number]
227227
findjsinstances -- List all objects which share the specified map.
228228
Accepts the same options as `v8 inspect`
229-
findjsobjects -- List all object types and instance counts grouped by map and sorted by instance count.
229+
findjsobjects -- List all object types and instance counts grouped by typename and sorted by instance count.
230230
Requires `LLNODE_RANGESFILE` environment variable to be set to a file containing memory ranges for the
231231
core file being debugged.
232232
There are scripts for generating this file on Linux and Mac in the scripts directory of the llnode
@@ -238,25 +238,25 @@ The following subcommands are supported:
238238
* -v, --value expr - all properties that refer to the specified JavaScript object (default)
239239
* -n, --name name - all properties with the specified name
240240
* -s, --string string - all properties that refer to the specified JavaScript string value
241-
* --array-length num - print maximum of `num` elements in array
242241
243-
getactivehandles -- *EXPERIMENTAL* Equivalent to running process._getActiveHandles. This command is still being developed and for now it only works building node from source.
244-
getactiverequests -- *EXPERIMENTAL* Equivalent to running process._getActiveRequests. This command is still being developed and for now it only works building node from source.
245242
inspect -- Print detailed description and contents of the JavaScript value.
246243
247244
Possible flags (all optional):
248245
249246
* -F, --full-string - print whole string without adding ellipsis
250247
* -m, --print-map - print object's map address
251248
* -s, --print-source - print source code for function objects
252-
* --string-length num - print maximum of `num` characters in string
249+
* -l num, --length num - print maximum of `num` elements from string/array
253250
254251
Syntax: v8 inspect [flags] expr
255252
nodeinfo -- Print information about Node.js
256253
print -- Print short description of the JavaScript value.
257254
258255
Syntax: v8 print expr
259256
source -- Source code information
257+
getactivehandles -- *EXPERIMENTAL* Equivalent to running process._getActiveHandles. This command is still being developed and for now it only works building node from source.
258+
getactiverequests -- *EXPERIMENTAL* Equivalent to running process._getActiveRequests. This command is still being developed and for now it only works building node from source.
259+
260260
261261
For more help on any particular subcommand, type 'help <command> <subcommand>'.
262262
```

package.json

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,6 @@
11
{
22
"name": "llnode",
3-
"version": "1.6.0",
3+
"version": "1.6.1",
44
"description": "Node.js plugin for LLDB",
55
"main": "no-entry-sorry.js",
66
"directories": {

src/llnode.cc

+11-13
Original file line numberDiff line numberDiff line change
@@ -46,8 +46,9 @@ char** CommandBase::ParseInspectOptions(char** cmd,
4646
v8::Value::InspectOptions* options) {
4747
static struct option opts[] = {
4848
{"full-string", no_argument, nullptr, 'F'},
49-
{"string-length", required_argument, nullptr, 0x1001},
50-
{"array-length", required_argument, nullptr, 0x1002},
49+
{"string-length", required_argument, nullptr, 'l'},
50+
{"array-length", required_argument, nullptr, 'l'},
51+
{"length", required_argument, nullptr, 'l'},
5152
{"print-map", no_argument, nullptr, 'm'},
5253
{"print-source", no_argument, nullptr, 's'},
5354
{nullptr, 0, nullptr, 0}};
@@ -67,21 +68,18 @@ char** CommandBase::ParseInspectOptions(char** cmd,
6768
optind = 0;
6869
opterr = 1;
6970
do {
70-
int arg = getopt_long(argc, args, "Fms", opts, nullptr);
71+
int arg = getopt_long(argc, args, "Fmsl:", opts, nullptr);
7172
if (arg == -1) break;
7273

7374
switch (arg) {
7475
case 'F':
75-
options->string_length = 0;
76+
options->length = 0;
7677
break;
7778
case 'm':
7879
options->print_map = true;
7980
break;
80-
case 0x1001:
81-
options->string_length = strtol(optarg, nullptr, 10);
82-
break;
83-
case 0x1002:
84-
options->array_length = strtol(optarg, nullptr, 10);
81+
case 'l':
82+
options->length = strtol(optarg, nullptr, 10);
8583
break;
8684
case 's':
8785
options->print_source = true;
@@ -431,8 +429,8 @@ bool PluginInitialize(SBDebugger d) {
431429
" * -F, --full-string - print whole string without adding ellipsis\n"
432430
" * -m, --print-map - print object's map address\n"
433431
" * -s, --print-source - print source code for function objects\n"
434-
" * --string-length num - print maximum of `num` characters in string\n"
435-
" * --array-length num - print maximum of `num` elements in array\n"
432+
" * -l num, --length num - print maximum of `num` elements from "
433+
"string/array\n"
436434
"\n"
437435
"Syntax: v8 inspect [flags] expr\n");
438436
interpreter.AddCommand("jsprint", new llnode::PrintCmd(true),
@@ -447,8 +445,8 @@ bool PluginInitialize(SBDebugger d) {
447445
"Alias for `v8 source list`");
448446

449447
v8.AddCommand("findjsobjects", new llnode::FindObjectsCmd(),
450-
"List all object types and instance counts grouped by map and "
451-
"sorted by instance count.\n"
448+
"List all object types and instance counts grouped by type"
449+
"name and sorted by instance count.\n"
452450
#ifndef LLDB_SBMemoryRegionInfoList_h_
453451
"Requires `LLNODE_RANGESFILE` environment variable to be set "
454452
"to a file containing memory ranges for the core file being "

src/llscan.cc

+16
Original file line numberDiff line numberDiff line change
@@ -617,6 +617,14 @@ void FindReferencesCmd::ReferenceScanner::PrintRefs(
617617
result.Printf("0x%" PRIx64 ": %s.%s=0x%" PRIx64 "\n", str.raw(),
618618
type_name.c_str(), "<Second>", search_value_.raw());
619619
}
620+
} else if (repr == v8->string()->kThinStringTag) {
621+
v8::ThinString thin_str(str);
622+
v8::String actual = thin_str.Actual(err);
623+
if (err.Success() && actual.raw() == search_value_.raw()) {
624+
std::string type_name = thin_str.GetTypeName(err);
625+
result.Printf("0x%" PRIx64 ": %s.%s=0x%" PRIx64 "\n", str.raw(),
626+
type_name.c_str(), "<Actual>", search_value_.raw());
627+
}
620628
}
621629
// Nothing to do for other kinds of string.
622630
}
@@ -694,6 +702,14 @@ void FindReferencesCmd::ReferenceScanner::ScanRefs(v8::String& str,
694702
references = llscan.GetReferencesByValue(second.raw());
695703
references->push_back(str.raw());
696704
}
705+
} else if (repr == v8->string()->kThinStringTag) {
706+
v8::ThinString thin_str(str);
707+
v8::String actual = thin_str.Actual(err);
708+
709+
if (err.Success()) {
710+
references = llscan.GetReferencesByValue(actual.raw());
711+
references->push_back(str.raw());
712+
}
697713
}
698714
// Nothing to do for other kinds of string.
699715
}

src/llv8-constants.cc

+5
Original file line numberDiff line numberDiff line change
@@ -337,6 +337,7 @@ void String::Load() {
337337
kConsStringTag = LoadConstant("ConsStringTag");
338338
kSlicedStringTag = LoadConstant("SlicedStringTag");
339339
kExternalStringTag = LoadConstant("ExternalStringTag");
340+
kThinStringTag = LoadConstant("ThinStringTag");
340341

341342
kLengthOffset = LoadConstant("class_String__length__SMI");
342343
}
@@ -365,6 +366,9 @@ void SlicedString::Load() {
365366
kOffsetOffset = LoadConstant("class_SlicedString__offset__SMI");
366367
}
367368

369+
void ThinString::Load() {
370+
kActualOffset = LoadConstant("class_ThinString__actual__String");
371+
}
368372

369373
void FixedArrayBase::Load() {
370374
kLengthOffset = LoadConstant("class_FixedArrayBase__length__SMI");
@@ -521,6 +525,7 @@ void Frame::Load() {
521525
kConstructFrame = LoadConstant("frametype_ConstructFrame");
522526
kJSFrame = LoadConstant("frametype_JavaScriptFrame");
523527
kOptimizedFrame = LoadConstant("frametype_OptimizedFrame");
528+
kStubFrame = LoadConstant("frametype_StubFrame");
524529
}
525530

526531

src/llv8-constants.h

+12
Original file line numberDiff line numberDiff line change
@@ -256,6 +256,7 @@ class String : public Module {
256256
int64_t kConsStringTag;
257257
int64_t kSlicedStringTag;
258258
int64_t kExternalStringTag;
259+
int64_t kThinStringTag;
259260

260261
int64_t kLengthOffset;
261262

@@ -305,6 +306,16 @@ class SlicedString : public Module {
305306
void Load();
306307
};
307308

309+
class ThinString : public Module {
310+
public:
311+
MODULE_DEFAULT_METHODS(ThinString);
312+
313+
int64_t kActualOffset;
314+
315+
protected:
316+
void Load();
317+
};
318+
308319
class FixedArrayBase : public Module {
309320
public:
310321
MODULE_DEFAULT_METHODS(FixedArrayBase);
@@ -449,6 +460,7 @@ class Frame : public Module {
449460
int64_t kConstructFrame;
450461
int64_t kJSFrame;
451462
int64_t kOptimizedFrame;
463+
int64_t kStubFrame;
452464

453465
protected:
454466
void Load();

src/llv8-inl.h

+12
Original file line numberDiff line numberDiff line change
@@ -272,6 +272,8 @@ ACCESSOR(ConsString, Second, cons_string()->kSecondOffset, String);
272272
ACCESSOR(SlicedString, Parent, sliced_string()->kParentOffset, String);
273273
ACCESSOR(SlicedString, Offset, sliced_string()->kOffsetOffset, Smi);
274274

275+
ACCESSOR(ThinString, Actual, thin_string()->kActualOffset, String);
276+
275277
ACCESSOR(FixedArrayBase, Length, fixed_array_base()->kLengthOffset, Smi);
276278

277279
inline std::string OneByteString::ToString(Error& err) {
@@ -319,6 +321,16 @@ inline std::string SlicedString::ToString(Error& err) {
319321
return tmp.substr(offset.GetValue(), length.GetValue());
320322
}
321323

324+
inline std::string ThinString::ToString(Error& err) {
325+
String actual = Actual(err);
326+
if (err.Fail()) return std::string();
327+
328+
std::string tmp = actual.ToString(err);
329+
if (err.Fail()) return std::string();
330+
331+
return tmp;
332+
}
333+
322334
inline int64_t FixedArray::LeaData() const {
323335
return LeaField(v8()->fixed_array()->kDataOffset);
324336
}

src/llv8.cc

+15-7
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,5 @@
11
#include <assert.h>
22

3-
#include <algorithm>
43
#include <algorithm>
54
#include <cinttypes>
65

@@ -43,6 +42,7 @@ void LLV8::Load(SBTarget target) {
4342
two_byte_string.Assign(target, &common);
4443
cons_string.Assign(target, &common);
4544
sliced_string.Assign(target, &common);
45+
thin_string.Assign(target, &common);
4646
fixed_array_base.Assign(target, &common);
4747
fixed_array.Assign(target, &common);
4848
oddball.Assign(target, &common);
@@ -269,6 +269,8 @@ std::string JSFrame::Inspect(bool with_args, Error& err) {
269269
return "<internal>";
270270
} else if (value == v8()->frame()->kConstructFrame) {
271271
return "<constructor>";
272+
} else if (value == v8()->frame()->kStubFrame) {
273+
return "<stub>";
272274
} else if (value != v8()->frame()->kJSFrame &&
273275
value != v8()->frame()->kOptimizedFrame) {
274276
err = Error::Failure("Unknown frame marker");
@@ -958,6 +960,11 @@ std::string String::ToString(Error& err) {
958960
return std::string("(external)");
959961
}
960962

963+
if (repr == v8()->string()->kThinStringTag) {
964+
ThinString thin(this);
965+
return thin.ToString(err);
966+
}
967+
961968
err = Error::Failure("Unsupported string representation");
962969
return std::string();
963970
}
@@ -967,7 +974,7 @@ std::string String::Inspect(InspectOptions* options, Error& err) {
967974
std::string val = ToString(err);
968975
if (err.Fail()) return std::string();
969976

970-
unsigned int len = options->string_length;
977+
unsigned int len = options->length;
971978

972979
if (len != 0 && val.length() > len) val = val.substr(0, len) + "...";
973980

@@ -1120,7 +1127,7 @@ std::string JSArrayBuffer::Inspect(InspectOptions* options, Error& err) {
11201127
if (options->detailed) {
11211128
res += ": [\n ";
11221129

1123-
int display_length = std::min<int>(byte_length, options->array_length);
1130+
int display_length = std::min<int>(byte_length, options->length);
11241131
res += v8()->LoadBytes(display_length, data, err);
11251132

11261133
if (display_length < byte_length) {
@@ -1156,16 +1163,17 @@ std::string JSArrayBufferView::Inspect(InspectOptions* options, Error& err) {
11561163
int byte_length = static_cast<int>(length.GetValue());
11571164
int byte_offset = static_cast<int>(off.GetValue());
11581165
char tmp[128];
1159-
snprintf(tmp, sizeof(tmp), "<ArrayBufferView: backingStore=0x%016" PRIx64
1160-
", byteOffset=%d, byteLength=%d",
1166+
snprintf(tmp, sizeof(tmp),
1167+
"<ArrayBufferView: backingStore=0x%016" PRIx64
1168+
", byteOffset=%d, byteLength=%d",
11611169
data, byte_offset, byte_length);
11621170

11631171
std::string res;
11641172
res += tmp;
11651173
if (options->detailed) {
11661174
res += ": [\n ";
11671175

1168-
int display_length = std::min<int>(byte_length, options->array_length);
1176+
int display_length = std::min<int>(byte_length, options->length);
11691177
res += v8()->LoadBytes(display_length, data + byte_offset, err);
11701178

11711179
if (display_length < byte_length) {
@@ -1915,7 +1923,7 @@ std::string JSArray::Inspect(InspectOptions* options, Error& err) {
19151923

19161924
std::string res = "<Array: length=" + std::to_string(length);
19171925
if (options->detailed) {
1918-
int64_t display_length = std::min<int64_t>(length, options->array_length);
1926+
int64_t display_length = std::min<int64_t>(length, options->length);
19191927
std::string elems = InspectElements(display_length, err);
19201928
if (err.Fail()) return std::string();
19211929

src/llv8.h

+14-6
Original file line numberDiff line numberDiff line change
@@ -52,17 +52,14 @@ class Value {
5252
: detailed(false),
5353
print_map(false),
5454
print_source(false),
55-
string_length(kStringLength),
56-
array_length(kArrayLength) {}
55+
length(kLength) {}
5756

58-
static const unsigned int kStringLength = 16;
59-
static const unsigned int kArrayLength = 16;
57+
static const unsigned int kLength = 16;
6058

6159
bool detailed;
6260
bool print_map;
6361
bool print_source;
64-
unsigned int string_length;
65-
unsigned int array_length;
62+
unsigned int length;
6663
};
6764

6865
Value(const Value& v) = default;
@@ -222,6 +219,15 @@ class SlicedString : public String {
222219
inline std::string ToString(Error& err);
223220
};
224221

222+
class ThinString : public String {
223+
public:
224+
V8_VALUE_DEFAULT_METHODS(ThinString, String)
225+
226+
inline String Actual(Error& err);
227+
228+
inline std::string ToString(Error& err);
229+
};
230+
225231
class HeapNumber : public HeapObject {
226232
public:
227233
V8_VALUE_DEFAULT_METHODS(HeapNumber, HeapObject)
@@ -481,6 +487,7 @@ class LLV8 {
481487
constants::TwoByteString two_byte_string;
482488
constants::ConsString cons_string;
483489
constants::SlicedString sliced_string;
490+
constants::ThinString thin_string;
484491
constants::FixedArrayBase fixed_array_base;
485492
constants::FixedArray fixed_array;
486493
constants::Oddball oddball;
@@ -507,6 +514,7 @@ class LLV8 {
507514
friend class TwoByteString;
508515
friend class ConsString;
509516
friend class SlicedString;
517+
friend class ThinString;
510518
friend class HeapNumber;
511519
friend class JSObject;
512520
friend class JSArray;

test/fixtures/inspect-scenario.js

+10
Original file line numberDiff line numberDiff line change
@@ -8,6 +8,13 @@ let outerVar = 'outer variable';
88

99
exports.holder = {};
1010

11+
function makeThin(a, b) {
12+
var str = a + b;
13+
var obj = {};
14+
obj[str]; // Turn the cons string into a thin string.
15+
return str;
16+
}
17+
1118
function closure() {
1219

1320
function Class() {
@@ -28,6 +35,9 @@ function closure() {
2835
c.hashmap['cons-string'] =
2936
'this could be a bit smaller, but v8 wants big str.';
3037
c.hashmap['cons-string'] += c.hashmap['cons-string'];
38+
c.hashmap['internalized-string'] = 'foobar';
39+
// This thin string points to the previous 'foobar'.
40+
c.hashmap['thin-string'] = makeThin('foo', 'bar');
3141
c.hashmap['array'] = [true, 1, undefined, null, 'test', Class];
3242
c.hashmap['long-array'] = new Array(20).fill(5);
3343
c.hashmap['array-buffer'] = new Uint8Array(

0 commit comments

Comments
 (0)