Skip to content

Commit

Permalink
src,lib: add constrainedMemory API for process
Browse files Browse the repository at this point in the history
  • Loading branch information
theanarkh committed Jan 15, 2023
1 parent 5d50b84 commit 7b17275
Show file tree
Hide file tree
Showing 5 changed files with 53 additions and 0 deletions.
1 change: 1 addition & 0 deletions lib/internal/bootstrap/node.js
Original file line number Diff line number Diff line change
Expand Up @@ -184,6 +184,7 @@ const rawMethods = internalBinding('process_methods');

process.hrtime = perThreadSetup.hrtime;
process.hrtime.bigint = perThreadSetup.hrtimeBigInt;
process.constrainedMemory = perThreadSetup.constrainedMemory;

process.openStdin = function() {
process.stdin.resume();
Expand Down
8 changes: 8 additions & 0 deletions lib/internal/process/per_thread.js
Original file line number Diff line number Diff line change
Expand Up @@ -60,6 +60,7 @@ const binding = internalBinding('process_methods');

let hrValues;
let hrBigintValues;
let constrainedMemoryValus;

function refreshHrtimeBuffer() {
// The 3 entries filled in by the original process.hrtime contains
Expand All @@ -69,6 +70,7 @@ function refreshHrtimeBuffer() {
// Use a BigUint64Array in the closure because this is actually a bit
// faster than simply returning a BigInt from C++ in V8 7.1.
hrBigintValues = new BigUint64Array(binding.hrtimeBuffer, 0, 1);
constrainedMemoryValus = new BigUint64Array(binding.hrtimeBuffer, 0, 1);
}

// Create the buffers.
Expand Down Expand Up @@ -100,6 +102,11 @@ function hrtimeBigInt() {
return hrBigintValues[0];
}

function constrainedMemory() {
binding.constrainedMemory();
return constrainedMemoryValus[0];
}

function nop() {}

// The execution of this function itself should not cause any side effects.
Expand Down Expand Up @@ -427,4 +434,5 @@ module.exports = {
hrtime,
hrtimeBigInt,
refreshHrtimeBuffer,
constrainedMemory,
};
6 changes: 6 additions & 0 deletions src/node_process.h
Original file line number Diff line number Diff line change
Expand Up @@ -81,6 +81,11 @@ class BindingData : public SnapshotableObject {

static void SlowBigInt(const v8::FunctionCallbackInfo<v8::Value>& args);

static void ConstrainedMemoryImpl(BindingData* receiver);
static void SlowGetConstrainedMemory(
const v8::FunctionCallbackInfo<v8::Value>& args);
static void FastGetConstrainedMemory(v8::Local<v8::Value> receiver);

private:
static constexpr size_t kBufferSize =
std::max(sizeof(uint64_t), sizeof(uint32_t) * 3);
Expand All @@ -92,6 +97,7 @@ class BindingData : public SnapshotableObject {
// time.
static v8::CFunction fast_number_;
static v8::CFunction fast_bigint_;
static v8::CFunction fast_get_constrained_memory_;
};

} // namespace process
Expand Down
26 changes: 26 additions & 0 deletions src/node_process_methods.cc
Original file line number Diff line number Diff line change
Expand Up @@ -471,11 +471,17 @@ BindingData::BindingData(Environment* env, v8::Local<v8::Object> object)

v8::CFunction BindingData::fast_number_(v8::CFunction::Make(FastNumber));
v8::CFunction BindingData::fast_bigint_(v8::CFunction::Make(FastBigInt));
v8::CFunction BindingData::fast_get_constrained_memory_ =
v8::CFunction::Make(FastGetConstrainedMemory);

void BindingData::AddMethods() {
Local<Context> ctx = env()->context();
SetFastMethod(ctx, object(), "hrtime", SlowNumber, &fast_number_);
SetFastMethod(ctx, object(), "hrtimeBigInt", SlowBigInt, &fast_bigint_);
SetFastMethod(ctx, object(),
"constrainedMemory",
SlowGetConstrainedMemory,
&fast_get_constrained_memory_);
}

void BindingData::RegisterExternalReferences(
Expand All @@ -486,6 +492,9 @@ void BindingData::RegisterExternalReferences(
registry->Register(FastBigInt);
registry->Register(fast_number_.GetTypeInfo());
registry->Register(fast_bigint_.GetTypeInfo());
registry->Register(SlowGetConstrainedMemory);
registry->Register(FastGetConstrainedMemory);
registry->Register(fast_get_constrained_memory_.GetTypeInfo());
}

BindingData* BindingData::FromV8Value(Local<Value> value) {
Expand Down Expand Up @@ -533,6 +542,23 @@ void BindingData::SlowNumber(const v8::FunctionCallbackInfo<v8::Value>& args) {
NumberImpl(FromJSObject<BindingData>(args.Holder()));
}

void BindingData::ConstrainedMemoryImpl(BindingData* receiver) {
// Make sure we don't accidentally access buffers wiped for snapshot.
CHECK(!receiver->array_buffer_.IsEmpty());
uint64_t t = uv_get_constrained_memory();
uint64_t* fields = static_cast<uint64_t*>(receiver->backing_store_->Data());
fields[0] = t;
}

void BindingData::SlowGetConstrainedMemory(
const FunctionCallbackInfo<Value>& args) {
ConstrainedMemoryImpl(FromJSObject<BindingData>(args.Holder()));
}

void BindingData::FastGetConstrainedMemory(v8::Local<v8::Value> receiver) {
ConstrainedMemoryImpl(FromV8Value(receiver));
}

bool BindingData::PrepareForSerialization(Local<Context> context,
v8::SnapshotCreator* creator) {
// It's not worth keeping.
Expand Down
12 changes: 12 additions & 0 deletions test/sequential/test-process-constrained-memory.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,12 @@
'use strict';
require('../common');
const assert = require('assert');
const { Worker } = require('worker_threads');

if (!process.env.isWorker) {
process.env.isWorker = true;
new Worker(__filename);
assert(process.constrainedMemory() >= 0);
} else {
assert(process.constrainedMemory() >= 0);
}

0 comments on commit 7b17275

Please sign in to comment.