// 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. #include "include/v8-context.h" #include "include/v8-function.h" #include "include/v8-isolate.h" #include "include/v8-local-handle.h" #include "include/v8-object.h" #include "include/v8-template.h" #include "src/api/api-inl.h" #include "src/debug/debug.h" #include "src/execution/isolate.h" #include "src/objects/objects-inl.h" #include "test/unittests/test-utils.h" #include "testing/gmock-support.h" #include "testing/gtest/include/gtest/gtest.h" namespace i = v8::internal; using testing::IsInt32; using testing::IsString; using testing::IsUndefined; using AccessorTest = v8::TestWithContext; // The goal is to avoid the callback. static void UnreachableCallback( const v8::FunctionCallbackInfo& info) { UNREACHABLE(); } TEST_F(AccessorTest, CachedAccessor) { // TurboFan support for fast accessors is not implemented; turbofanned // code uses the slow accessor which breaks this test's expectations. v8::internal::FLAG_always_turbofan = false; v8::Isolate* isolate = context()->GetIsolate(); v8::HandleScope scope(isolate); // Create 'foo' class, with a hidden property. v8::Local foo = v8::ObjectTemplate::New(isolate); v8::Local priv = v8::Private::ForApi(isolate, NewString("Foo#draft")); foo->SetAccessorProperty( NewString("draft"), v8::FunctionTemplate::NewWithCache(isolate, UnreachableCallback, priv, v8::Local())); // Create 'obj', instance of 'foo'. v8::Local obj = foo->NewInstance(context()).ToLocalChecked(); // Install the private property on the instance. CHECK(obj->SetPrivate(isolate->GetCurrentContext(), priv, v8::Undefined(isolate)) .FromJust()); CHECK(context()->Global()->Set(context(), NewString("obj"), obj).FromJust()); // Access cached accessor. EXPECT_THAT(RunJS("obj.draft"), IsUndefined()); // Set hidden property. CHECK(obj->SetPrivate(isolate->GetCurrentContext(), priv, NewString("Shhh, I'm private!")) .FromJust()); EXPECT_THAT(RunJS("obj.draft"), IsString("Shhh, I'm private!")); // Stress the accessor to use the IC. EXPECT_THAT(RunJS("var result = '';" "for (var i = 0; i < 10; ++i) { " " result = obj.draft; " "} " "result; "), IsString("Shhh, I'm private!")); } TEST_F(AccessorTest, CachedAccessorTurboFan) { i::FLAG_allow_natives_syntax = true; // v8::internal::FLAG_always_turbofan = false; v8::Isolate* isolate = context()->GetIsolate(); v8::HandleScope scope(isolate); // Create 'foo' class, with a hidden property. v8::Local foo = v8::ObjectTemplate::New(isolate); v8::Local priv = v8::Private::ForApi(isolate, NewString("Foo#draft")); // Install the private property on the template. // foo->SetPrivate(priv, v8::Undefined(isolate)); foo->SetAccessorProperty( NewString("draft"), v8::FunctionTemplate::NewWithCache(isolate, UnreachableCallback, priv, v8::Local())); // Create 'obj', instance of 'foo'. v8::Local obj = foo->NewInstance(context()).ToLocalChecked(); // Install the private property on the instance. CHECK(obj->SetPrivate(isolate->GetCurrentContext(), priv, v8::Undefined(isolate)) .FromJust()); CHECK(context()->Global()->Set(context(), NewString("obj"), obj).FromJust()); // Access surrogate accessor. EXPECT_THAT(RunJS("obj.draft"), IsUndefined()); // Set hidden property. CHECK(obj->SetPrivate(context(), priv, v8::Integer::New(isolate, 123)) .FromJust()); // Test ICs. RunJS( "function f() {" " var x;" " for (var i = 0; i < 100; i++) {" " x = obj.draft;" " }" " return x;" "};" "%PrepareFunctionForOptimization(f);"); EXPECT_THAT(RunJS("f()"), IsInt32(123)); // Reset hidden property. CHECK(obj->SetPrivate(context(), priv, v8::Integer::New(isolate, 456)) .FromJust()); // Test TurboFan. RunJS("%OptimizeFunctionOnNextCall(f);"); EXPECT_THAT(RunJS("f()"), IsInt32(456)); CHECK(obj->SetPrivate(context(), priv, v8::Integer::New(isolate, 456)) .FromJust()); // Test non-global ICs. RunJS( "function g() {" " var x = obj;" " var r = 0;" " for (var i = 0; i < 100; i++) {" " r = x.draft;" " }" " return r;" "};" "%PrepareFunctionForOptimization(g);"); EXPECT_THAT(RunJS("g()"), IsInt32(456)); // Reset hidden property. CHECK(obj->SetPrivate(context(), priv, v8::Integer::New(isolate, 789)) .FromJust()); // Test non-global access in TurboFan. RunJS("%OptimizeFunctionOnNextCall(g);"); EXPECT_THAT(RunJS("g()"), IsInt32(789)); } TEST_F(AccessorTest, CachedAccessorOnGlobalObject) { i::FLAG_allow_natives_syntax = true; v8::HandleScope scope(isolate()); v8::Local templ = v8::FunctionTemplate::New(isolate()); v8::Local object_template = templ->InstanceTemplate(); v8::Local priv = v8::Private::ForApi(isolate(), NewString("Foo#draft")); object_template->SetAccessorProperty( NewString("draft"), v8::FunctionTemplate::NewWithCache(isolate(), UnreachableCallback, priv, v8::Local())); v8::Local ctx = v8::Context::New(isolate(), nullptr, object_template); v8::Local obj = ctx->Global(); // Install the private property on the instance. CHECK(obj->SetPrivate(isolate()->GetCurrentContext(), priv, v8::Undefined(isolate())) .FromJust()); { v8::Context::Scope context_scope(ctx); // Access surrogate accessor. EXPECT_THAT(RunJS("draft"), IsUndefined()); // Set hidden property. CHECK(obj->SetPrivate(context(), priv, v8::Integer::New(isolate(), 123)) .FromJust()); // Test ICs. RunJS( "function f() {" " var x;" " for (var i = 0; i < 100; i++) {" " x = draft;" " }" " return x;" "}" "%PrepareFunctionForOptimization(f);"); EXPECT_THAT(RunJS("f()"), IsInt32(123)); // Reset hidden property. CHECK(obj->SetPrivate(context(), priv, v8::Integer::New(isolate(), 456)) .FromJust()); // Test TurboFan. RunJS("%OptimizeFunctionOnNextCall(f);"); EXPECT_THAT(RunJS("f()"), IsInt32(456)); CHECK(obj->SetPrivate(context(), priv, v8::Integer::New(isolate(), 456)) .FromJust()); // Test non-global ICs. RunJS( "var x = this;" "function g() {" " var r = 0;" " for (var i = 0; i < 100; i++) {" " r = x.draft;" " }" " return r;" "}" "%PrepareFunctionForOptimization(g);"); EXPECT_THAT(RunJS("g()"), IsInt32(456)); // Reset hidden property. CHECK(obj->SetPrivate(context(), priv, v8::Integer::New(isolate(), 789)) .FromJust()); // Test non-global access in TurboFan. RunJS("%OptimizeFunctionOnNextCall(g);"); EXPECT_THAT(RunJS("g()"), IsInt32(789)); } } namespace { // Getter return value should be non-null to trigger lazy property paths. static void Getter(v8::Local name, const v8::PropertyCallbackInfo& info) { info.GetReturnValue().Set( v8::String::NewFromUtf8(info.GetIsolate(), "return value") .ToLocalChecked()); } static void StringGetter(v8::Local name, const v8::PropertyCallbackInfo& info) {} static int set_accessor_call_count = 0; static void Setter(v8::Local name, v8::Local value, const v8::PropertyCallbackInfo& info) { set_accessor_call_count++; } static void EmptyCallback(const v8::FunctionCallbackInfo& info) {} } // namespace // Re-declaration of non-configurable accessors should throw. TEST_F(AccessorTest, RedeclareAccessor) { v8::HandleScope scope(isolate()); v8::Local templ = v8::FunctionTemplate::New(isolate()); v8::Local object_template = templ->InstanceTemplate(); object_template->SetAccessor( NewString("foo"), nullptr, Setter, v8::Local(), v8::AccessControl::DEFAULT, v8::PropertyAttribute::DontDelete); v8::Local ctx = v8::Context::New(isolate(), nullptr, object_template); // Declare function. v8::Local code = NewString("function foo() {};"); v8::TryCatch try_catch(isolate()); v8::Script::Compile(ctx, code).ToLocalChecked()->Run(ctx).IsEmpty(); CHECK(try_catch.HasCaught()); } class NoopDelegate : public v8::debug::DebugDelegate {}; static void CheckSideEffectFreeAccesses(v8::Isolate* isolate, v8::Local call_getter, v8::Local call_setter) { const int kIterationsCountForICProgression = 20; v8::Local context = isolate->GetCurrentContext(); i::Isolate* i_isolate = reinterpret_cast(isolate); NoopDelegate delegate; i_isolate->debug()->SetDebugDelegate(&delegate); v8::Local func = v8::Script::Compile(context, call_getter).ToLocalChecked(); // Check getter. Run enough number of times to ensure IC creates data handler. for (int i = 0; i < kIterationsCountForICProgression; i++) { v8::TryCatch try_catch(isolate); CHECK(EvaluateGlobalForTesting( isolate, func, v8::debug::EvaluateGlobalMode::kDisableBreaksAndThrowOnSideEffect, true) .IsEmpty()); CHECK(try_catch.HasCaught()); // Ensure that IC state progresses. CHECK(!func->Run(context).IsEmpty()); } func = v8::Script::Compile(context, call_setter).ToLocalChecked(); // Check setter. Run enough number of times to ensure IC creates data handler. for (int i = 0; i < kIterationsCountForICProgression; i++) { v8::TryCatch try_catch(isolate); CHECK(EvaluateGlobalForTesting( isolate, func, v8::debug::EvaluateGlobalMode::kDisableBreaksAndThrowOnSideEffect, true) .IsEmpty()); CHECK(try_catch.HasCaught()); // Ensure that IC state progresses. CHECK(!func->Run(context).IsEmpty()); } } TEST_F(AccessorTest, AccessorsWithSideEffects) { v8::HandleScope scope(isolate()); i::Isolate* i_isolate = reinterpret_cast(isolate()); NoopDelegate delegate; i_isolate->debug()->SetDebugDelegate(&delegate); v8::Local templ = v8::ObjectTemplate::New(isolate()); v8::Local obj = templ->NewInstance(context()).ToLocalChecked(); CHECK(context()->Global()->Set(context(), NewString("obj"), obj).FromJust()); v8::Local templ_with_sideffect = v8::FunctionTemplate::New( isolate(), EmptyCallback, v8::Local(), v8::Local(), 0, v8::ConstructorBehavior::kAllow, v8::SideEffectType::kHasSideEffect); v8::Local templ_no_sideffect = v8::FunctionTemplate::New( isolate(), EmptyCallback, v8::Local(), v8::Local(), 0, v8::ConstructorBehavior::kAllow, v8::SideEffectType::kHasNoSideEffect); // Install non-native properties with side effects obj->SetAccessorProperty( NewString("get"), templ_with_sideffect->GetFunction(context()).ToLocalChecked(), {}, v8::PropertyAttribute::None, v8::AccessControl::DEFAULT); obj->SetAccessorProperty( NewString("set"), templ_no_sideffect->GetFunction(context()).ToLocalChecked(), templ_with_sideffect->GetFunction(context()).ToLocalChecked(), v8::PropertyAttribute::None, v8::AccessControl::DEFAULT); CheckSideEffectFreeAccesses(isolate(), NewString("obj.get"), NewString("obj.set = 123;")); } TEST_F(AccessorTest, TemplateAccessorsWithSideEffects) { v8::HandleScope scope(isolate()); i::Isolate* i_isolate = reinterpret_cast(isolate()); NoopDelegate delegate; i_isolate->debug()->SetDebugDelegate(&delegate); v8::Local templ_with_sideffect = v8::FunctionTemplate::New( isolate(), EmptyCallback, v8::Local(), v8::Local(), 0, v8::ConstructorBehavior::kAllow, v8::SideEffectType::kHasSideEffect); v8::Local templ_no_sideffect = v8::FunctionTemplate::New( isolate(), EmptyCallback, v8::Local(), v8::Local(), 0, v8::ConstructorBehavior::kAllow, v8::SideEffectType::kHasNoSideEffect); v8::Local templ = v8::ObjectTemplate::New(isolate()); templ->SetAccessorProperty(NewString("get"), templ_with_sideffect); templ->SetAccessorProperty(NewString("set"), templ_no_sideffect, templ_with_sideffect); v8::Local obj = templ->NewInstance(context()).ToLocalChecked(); CHECK(context()->Global()->Set(context(), NewString("obj"), obj).FromJust()); CheckSideEffectFreeAccesses(isolate(), NewString("obj.get"), NewString("obj.set = 123;")); } TEST_F(AccessorTest, NativeTemplateAccessorWithSideEffects) { v8::HandleScope scope(isolate()); i::Isolate* i_isolate = reinterpret_cast(isolate()); NoopDelegate delegate; i_isolate->debug()->SetDebugDelegate(&delegate); v8::Local templ = v8::ObjectTemplate::New(isolate()); templ->SetAccessor(NewString("get"), Getter, nullptr, v8::Local(), v8::AccessControl::DEFAULT, v8::PropertyAttribute::None, v8::SideEffectType::kHasSideEffect); templ->SetAccessor(NewString("set"), Getter, Setter, v8::Local(), v8::AccessControl::DEFAULT, v8::PropertyAttribute::None, v8::SideEffectType::kHasNoSideEffect, v8::SideEffectType::kHasSideEffect); v8::Local obj = templ->NewInstance(context()).ToLocalChecked(); CHECK(context()->Global()->Set(context(), NewString("obj"), obj).FromJust()); CheckSideEffectFreeAccesses(isolate(), NewString("obj.get"), NewString("obj.set = 123;")); } TEST_F(AccessorTest, NativeAccessorsWithSideEffects) { v8::HandleScope scope(isolate()); v8::Local templ = v8::ObjectTemplate::New(isolate()); v8::Local obj = templ->NewInstance(context()).ToLocalChecked(); CHECK(context()->Global()->Set(context(), NewString("obj"), obj).FromJust()); // Install native data property with side effects. obj->SetAccessor(context(), NewString("get"), Getter, nullptr, v8::MaybeLocal(), v8::AccessControl::DEFAULT, v8::PropertyAttribute::None, v8::SideEffectType::kHasSideEffect) .ToChecked(); obj->SetAccessor(context(), NewString("set"), Getter, Setter, v8::MaybeLocal(), v8::AccessControl::DEFAULT, v8::PropertyAttribute::None, v8::SideEffectType::kHasNoSideEffect, v8::SideEffectType::kHasSideEffect) .ToChecked(); CheckSideEffectFreeAccesses(isolate(), NewString("obj.get"), NewString("obj.set = 123;")); } // Accessors can be allowlisted as side-effect-free via SetAccessor. TEST_F(AccessorTest, AccessorSetHasNoSideEffect) { v8::HandleScope scope(isolate()); i::Isolate* i_isolate = reinterpret_cast(isolate()); NoopDelegate delegate; i_isolate->debug()->SetDebugDelegate(&delegate); v8::Local templ = v8::ObjectTemplate::New(isolate()); v8::Local obj = templ->NewInstance(context()).ToLocalChecked(); CHECK(context()->Global()->Set(context(), NewString("obj"), obj).FromJust()); obj->SetAccessor(context(), NewString("foo"), Getter).ToChecked(); CHECK(v8::debug::EvaluateGlobal( isolate(), NewString("obj.foo"), v8::debug::EvaluateGlobalMode::kDisableBreaksAndThrowOnSideEffect) .IsEmpty()); obj->SetAccessor(context(), NewString("foo"), Getter, nullptr, v8::MaybeLocal(), v8::AccessControl::DEFAULT, v8::PropertyAttribute::None, v8::SideEffectType::kHasNoSideEffect) .ToChecked(); v8::debug::EvaluateGlobal( isolate(), NewString("obj.foo"), v8::debug::EvaluateGlobalMode::kDisableBreaksAndThrowOnSideEffect) .ToLocalChecked(); // Check that setter is not allowlisted. v8::TryCatch try_catch(isolate()); CHECK(v8::debug::EvaluateGlobal( isolate(), NewString("obj.foo = 1"), v8::debug::EvaluateGlobalMode::kDisableBreaksAndThrowOnSideEffect) .IsEmpty()); CHECK(try_catch.HasCaught()); CHECK_NE(1, v8::debug::EvaluateGlobal(isolate(), NewString("obj.foo"), v8::debug::EvaluateGlobalMode::kDefault) .ToLocalChecked() ->Int32Value(context()) .FromJust()); CHECK_EQ(0, set_accessor_call_count); } // Set accessors can be allowlisted as side-effect-free via SetAccessor. TEST_F(AccessorTest, SetAccessorSetSideEffectReceiverCheck1) { v8::HandleScope scope(isolate()); i::Isolate* i_isolate = reinterpret_cast(isolate()); NoopDelegate delegate; i_isolate->debug()->SetDebugDelegate(&delegate); v8::Local templ = v8::ObjectTemplate::New(isolate()); v8::Local obj = templ->NewInstance(context()).ToLocalChecked(); CHECK(context()->Global()->Set(context(), NewString("obj"), obj).FromJust()); obj->SetAccessor(context(), NewString("foo"), Getter, Setter, v8::MaybeLocal(), v8::AccessControl::DEFAULT, v8::PropertyAttribute::None, v8::SideEffectType::kHasNoSideEffect, v8::SideEffectType::kHasSideEffectToReceiver) .ToChecked(); CHECK(v8::debug::EvaluateGlobal( isolate(), NewString("obj.foo"), v8::debug::EvaluateGlobalMode::kDisableBreaksAndThrowOnSideEffect) .ToLocalChecked() ->Equals(context(), NewString("return value")) .FromJust()); v8::TryCatch try_catch(isolate()); CHECK(v8::debug::EvaluateGlobal( isolate(), NewString("obj.foo = 1"), v8::debug::EvaluateGlobalMode::kDisableBreaksAndThrowOnSideEffect) .IsEmpty()); CHECK(try_catch.HasCaught()); CHECK_EQ(0, set_accessor_call_count); } static void ConstructCallback(const v8::FunctionCallbackInfo& info) { } TEST_F(AccessorTest, SetAccessorSetSideEffectReceiverCheck2) { v8::HandleScope scope(isolate()); i::Isolate* i_isolate = reinterpret_cast(isolate()); NoopDelegate delegate; i_isolate->debug()->SetDebugDelegate(&delegate); v8::Local templ = v8::FunctionTemplate::New( isolate(), ConstructCallback, v8::Local(), v8::Local(), 0, v8::ConstructorBehavior::kAllow, v8::SideEffectType::kHasNoSideEffect); templ->InstanceTemplate()->SetAccessor( NewString("bar"), Getter, Setter, v8::Local(), v8::AccessControl::DEFAULT, v8::PropertyAttribute::None, v8::SideEffectType::kHasSideEffectToReceiver, v8::SideEffectType::kHasSideEffectToReceiver); CHECK(context() ->Global() ->Set(context(), NewString("f"), templ->GetFunction(context()).ToLocalChecked()) .FromJust()); CHECK(v8::debug::EvaluateGlobal( isolate(), NewString("new f().bar"), v8::debug::EvaluateGlobalMode::kDisableBreaksAndThrowOnSideEffect) .ToLocalChecked() ->Equals(context(), NewString("return value")) .FromJust()); v8::debug::EvaluateGlobal( isolate(), NewString("new f().bar = 1"), v8::debug::EvaluateGlobalMode::kDisableBreaksAndThrowOnSideEffect) .ToLocalChecked(); CHECK_EQ(1, set_accessor_call_count); } // Accessors can be allowlisted as side-effect-free via SetNativeDataProperty. TEST_F(AccessorTest, AccessorSetNativeDataPropertyHasNoSideEffect) { v8::HandleScope scope(isolate()); i::Isolate* i_isolate = reinterpret_cast(isolate()); NoopDelegate delegate; i_isolate->debug()->SetDebugDelegate(&delegate); v8::Local templ = v8::ObjectTemplate::New(isolate()); v8::Local obj = templ->NewInstance(context()).ToLocalChecked(); CHECK(context()->Global()->Set(context(), NewString("obj"), obj).FromJust()); obj->SetNativeDataProperty(context(), NewString("foo"), Getter).ToChecked(); CHECK(v8::debug::EvaluateGlobal( isolate(), NewString("obj.foo"), v8::debug::EvaluateGlobalMode::kDisableBreaksAndThrowOnSideEffect) .IsEmpty()); obj->SetNativeDataProperty( context(), NewString("foo"), Getter, nullptr, v8::Local(), v8::PropertyAttribute::None, v8::SideEffectType::kHasNoSideEffect) .ToChecked(); v8::debug::EvaluateGlobal( isolate(), NewString("obj.foo"), v8::debug::EvaluateGlobalMode::kDisableBreaksAndThrowOnSideEffect) .ToLocalChecked(); // Check that setter is not allowlisted. v8::TryCatch try_catch(isolate()); CHECK(v8::debug::EvaluateGlobal( isolate(), NewString("obj.foo = 1"), v8::debug::EvaluateGlobalMode::kDisableBreaksAndThrowOnSideEffect) .IsEmpty()); CHECK(try_catch.HasCaught()); CHECK_NE(1, v8::debug::EvaluateGlobal(isolate(), NewString("obj.foo"), v8::debug::EvaluateGlobalMode::kDefault) .ToLocalChecked() ->Int32Value(context()) .FromJust()); } // Accessors can be allowlisted as side-effect-free via SetLazyDataProperty. TEST_F(AccessorTest, AccessorSetLazyDataPropertyHasNoSideEffect) { v8::HandleScope scope(isolate()); i::Isolate* i_isolate = reinterpret_cast(isolate()); NoopDelegate delegate; i_isolate->debug()->SetDebugDelegate(&delegate); v8::Local templ = v8::ObjectTemplate::New(isolate()); v8::Local obj = templ->NewInstance(context()).ToLocalChecked(); CHECK(context()->Global()->Set(context(), NewString("obj"), obj).FromJust()); obj->SetLazyDataProperty(context(), NewString("foo"), Getter).ToChecked(); CHECK(v8::debug::EvaluateGlobal( isolate(), NewString("obj.foo"), v8::debug::EvaluateGlobalMode::kDisableBreaksAndThrowOnSideEffect) .IsEmpty()); obj->SetLazyDataProperty(context(), NewString("foo"), Getter, v8::Local(), v8::PropertyAttribute::None, v8::SideEffectType::kHasNoSideEffect) .ToChecked(); v8::debug::EvaluateGlobal( isolate(), NewString("obj.foo"), v8::debug::EvaluateGlobalMode::kDisableBreaksAndThrowOnSideEffect) .ToLocalChecked(); // Check that setter is not allowlisted. v8::TryCatch try_catch(isolate()); CHECK(v8::debug::EvaluateGlobal( isolate(), NewString("obj.foo = 1"), v8::debug::EvaluateGlobalMode::kDisableBreaksAndThrowOnSideEffect) .IsEmpty()); CHECK(try_catch.HasCaught()); CHECK_NE(1, v8::debug::EvaluateGlobal(isolate(), NewString("obj.foo"), v8::debug::EvaluateGlobalMode::kDefault) .ToLocalChecked() ->Int32Value(context()) .FromJust()); } TEST_F(AccessorTest, ObjectTemplateSetAccessorHasNoSideEffect) { v8::HandleScope scope(isolate()); i::Isolate* i_isolate = reinterpret_cast(isolate()); NoopDelegate delegate; i_isolate->debug()->SetDebugDelegate(&delegate); v8::Local templ = v8::ObjectTemplate::New(isolate()); templ->SetAccessor(NewString("foo"), StringGetter); templ->SetAccessor(NewString("foo2"), StringGetter, nullptr, v8::Local(), v8::AccessControl::DEFAULT, v8::PropertyAttribute::None, v8::SideEffectType::kHasNoSideEffect); v8::Local obj = templ->NewInstance(context()).ToLocalChecked(); CHECK(context()->Global()->Set(context(), NewString("obj"), obj).FromJust()); CHECK(v8::debug::EvaluateGlobal( isolate(), NewString("obj.foo"), v8::debug::EvaluateGlobalMode::kDisableBreaksAndThrowOnSideEffect) .IsEmpty()); v8::debug::EvaluateGlobal( isolate(), NewString("obj.foo2"), v8::debug::EvaluateGlobalMode::kDisableBreaksAndThrowOnSideEffect) .ToLocalChecked(); // Check that setter is not allowlisted. v8::TryCatch try_catch(isolate()); CHECK(v8::debug::EvaluateGlobal( isolate(), NewString("obj.foo2 = 1"), v8::debug::EvaluateGlobalMode::kDisableBreaksAndThrowOnSideEffect) .IsEmpty()); CHECK(try_catch.HasCaught()); CHECK_NE(1, v8::debug::EvaluateGlobal(isolate(), NewString("obj.foo2"), v8::debug::EvaluateGlobalMode::kDefault) .ToLocalChecked() ->Int32Value(context()) .FromJust()); } TEST_F(AccessorTest, ObjectTemplateSetNativePropertyHasNoSideEffect) { v8::HandleScope scope(isolate()); i::Isolate* i_isolate = reinterpret_cast(isolate()); NoopDelegate delegate; i_isolate->debug()->SetDebugDelegate(&delegate); v8::Local templ = v8::ObjectTemplate::New(isolate()); templ->SetNativeDataProperty(NewString("foo"), Getter); templ->SetNativeDataProperty( NewString("foo2"), Getter, nullptr, v8::Local(), v8::PropertyAttribute::None, v8::AccessControl::DEFAULT, v8::SideEffectType::kHasNoSideEffect); v8::Local obj = templ->NewInstance(context()).ToLocalChecked(); CHECK(context()->Global()->Set(context(), NewString("obj"), obj).FromJust()); CHECK(v8::debug::EvaluateGlobal( isolate(), NewString("obj.foo"), v8::debug::EvaluateGlobalMode::kDisableBreaksAndThrowOnSideEffect) .IsEmpty()); v8::debug::EvaluateGlobal( isolate(), NewString("obj.foo2"), v8::debug::EvaluateGlobalMode::kDisableBreaksAndThrowOnSideEffect) .ToLocalChecked(); // Check that setter is not allowlisted. v8::TryCatch try_catch(isolate()); CHECK(v8::debug::EvaluateGlobal( isolate(), NewString("obj.foo2 = 1"), v8::debug::EvaluateGlobalMode::kDisableBreaksAndThrowOnSideEffect) .IsEmpty()); CHECK(try_catch.HasCaught()); CHECK_NE(1, v8::debug::EvaluateGlobal(isolate(), NewString("obj.foo2"), v8::debug::EvaluateGlobalMode::kDefault) .ToLocalChecked() ->Int32Value(context()) .FromJust()); } TEST_F(AccessorTest, ObjectTemplateSetLazyPropertyHasNoSideEffect) { v8::HandleScope scope(isolate()); i::Isolate* i_isolate = reinterpret_cast(isolate()); NoopDelegate delegate; i_isolate->debug()->SetDebugDelegate(&delegate); v8::Local templ = v8::ObjectTemplate::New(isolate()); templ->SetLazyDataProperty(NewString("foo"), Getter); templ->SetLazyDataProperty(NewString("foo2"), Getter, v8::Local(), v8::PropertyAttribute::None, v8::SideEffectType::kHasNoSideEffect); v8::Local obj = templ->NewInstance(context()).ToLocalChecked(); CHECK(context()->Global()->Set(context(), NewString("obj"), obj).FromJust()); CHECK(v8::debug::EvaluateGlobal( isolate(), NewString("obj.foo"), v8::debug::EvaluateGlobalMode::kDisableBreaksAndThrowOnSideEffect) .IsEmpty()); v8::debug::EvaluateGlobal( isolate(), NewString("obj.foo2"), v8::debug::EvaluateGlobalMode::kDisableBreaksAndThrowOnSideEffect) .ToLocalChecked(); // Check that setter is not allowlisted. v8::TryCatch try_catch(isolate()); CHECK(v8::debug::EvaluateGlobal( isolate(), NewString("obj.foo2 = 1"), v8::debug::EvaluateGlobalMode::kDisableBreaksAndThrowOnSideEffect) .IsEmpty()); CHECK(try_catch.HasCaught()); CHECK_NE(1, v8::debug::EvaluateGlobal(isolate(), NewString("obj.foo2"), v8::debug::EvaluateGlobalMode::kDefault) .ToLocalChecked() ->Int32Value(context()) .FromJust()); } namespace { void FunctionNativeGetter(v8::Local property, const v8::PropertyCallbackInfo& info) { info.GetIsolate()->ThrowError( v8::String::NewFromUtf8(info.GetIsolate(), "side effect in getter") .ToLocalChecked()); } } // namespace TEST_F(AccessorTest, BindFunctionTemplateSetNativeDataProperty) { v8::HandleScope scope(isolate()); // Check that getter is called on Function.prototype.bind. { v8::Local templ = v8::FunctionTemplate::New(isolate()); templ->SetNativeDataProperty(NewString("name"), FunctionNativeGetter); v8::Local func = templ->GetFunction(context()).ToLocalChecked(); CHECK(context() ->Global() ->Set(context(), NewString("func"), func) .FromJust()); v8::TryCatch try_catch(isolate()); CHECK(TryRunJS("func.bind()").IsEmpty()); CHECK(try_catch.HasCaught()); } // Check that getter is called on Function.prototype.bind. { v8::Local templ = v8::FunctionTemplate::New(isolate()); templ->SetNativeDataProperty(NewString("length"), FunctionNativeGetter); v8::Local func = templ->GetFunction(context()).ToLocalChecked(); CHECK(context() ->Global() ->Set(context(), NewString("func"), func) .FromJust()); v8::TryCatch try_catch(isolate()); CHECK(TryRunJS("func.bind()").IsEmpty()); CHECK(try_catch.HasCaught()); } } namespace { v8::MaybeLocal TestHostCreateShadowRealmContextCallback( v8::Local initiator_context) { v8::Isolate* isolate = initiator_context->GetIsolate(); v8::Local global_constructor = v8::FunctionTemplate::New(isolate); v8::Local global_template = global_constructor->InstanceTemplate(); // Check that getter is called on Function.prototype.bind. global_template->SetNativeDataProperty( v8::String::NewFromUtf8(isolate, "func1").ToLocalChecked(), [](v8::Local property, const v8::PropertyCallbackInfo& info) { v8::Isolate* isolate = info.GetIsolate(); v8::Local templ = v8::FunctionTemplate::New(isolate); templ->SetNativeDataProperty( v8::String::NewFromUtf8(isolate, "name").ToLocalChecked(), FunctionNativeGetter); info.GetReturnValue().Set( templ->GetFunction(isolate->GetCurrentContext()).ToLocalChecked()); }); // Check that getter is called on Function.prototype.bind. global_template->SetNativeDataProperty( v8::String::NewFromUtf8(isolate, "func2").ToLocalChecked(), [](v8::Local property, const v8::PropertyCallbackInfo& info) { v8::Isolate* isolate = info.GetIsolate(); v8::Local templ = v8::FunctionTemplate::New(isolate); templ->SetNativeDataProperty( v8::String::NewFromUtf8(isolate, "length").ToLocalChecked(), FunctionNativeGetter); info.GetReturnValue().Set( templ->GetFunction(isolate->GetCurrentContext()).ToLocalChecked()); }); return v8::Context::New(isolate, nullptr, global_template); } } // namespace TEST_F(AccessorTest, WrapFunctionTemplateSetNativeDataProperty) { i::FLAG_harmony_shadow_realm = true; isolate()->SetHostCreateShadowRealmContextCallback( TestHostCreateShadowRealmContextCallback); v8::HandleScope scope(isolate()); // Check that getter is called on WrappedFunctionCreate. { v8::TryCatch try_catch(isolate()); CHECK(TryRunJS("new ShadowRealm().evaluate('globalThis.func1')").IsEmpty()); CHECK(try_catch.HasCaught()); } // Check that getter is called on WrappedFunctionCreate. { v8::TryCatch try_catch(isolate()); CHECK(TryRunJS("new ShadowRealm().evaluate('globalThis.func2')").IsEmpty()); CHECK(try_catch.HasCaught()); } }