-
Notifications
You must be signed in to change notification settings - Fork 29.6k
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
src,test: ensure that V8 fast APIs are called
Adds a debug-only macro that can be used to track when a V8 fast API is called. A map of counters is maintained in in thread-local storage and an internal API can be called to get the total count associated with a call id. Specific tests are added and `crypto.timingSafeEqual` as well as internal documentation are updated to show how to use the macro and test fast API calls without running long loops. PR-URL: #54317 Reviewed-By: Robert Nagy <[email protected]> Reviewed-By: James M Snell <[email protected]> Reviewed-By: Yagiz Nizipli <[email protected]> Reviewed-By: Tobias Nießen <[email protected]>
- Loading branch information
Showing
12 changed files
with
284 additions
and
3 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,101 @@ | ||
#include "node_debug.h" | ||
|
||
#ifdef DEBUG | ||
#include "node_binding.h" | ||
|
||
#include "env-inl.h" | ||
#include "util.h" | ||
#include "v8-fast-api-calls.h" | ||
#include "v8.h" | ||
|
||
#include <string_view> | ||
#include <unordered_map> | ||
#endif // DEBUG | ||
|
||
namespace node { | ||
namespace debug { | ||
|
||
#ifdef DEBUG | ||
using v8::Context; | ||
using v8::FastApiCallbackOptions; | ||
using v8::FunctionCallbackInfo; | ||
using v8::Local; | ||
using v8::Number; | ||
using v8::Object; | ||
using v8::Value; | ||
|
||
thread_local std::unordered_map<std::string_view, int> v8_fast_api_call_counts; | ||
|
||
void TrackV8FastApiCall(std::string_view key) { | ||
v8_fast_api_call_counts[key]++; | ||
} | ||
|
||
int GetV8FastApiCallCount(std::string_view key) { | ||
return v8_fast_api_call_counts[key]; | ||
} | ||
|
||
void GetV8FastApiCallCount(const FunctionCallbackInfo<Value>& args) { | ||
Environment* env = Environment::GetCurrent(args); | ||
if (!args[0]->IsString()) { | ||
env->ThrowError("getV8FastApiCallCount must be called with a string"); | ||
return; | ||
} | ||
Utf8Value utf8_key(env->isolate(), args[0]); | ||
args.GetReturnValue().Set(GetV8FastApiCallCount(utf8_key.ToString())); | ||
} | ||
|
||
void SlowIsEven(const FunctionCallbackInfo<Value>& args) { | ||
Environment* env = Environment::GetCurrent(args); | ||
if (!args[0]->IsNumber()) { | ||
env->ThrowError("isEven must be called with a number"); | ||
return; | ||
} | ||
int64_t value = args[0].As<Number>()->Value(); | ||
args.GetReturnValue().Set(value % 2 == 0); | ||
} | ||
|
||
bool FastIsEven(Local<Value> receiver, | ||
const int64_t value, | ||
// NOLINTNEXTLINE(runtime/references) | ||
FastApiCallbackOptions& options) { | ||
TRACK_V8_FAST_API_CALL("debug.isEven"); | ||
return value % 2 == 0; | ||
} | ||
|
||
void SlowIsOdd(const FunctionCallbackInfo<Value>& args) { | ||
Environment* env = Environment::GetCurrent(args); | ||
if (!args[0]->IsNumber()) { | ||
env->ThrowError("isOdd must be called with a number"); | ||
return; | ||
} | ||
int64_t value = args[0].As<Number>()->Value(); | ||
args.GetReturnValue().Set(value % 2 != 0); | ||
} | ||
|
||
bool FastIsOdd(Local<Value> receiver, | ||
const int64_t value, | ||
// NOLINTNEXTLINE(runtime/references) | ||
FastApiCallbackOptions& options) { | ||
TRACK_V8_FAST_API_CALL("debug.isOdd"); | ||
return value % 2 != 0; | ||
} | ||
|
||
static v8::CFunction fast_is_even(v8::CFunction::Make(FastIsEven)); | ||
static v8::CFunction fast_is_odd(v8::CFunction::Make(FastIsOdd)); | ||
|
||
void Initialize(Local<Object> target, | ||
Local<Value> unused, | ||
Local<Context> context, | ||
void* priv) { | ||
SetMethod(context, target, "getV8FastApiCallCount", GetV8FastApiCallCount); | ||
SetFastMethod(context, target, "isEven", SlowIsEven, &fast_is_even); | ||
SetFastMethod(context, target, "isOdd", SlowIsOdd, &fast_is_odd); | ||
} | ||
#endif // DEBUG | ||
|
||
} // namespace debug | ||
} // namespace node | ||
|
||
#ifdef DEBUG | ||
NODE_BINDING_CONTEXT_AWARE_INTERNAL(debug, node::debug::Initialize) | ||
#endif // DEBUG |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,24 @@ | ||
#pragma once | ||
|
||
#if defined(NODE_WANT_INTERNALS) && NODE_WANT_INTERNALS | ||
|
||
#ifdef DEBUG | ||
#include <string_view> | ||
#endif // DEBUG | ||
|
||
namespace node { | ||
namespace debug { | ||
|
||
#ifdef DEBUG | ||
void TrackV8FastApiCall(std::string_view key); | ||
int GetV8FastApiCallCount(std::string_view key); | ||
|
||
#define TRACK_V8_FAST_API_CALL(key) node::debug::TrackV8FastApiCall(key) | ||
#else // !DEBUG | ||
#define TRACK_V8_FAST_API_CALL(key) | ||
#endif // DEBUG | ||
|
||
} // namespace debug | ||
} // namespace node | ||
|
||
#endif // defined(NODE_WANT_INTERNALS) && NODE_WANT_INTERNALS |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,59 @@ | ||
// Flags: --expose-internals --no-warnings --allow-natives-syntax | ||
'use strict'; | ||
const common = require('../common'); | ||
|
||
const assert = require('assert'); | ||
const { internalBinding } = require('internal/test/binding'); | ||
|
||
if (!common.isDebug) { | ||
assert.throws(() => internalBinding('debug'), { | ||
message: 'No such binding: debug' | ||
}); | ||
return; | ||
} | ||
|
||
const { | ||
getV8FastApiCallCount, | ||
isEven, | ||
isOdd, | ||
} = internalBinding('debug'); | ||
|
||
assert.throws(() => getV8FastApiCallCount(), { | ||
message: 'getV8FastApiCallCount must be called with a string', | ||
}); | ||
|
||
function testIsEven() { | ||
for (let i = 0; i < 10; i++) { | ||
assert.strictEqual(isEven(i), i % 2 === 0); | ||
} | ||
} | ||
|
||
function testIsOdd() { | ||
for (let i = 0; i < 20; i++) { | ||
assert.strictEqual(isOdd(i), i % 2 !== 0); | ||
} | ||
} | ||
|
||
// Should return 0 by default for any string. | ||
assert.strictEqual(getV8FastApiCallCount(''), 0); | ||
assert.strictEqual(getV8FastApiCallCount('foo'), 0); | ||
assert.strictEqual(getV8FastApiCallCount('debug.isEven'), 0); | ||
assert.strictEqual(getV8FastApiCallCount('debug.isOdd'), 0); | ||
|
||
eval('%PrepareFunctionForOptimization(testIsEven)'); | ||
testIsEven(); | ||
eval('%PrepareFunctionForOptimization(testIsOdd)'); | ||
testIsOdd(); | ||
|
||
// Functions should not be optimized yet. | ||
assert.strictEqual(getV8FastApiCallCount('debug.isEven'), 0); | ||
assert.strictEqual(getV8FastApiCallCount('debug.isOdd'), 0); | ||
|
||
eval('%OptimizeFunctionOnNextCall(testIsEven)'); | ||
testIsEven(); | ||
eval('%OptimizeFunctionOnNextCall(testIsOdd)'); | ||
testIsOdd(); | ||
|
||
// Functions should have been optimized and fast path taken. | ||
assert.strictEqual(getV8FastApiCallCount('debug.isEven'), 10); | ||
assert.strictEqual(getV8FastApiCallCount('debug.isOdd'), 20); |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Oops, something went wrong.