// Copyright 2015 the V8 project authors. All rights reserved. // Use of this source code is governed by a BSD-style license that can be // found in the LICENSE file. #ifndef V8_TEST_CCTEST_TEST_API_H_ #define V8_TEST_CCTEST_TEST_API_H_ #include "src/api/api.h" #include "src/execution/isolate.h" #include "src/execution/vm-state.h" #include "test/cctest/cctest.h" template static void CheckReturnValue(const T& info, i::Address callback) { v8::ReturnValue returnValue = info.GetReturnValue(); i::FullObjectSlot returnObjectSlot( *reinterpret_cast(&returnValue)); CHECK_EQ(CcTest::isolate(), info.GetIsolate()); i::Isolate* isolate = reinterpret_cast(info.GetIsolate()); CHECK_EQ(info.GetIsolate(), returnValue.GetIsolate()); CHECK((*returnObjectSlot).IsTheHole(isolate) || (*returnObjectSlot).IsUndefined(isolate)); // Verify reset bool is_runtime = (*returnObjectSlot).IsTheHole(isolate); if (is_runtime) { CHECK(returnValue.Get()->IsUndefined()); } else { i::Handle v = v8::Utils::OpenHandle(*returnValue.Get()); CHECK_EQ(*v, *returnObjectSlot); } returnValue.Set(true); CHECK_EQ(*returnObjectSlot, i::ReadOnlyRoots(isolate).true_value()); returnValue.Set(v8::Local()); CHECK((*returnObjectSlot).IsTheHole(isolate) || (*returnObjectSlot).IsUndefined(isolate)); // If CPU profiler is active check that when API callback is invoked // VMState is set to EXTERNAL. if (isolate->is_profiling()) { CHECK_EQ(v8::EXTERNAL, isolate->current_vm_state()); CHECK(isolate->external_callback_scope()); CHECK_EQ(callback, isolate->external_callback_scope()->callback()); } } template static void CheckInternalFieldsAreZero(v8::Local value) { CHECK_EQ(T::kInternalFieldCount, value->InternalFieldCount()); for (int i = 0; i < value->InternalFieldCount(); i++) { CHECK_EQ(0, value->GetInternalField(i) ->Int32Value(CcTest::isolate()->GetCurrentContext()) .FromJust()); } } template struct ConvertJSValue { static v8::Maybe Get(v8::Local value, v8::Local context); }; template <> struct ConvertJSValue { static v8::Maybe Get(v8::Local value, v8::Local context) { return value->Int32Value(context); } }; template <> struct ConvertJSValue { static v8::Maybe Get(v8::Local value, v8::Local context) { return value->Uint32Value(context); } }; template <> struct ConvertJSValue { static v8::Maybe Get(v8::Local value, v8::Local context) { return value->IsNull() ? v8::Just(nullptr) : v8::Nothing(); } }; // NaNs and +/-Infinity should be 0, otherwise (modulo 2^64) - 2^63. // Step 8 - 12 of https://heycam.github.io/webidl/#abstract-opdef-converttoint // The int64_t and uint64_t implementations below are copied from Blink: // https://source.chromium.org/chromium/chromium/src/+/master:third_party/blink/renderer/bindings/core/v8/v8_binding_for_core.h;l=249?q=doubletointeger&sq=&ss=chromium%2Fchromium%2Fsrc template <> struct ConvertJSValue { static v8::Maybe Get(v8::Local value, v8::Local context) { v8::Maybe double_value = value->NumberValue(context); if (!double_value.IsJust()) { return v8::Nothing(); } double result = double_value.ToChecked(); if (std::isinf(result) || std::isnan(result)) { return v8::Just(int64_t(0)); } result = trunc(result); constexpr uint64_t kMaxULL = std::numeric_limits::max(); // -2^{64} < fmod_value < 2^{64}. double fmod_value = fmod(result, static_cast(kMaxULL)); if (fmod_value >= 0) { if (fmod_value < pow(2, 63)) { // 0 <= fmod_value < 2^{63}. // 0 <= value < 2^{63}. This cast causes no loss. return v8::Just(static_cast(fmod_value)); } else { // 2^{63} <= fmod_value < 2^{64}. // 2^{63} <= value < 2^{64}. This cast causes no loss. return v8::Just(static_cast(fmod_value - pow(2, 64))); } } // -2^{64} < fmod_value < 0. // 0 < fmod_value_uint64 < 2^{64}. This cast causes no loss. uint64_t fmod_value_uint64 = static_cast(-fmod_value); // -1 < (kMaxULL - fmod_value_uint64) < 2^{64} - 1. // 0 < value < 2^{64}. return v8::Just(static_cast(kMaxULL - fmod_value_uint64 + 1)); } }; template <> struct ConvertJSValue { static v8::Maybe Get(v8::Local value, v8::Local context) { v8::Maybe double_value = value->NumberValue(context); if (!double_value.IsJust()) { return v8::Nothing(); } double result = double_value.ToChecked(); if (std::isinf(result) || std::isnan(result)) { return v8::Just(uint64_t(0)); } result = trunc(result); constexpr uint64_t kMaxULL = std::numeric_limits::max(); // -2^{64} < fmod_value < 2^{64}. double fmod_value = fmod(result, static_cast(kMaxULL)); if (fmod_value >= 0) { return v8::Just(static_cast(fmod_value)); } // -2^{64} < fmod_value < 0. // 0 < fmod_value_uint64 < 2^{64}. This cast causes no loss. uint64_t fmod_value_uint64 = static_cast(-fmod_value); // -1 < (kMaxULL - fmod_value_uint64) < 2^{64} - 1. // 0 < value < 2^{64}. return v8::Just(static_cast(kMaxULL - fmod_value_uint64 + 1)); } }; template <> struct ConvertJSValue { static v8::Maybe> Get(v8::Local value, v8::Local context) { if (value->IsBigInt()) { return v8::Just(value.As()); } return v8::Nothing>(); } }; template <> struct ConvertJSValue { static v8::Maybe Get(v8::Local value, v8::Local context) { v8::Maybe val = value->NumberValue(context); if (val.IsNothing()) return v8::Nothing(); return v8::Just(static_cast(val.ToChecked())); } }; template <> struct ConvertJSValue { static v8::Maybe Get(v8::Local value, v8::Local context) { return value->NumberValue(context); } }; template <> struct ConvertJSValue { static v8::Maybe Get(v8::Local value, v8::Local context) { return v8::Just(value->BooleanValue(CcTest::isolate())); } }; #endif // V8_TEST_CCTEST_TEST_API_H_