Skip to content

Commit 6723156

Browse files
addaleaxcodebytere
authored andcommitted
process: report ArrayBuffer memory in memoryUsage()
Report memory allocations performed by the `ArrayBuffer::Allocator`. PR-URL: #31550 Reviewed-By: Richard Lau <[email protected]> Reviewed-By: Colin Ihrig <[email protected]> Reviewed-By: Gus Caplan <[email protected]> Reviewed-By: Ben Noordhuis <[email protected]> Reviewed-By: James M Snell <[email protected]> Reviewed-By: Rich Trott <[email protected]> Reviewed-By: Gireesh Punathil <[email protected]>
1 parent 6fdef45 commit 6723156

File tree

6 files changed

+80
-23
lines changed

6 files changed

+80
-23
lines changed

doc/api/process.md

+18-10
Original file line numberDiff line numberDiff line change
@@ -1482,6 +1482,9 @@ is no entry script.
14821482
<!-- YAML
14831483
added: v0.1.16
14841484
changes:
1485+
- version: REPLACEME
1486+
pr-url: https://github.com/nodejs/node/pull/31550
1487+
description: Added `arrayBuffers` to the returned object.
14851488
- version: v7.2.0
14861489
pr-url: https://github.com/nodejs/node/pull/9587
14871490
description: Added `external` to the returned object.
@@ -1492,6 +1495,7 @@ changes:
14921495
* `heapTotal` {integer}
14931496
* `heapUsed` {integer}
14941497
* `external` {integer}
1498+
* `arrayBuffers` {integer}
14951499

14961500
The `process.memoryUsage()` method returns an object describing the memory usage
14971501
of the Node.js process measured in bytes.
@@ -1510,19 +1514,22 @@ Will generate:
15101514
rss: 4935680,
15111515
heapTotal: 1826816,
15121516
heapUsed: 650472,
1513-
external: 49879
1517+
external: 49879,
1518+
arrayBuffers: 9386
15141519
}
15151520
```
15161521

1517-
`heapTotal` and `heapUsed` refer to V8's memory usage.
1518-
`external` refers to the memory usage of C++ objects bound to JavaScript
1519-
objects managed by V8. `rss`, Resident Set Size, is the amount of space
1520-
occupied in the main memory device (that is a subset of the total allocated
1521-
memory) for the process, which includes the _heap_, _code segment_ and _stack_.
1522-
1523-
The _heap_ is where objects, strings, and closures are stored. Variables are
1524-
stored in the _stack_ and the actual JavaScript code resides in the
1525-
_code segment_.
1522+
* `heapTotal` and `heapUsed` refer to V8's memory usage.
1523+
* `external` refers to the memory usage of C++ objects bound to JavaScript
1524+
objects managed by V8.
1525+
* `rss`, Resident Set Size, is the amount of space occupied in the main
1526+
memory device (that is a subset of the total allocated memory) for the
1527+
process, including all C++ and JavaScript objects and code.
1528+
* `arrayBuffers` refers to memory allocated for `ArrayBuffer`s and
1529+
`SharedArrayBuffer`s, including all Node.js [`Buffer`][]s.
1530+
This is also included in the `external` value. When Node.js is used as an
1531+
embedded library, this value may be `0` because allocations for `ArrayBuffer`s
1532+
may not be tracked in that case.
15261533

15271534
When using [`Worker`][] threads, `rss` will be a value that is valid for the
15281535
entire process, while the other fields will only refer to the current thread.
@@ -2490,6 +2497,7 @@ cases:
24902497
[`'exit'`]: #process_event_exit
24912498
[`'message'`]: child_process.html#child_process_event_message
24922499
[`'uncaughtException'`]: #process_event_uncaughtexception
2500+
[`Buffer`]: buffer.html
24932501
[`ChildProcess.disconnect()`]: child_process.html#child_process_subprocess_disconnect
24942502
[`ChildProcess.send()`]: child_process.html#child_process_subprocess_send_message_sendhandle_options_callback
24952503
[`ChildProcess`]: child_process.html#child_process_class_childprocess

lib/internal/process/per_thread.js

+3-2
Original file line numberDiff line numberDiff line change
@@ -146,14 +146,15 @@ function wrapProcessMethods(binding) {
146146
return hrBigintValues[0];
147147
}
148148

149-
const memValues = new Float64Array(4);
149+
const memValues = new Float64Array(5);
150150
function memoryUsage() {
151151
_memoryUsage(memValues);
152152
return {
153153
rss: memValues[0],
154154
heapTotal: memValues[1],
155155
heapUsed: memValues[2],
156-
external: memValues[3]
156+
external: memValues[3],
157+
arrayBuffers: memValues[4]
157158
};
158159
}
159160

src/api/environment.cc

+28-2
Original file line numberDiff line numberDiff line change
@@ -87,10 +87,34 @@ static void HostCleanupFinalizationGroupCallback(
8787
}
8888

8989
void* NodeArrayBufferAllocator::Allocate(size_t size) {
90+
void* ret;
9091
if (zero_fill_field_ || per_process::cli_options->zero_fill_all_buffers)
91-
return UncheckedCalloc(size);
92+
ret = UncheckedCalloc(size);
9293
else
93-
return UncheckedMalloc(size);
94+
ret = UncheckedMalloc(size);
95+
if (LIKELY(ret != nullptr))
96+
total_mem_usage_.fetch_add(size, std::memory_order_relaxed);
97+
return ret;
98+
}
99+
100+
void* NodeArrayBufferAllocator::AllocateUninitialized(size_t size) {
101+
void* ret = node::UncheckedMalloc(size);
102+
if (LIKELY(ret != nullptr))
103+
total_mem_usage_.fetch_add(size, std::memory_order_relaxed);
104+
return ret;
105+
}
106+
107+
void* NodeArrayBufferAllocator::Reallocate(
108+
void* data, size_t old_size, size_t size) {
109+
void* ret = UncheckedRealloc<char>(static_cast<char*>(data), size);
110+
if (LIKELY(ret != nullptr) || UNLIKELY(size == 0))
111+
total_mem_usage_.fetch_add(size - old_size, std::memory_order_relaxed);
112+
return ret;
113+
}
114+
115+
void NodeArrayBufferAllocator::Free(void* data, size_t size) {
116+
total_mem_usage_.fetch_sub(size, std::memory_order_relaxed);
117+
free(data);
94118
}
95119

96120
DebuggingArrayBufferAllocator::~DebuggingArrayBufferAllocator() {
@@ -140,11 +164,13 @@ void* DebuggingArrayBufferAllocator::Reallocate(void* data,
140164

141165
void DebuggingArrayBufferAllocator::RegisterPointer(void* data, size_t size) {
142166
Mutex::ScopedLock lock(mutex_);
167+
NodeArrayBufferAllocator::RegisterPointer(data, size);
143168
RegisterPointerInternal(data, size);
144169
}
145170

146171
void DebuggingArrayBufferAllocator::UnregisterPointer(void* data, size_t size) {
147172
Mutex::ScopedLock lock(mutex_);
173+
NodeArrayBufferAllocator::UnregisterPointer(data, size);
148174
UnregisterPointerInternal(data, size);
149175
}
150176

src/node_internals.h

+12-8
Original file line numberDiff line numberDiff line change
@@ -109,20 +109,24 @@ class NodeArrayBufferAllocator : public ArrayBufferAllocator {
109109
inline uint32_t* zero_fill_field() { return &zero_fill_field_; }
110110

111111
void* Allocate(size_t size) override; // Defined in src/node.cc
112-
void* AllocateUninitialized(size_t size) override
113-
{ return node::UncheckedMalloc(size); }
114-
void Free(void* data, size_t) override { free(data); }
115-
virtual void* Reallocate(void* data, size_t old_size, size_t size) {
116-
return static_cast<void*>(
117-
UncheckedRealloc<char>(static_cast<char*>(data), size));
112+
void* AllocateUninitialized(size_t size) override;
113+
void Free(void* data, size_t size) override;
114+
virtual void* Reallocate(void* data, size_t old_size, size_t size);
115+
virtual void RegisterPointer(void* data, size_t size) {
116+
total_mem_usage_.fetch_add(size, std::memory_order_relaxed);
117+
}
118+
virtual void UnregisterPointer(void* data, size_t size) {
119+
total_mem_usage_.fetch_sub(size, std::memory_order_relaxed);
118120
}
119-
virtual void RegisterPointer(void* data, size_t size) {}
120-
virtual void UnregisterPointer(void* data, size_t size) {}
121121

122122
NodeArrayBufferAllocator* GetImpl() final { return this; }
123+
inline uint64_t total_mem_usage() const {
124+
return total_mem_usage_.load(std::memory_order_relaxed);
125+
}
123126

124127
private:
125128
uint32_t zero_fill_field_ = 1; // Boolean but exposed as uint32 to JS land.
129+
std::atomic<size_t> total_mem_usage_ {0};
126130
};
127131

128132
class DebuggingArrayBufferAllocator final : public NodeArrayBufferAllocator {

src/node_process_methods.cc

+6-1
Original file line numberDiff line numberDiff line change
@@ -200,17 +200,22 @@ static void MemoryUsage(const FunctionCallbackInfo<Value>& args) {
200200
HeapStatistics v8_heap_stats;
201201
isolate->GetHeapStatistics(&v8_heap_stats);
202202

203+
NodeArrayBufferAllocator* array_buffer_allocator =
204+
env->isolate_data()->node_allocator();
205+
203206
// Get the double array pointer from the Float64Array argument.
204207
CHECK(args[0]->IsFloat64Array());
205208
Local<Float64Array> array = args[0].As<Float64Array>();
206-
CHECK_EQ(array->Length(), 4);
209+
CHECK_EQ(array->Length(), 5);
207210
Local<ArrayBuffer> ab = array->Buffer();
208211
double* fields = static_cast<double*>(ab->GetBackingStore()->Data());
209212

210213
fields[0] = rss;
211214
fields[1] = v8_heap_stats.total_heap_size();
212215
fields[2] = v8_heap_stats.used_heap_size();
213216
fields[3] = v8_heap_stats.external_memory();
217+
fields[4] = array_buffer_allocator == nullptr ?
218+
0 : array_buffer_allocator->total_mem_usage();
214219
}
215220

216221
void RawDebug(const FunctionCallbackInfo<Value>& args) {

test/parallel/test-memory-usage.js

+13
Original file line numberDiff line numberDiff line change
@@ -30,3 +30,16 @@ if (!common.isIBMi)
3030
assert.ok(r.heapTotal > 0);
3131
assert.ok(r.heapUsed > 0);
3232
assert.ok(r.external > 0);
33+
34+
assert.strictEqual(typeof r.arrayBuffers, 'number');
35+
if (r.arrayBuffers > 0) {
36+
const size = 10 * 1024 * 1024;
37+
// eslint-disable-next-line no-unused-vars
38+
const ab = new ArrayBuffer(size);
39+
40+
const after = process.memoryUsage();
41+
assert(after.external - r.external >= size,
42+
`${after.external} - ${r.external} >= ${size}`);
43+
assert.strictEqual(after.arrayBuffers - r.arrayBuffers, size,
44+
`${after.arrayBuffers} - ${r.arrayBuffers} >= ${size}`);
45+
}

0 commit comments

Comments
 (0)