// 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 "src/interface-descriptors.h" #include "src/isolate.h" #include "test/cctest/compiler/function-tester.h" namespace v8 { namespace internal { namespace compiler { class CodeStubAssemblerTester : public CodeStubAssembler { public: CodeStubAssemblerTester(Isolate* isolate, const CallInterfaceDescriptor& descriptor) : CodeStubAssembler(isolate, isolate->runtime_zone(), descriptor, Code::ComputeFlags(Code::STUB), "test"), scope_(isolate) {} private: HandleScope scope_; LocalContext context_; }; TEST(SimpleSmiReturn) { Isolate* isolate(CcTest::InitIsolateOnce()); VoidDescriptor descriptor(isolate); CodeStubAssemblerTester m(isolate, descriptor); m.Return(m.SmiTag(m.Int32Constant(37))); Handle code = m.GenerateCode(); FunctionTester ft(descriptor, code); MaybeHandle result = ft.Call(); CHECK_EQ(37, Handle::cast(result.ToHandleChecked())->value()); } TEST(SimpleIntPtrReturn) { Isolate* isolate(CcTest::InitIsolateOnce()); VoidDescriptor descriptor(isolate); CodeStubAssemblerTester m(isolate, descriptor); int test; m.Return(m.IntPtrConstant(reinterpret_cast(&test))); Handle code = m.GenerateCode(); FunctionTester ft(descriptor, code); MaybeHandle result = ft.Call(); CHECK_EQ(reinterpret_cast(&test), reinterpret_cast(*result.ToHandleChecked())); } TEST(SimpleDoubleReturn) { Isolate* isolate(CcTest::InitIsolateOnce()); VoidDescriptor descriptor(isolate); CodeStubAssemblerTester m(isolate, descriptor); m.Return(m.NumberConstant(0.5)); Handle code = m.GenerateCode(); FunctionTester ft(descriptor, code); MaybeHandle result = ft.Call(); CHECK_EQ(0.5, Handle::cast(result.ToHandleChecked())->value()); } TEST(SimpleCallRuntime1Arg) { Isolate* isolate(CcTest::InitIsolateOnce()); VoidDescriptor descriptor(isolate); CodeStubAssemblerTester m(isolate, descriptor); Node* context = m.HeapConstant(Handle(isolate->native_context())); Node* b = m.SmiTag(m.Int32Constant(256)); m.Return(m.CallRuntime(Runtime::kMathSqrt, context, b)); Handle code = m.GenerateCode(); FunctionTester ft(descriptor, code); MaybeHandle result = ft.Call(); CHECK_EQ(16, Handle::cast(result.ToHandleChecked())->value()); } TEST(SimpleTailCallRuntime1Arg) { Isolate* isolate(CcTest::InitIsolateOnce()); VoidDescriptor descriptor(isolate); CodeStubAssemblerTester m(isolate, descriptor); Node* context = m.HeapConstant(Handle(isolate->native_context())); Node* b = m.SmiTag(m.Int32Constant(256)); m.TailCallRuntime(Runtime::kMathSqrt, context, b); Handle code = m.GenerateCode(); FunctionTester ft(descriptor, code); MaybeHandle result = ft.Call(); CHECK_EQ(16, Handle::cast(result.ToHandleChecked())->value()); } TEST(SimpleCallRuntime2Arg) { Isolate* isolate(CcTest::InitIsolateOnce()); VoidDescriptor descriptor(isolate); CodeStubAssemblerTester m(isolate, descriptor); Node* context = m.HeapConstant(Handle(isolate->native_context())); Node* a = m.SmiTag(m.Int32Constant(2)); Node* b = m.SmiTag(m.Int32Constant(4)); m.Return(m.CallRuntime(Runtime::kMathPow, context, a, b)); Handle code = m.GenerateCode(); FunctionTester ft(descriptor, code); MaybeHandle result = ft.Call(); CHECK_EQ(16, Handle::cast(result.ToHandleChecked())->value()); } TEST(SimpleTailCallRuntime2Arg) { Isolate* isolate(CcTest::InitIsolateOnce()); VoidDescriptor descriptor(isolate); CodeStubAssemblerTester m(isolate, descriptor); Node* context = m.HeapConstant(Handle(isolate->native_context())); Node* a = m.SmiTag(m.Int32Constant(2)); Node* b = m.SmiTag(m.Int32Constant(4)); m.TailCallRuntime(Runtime::kMathPow, context, a, b); Handle code = m.GenerateCode(); FunctionTester ft(descriptor, code); MaybeHandle result = ft.Call(); CHECK_EQ(16, Handle::cast(result.ToHandleChecked())->value()); } TEST(VariableMerge1) { Isolate* isolate(CcTest::InitIsolateOnce()); VoidDescriptor descriptor(isolate); CodeStubAssemblerTester m(isolate, descriptor); CodeStubAssembler::Variable var1(&m, MachineRepresentation::kTagged); CodeStubAssembler::Label l1(&m), l2(&m), merge(&m); Node* temp = m.Int32Constant(0); var1.Bind(temp); m.Branch(m.Int32Constant(1), &l1, &l2); m.Bind(&l1); CHECK_EQ(var1.value(), temp); m.Goto(&merge); m.Bind(&l2); CHECK_EQ(var1.value(), temp); m.Goto(&merge); m.Bind(&merge); CHECK_EQ(var1.value(), temp); } TEST(VariableMerge2) { Isolate* isolate(CcTest::InitIsolateOnce()); VoidDescriptor descriptor(isolate); CodeStubAssemblerTester m(isolate, descriptor); CodeStubAssembler::Variable var1(&m, MachineRepresentation::kTagged); CodeStubAssembler::Label l1(&m), l2(&m), merge(&m); Node* temp = m.Int32Constant(0); var1.Bind(temp); m.Branch(m.Int32Constant(1), &l1, &l2); m.Bind(&l1); CHECK_EQ(var1.value(), temp); m.Goto(&merge); m.Bind(&l2); Node* temp2 = m.Int32Constant(2); var1.Bind(temp2); CHECK_EQ(var1.value(), temp2); m.Goto(&merge); m.Bind(&merge); CHECK_NE(var1.value(), temp); } TEST(VariableMerge3) { Isolate* isolate(CcTest::InitIsolateOnce()); VoidDescriptor descriptor(isolate); CodeStubAssemblerTester m(isolate, descriptor); CodeStubAssembler::Variable var1(&m, MachineRepresentation::kTagged); CodeStubAssembler::Variable var2(&m, MachineRepresentation::kTagged); CodeStubAssembler::Label l1(&m), l2(&m), merge(&m); Node* temp = m.Int32Constant(0); var1.Bind(temp); var2.Bind(temp); m.Branch(m.Int32Constant(1), &l1, &l2); m.Bind(&l1); CHECK_EQ(var1.value(), temp); m.Goto(&merge); m.Bind(&l2); Node* temp2 = m.Int32Constant(2); var1.Bind(temp2); CHECK_EQ(var1.value(), temp2); m.Goto(&merge); m.Bind(&merge); CHECK_NE(var1.value(), temp); CHECK_NE(var1.value(), temp2); CHECK_EQ(var2.value(), temp); } TEST(VariableMergeBindFirst) { Isolate* isolate(CcTest::InitIsolateOnce()); VoidDescriptor descriptor(isolate); CodeStubAssemblerTester m(isolate, descriptor); CodeStubAssembler::Variable var1(&m, MachineRepresentation::kTagged); CodeStubAssembler::Label l1(&m), l2(&m), merge(&m, &var1), end(&m); Node* temp = m.Int32Constant(0); var1.Bind(temp); m.Branch(m.Int32Constant(1), &l1, &l2); m.Bind(&l1); CHECK_EQ(var1.value(), temp); m.Goto(&merge); m.Bind(&merge); CHECK(var1.value() != temp); CHECK(var1.value() != nullptr); m.Goto(&end); m.Bind(&l2); Node* temp2 = m.Int32Constant(2); var1.Bind(temp2); CHECK_EQ(var1.value(), temp2); m.Goto(&merge); m.Bind(&end); CHECK(var1.value() != temp); CHECK(var1.value() != nullptr); } TEST(VariableMergeSwitch) { Isolate* isolate(CcTest::InitIsolateOnce()); VoidDescriptor descriptor(isolate); CodeStubAssemblerTester m(isolate, descriptor); CodeStubAssembler::Variable var1(&m, MachineRepresentation::kTagged); CodeStubAssembler::Label l1(&m), l2(&m), default_label(&m); CodeStubAssembler::Label* labels[] = {&l1, &l2}; int32_t values[] = {1, 2}; Node* temp = m.Int32Constant(0); var1.Bind(temp); m.Switch(m.Int32Constant(2), &default_label, values, labels, 2); m.Bind(&l1); DCHECK_EQ(temp, var1.value()); m.Return(temp); m.Bind(&l2); DCHECK_EQ(temp, var1.value()); m.Return(temp); m.Bind(&default_label); DCHECK_EQ(temp, var1.value()); m.Return(temp); } TEST(FixedArrayAccessSmiIndex) { Isolate* isolate(CcTest::InitIsolateOnce()); VoidDescriptor descriptor(isolate); CodeStubAssemblerTester m(isolate, descriptor); Handle array = isolate->factory()->NewFixedArray(5); array->set(4, Smi::FromInt(733)); m.Return(m.LoadFixedArrayElementSmiIndex(m.HeapConstant(array), m.SmiTag(m.Int32Constant(4)))); Handle code = m.GenerateCode(); FunctionTester ft(descriptor, code); MaybeHandle result = ft.Call(); CHECK_EQ(733, Handle::cast(result.ToHandleChecked())->value()); } } // namespace compiler } // namespace internal } // namespace v8