/* * Copyright (c) 2021 Huawei Device Co., Ltd. * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ #include "ecmascript/builtins/builtins_global.h" #include "ecmascript/js_hclass.h" #include "ecmascript/ecma_runtime_call_info.h" #include "ecmascript/ecma_vm.h" #include "ecmascript/global_env.h" #include "ecmascript/js_tagged_value-inl.h" #include "ecmascript/js_thread.h" #include "ecmascript/object_factory.h" #include "ecmascript/js_array.h" #include "ecmascript/shared_objects/js_shared_array.h" #include "ecmascript/tests/test_helper.h" using namespace panda::ecmascript; using namespace panda::ecmascript::builtins; using BuiltinsBase = panda::ecmascript::base::BuiltinsBase; namespace panda::test { class BuiltinsGlobalTest : public BaseTestWithScope { public: static JSHandle NewJSObject(JSThread *thread, ObjectFactory *factory, JSHandle globalEnv) { JSFunction *jsFunc = globalEnv->GetObjectFunction().GetObject(); JSHandle jsFunc1(thread, jsFunc); JSHandle jsObj = factory->NewJSObjectByConstructor(JSHandle(jsFunc1), jsFunc1); return jsObj; } }; HWTEST_F_L0(BuiltinsGlobalTest, DecodeURIComponent) { auto ecmaRuntimeCallInfo = TestHelper::CreateEcmaRuntimeCallInfo(thread, JSTaggedValue::Undefined(), 6); ecmaRuntimeCallInfo->SetFunction(JSTaggedValue::Undefined()); ecmaRuntimeCallInfo->SetThis(JSTaggedValue::Undefined()); ecmaRuntimeCallInfo->SetCallArg(0, JSTaggedValue(123)); [[maybe_unused]] auto prev = TestHelper::SetupFrame(thread, ecmaRuntimeCallInfo); JSTaggedValue result = BuiltinsGlobal::DecodeURIComponent(ecmaRuntimeCallInfo); TestHelper::TearDownFrame(thread, prev); EXPECT_TRUE(result.IsString()); JSHandle ecmaStrHandle(thread, result); EXPECT_STREQ("123", EcmaStringAccessor(ecmaStrHandle).ToCString().c_str()); } HWTEST_F_L0(BuiltinsGlobalTest, EncodeURIComponent) { auto ecmaRuntimeCallInfo = TestHelper::CreateEcmaRuntimeCallInfo(thread, JSTaggedValue::Undefined(), 6); ecmaRuntimeCallInfo->SetFunction(JSTaggedValue::Undefined()); ecmaRuntimeCallInfo->SetThis(JSTaggedValue::Undefined()); ecmaRuntimeCallInfo->SetCallArg(0, JSTaggedValue(123)); [[maybe_unused]] auto prev = TestHelper::SetupFrame(thread, ecmaRuntimeCallInfo); JSTaggedValue result = BuiltinsGlobal::EncodeURIComponent(ecmaRuntimeCallInfo); TestHelper::TearDownFrame(thread, prev); EXPECT_TRUE(result.IsString()); JSHandle ecmaStrHandle(thread, result); EXPECT_STREQ("123", EcmaStringAccessor(ecmaStrHandle).ToCString().c_str()); } HWTEST_F_L0(BuiltinsGlobalTest, PrintEntrypointAndPrintString) { // msg == nullptr JSTaggedValue result = BuiltinsGlobal::PrintEntrypoint(nullptr); EXPECT_EQ(result, JSTaggedValue::Undefined()); } HWTEST_F_L0(BuiltinsGlobalTest, CallJsProxy) { ObjectFactory *factory = thread->GetEcmaVM()->GetFactory(); auto ecmaVM = thread->GetEcmaVM(); JSHandle env = ecmaVM->GetGlobalEnv(); JSHandle emptyObj(thread, NewJSObject(thread, factory, env).GetTaggedValue()); JSHandle proxy = factory->NewJSProxy(emptyObj, emptyObj); proxy->SetCallable(false); auto ecmaRuntimeCallInfo = TestHelper::CreateEcmaRuntimeCallInfo(thread, JSTaggedValue::Undefined(), 6); ecmaRuntimeCallInfo->SetFunction(proxy.GetTaggedValue()); ecmaRuntimeCallInfo->SetThis(JSTaggedValue::Undefined()); [[maybe_unused]] auto prev = TestHelper::SetupFrame(thread, ecmaRuntimeCallInfo); JSTaggedValue result = BuiltinsGlobal::CallJsProxy(ecmaRuntimeCallInfo); TestHelper::TearDownFrame(thread, prev); EXPECT_TRUE(thread->HasPendingException()); EXPECT_EQ(result, JSTaggedValue::Undefined()); thread->ClearException(); } HWTEST_F_L0(BuiltinsGlobalTest, Escape) { ObjectFactory *factory = thread->GetEcmaVM()->GetFactory(); JSHandle str1 = factory->NewFromASCII("?!=()#%&"); auto ecmaRuntimeCallInfo1 = TestHelper::CreateEcmaRuntimeCallInfo(thread, JSTaggedValue::Undefined(), 6); // NOLINT, 6 means 3 paras ecmaRuntimeCallInfo1->SetFunction(JSTaggedValue::Undefined()); ecmaRuntimeCallInfo1->SetThis(JSTaggedValue::Undefined()); ecmaRuntimeCallInfo1->SetCallArg(0, str1.GetTaggedValue()); [[maybe_unused]] auto prev1 = TestHelper::SetupFrame(thread, ecmaRuntimeCallInfo1); JSTaggedValue result1 = BuiltinsGlobal::Escape(ecmaRuntimeCallInfo1); TestHelper::TearDownFrame(thread, prev1); EXPECT_TRUE(result1.IsString()); JSHandle ecmaStrHandle1(thread, result1); EXPECT_STREQ("%3F%21%3D%28%29%23%25%26", EcmaStringAccessor(ecmaStrHandle1).ToCString().c_str()); // NOLINT JSHandle str2 = factory->NewFromASCII("%u%u0%u9%ua%uF%u00%u09%u0f%u0F%u000%u00a%u00F"); auto ecmaRuntimeCallInfo2 = TestHelper::CreateEcmaRuntimeCallInfo(thread, JSTaggedValue::Undefined(), 6); // NOLINT ecmaRuntimeCallInfo2->SetFunction(JSTaggedValue::Undefined()); ecmaRuntimeCallInfo2->SetThis(JSTaggedValue::Undefined()); ecmaRuntimeCallInfo2->SetCallArg(0, str2.GetTaggedValue()); // NOLINT [[maybe_unused]] auto prev2 = TestHelper::SetupFrame(thread, ecmaRuntimeCallInfo2); JSTaggedValue result2 = BuiltinsGlobal::Escape(ecmaRuntimeCallInfo2); TestHelper::TearDownFrame(thread, prev2); EXPECT_TRUE(result2.IsString()); JSHandle ecmaStrHandle2(thread, result2); EXPECT_STREQ("%25u%25u0%25u9%25ua%25uF%25u00%25u09%25u0f%25u0F%25u000%25u00a%25u00F", // NOLINT special value EcmaStringAccessor(ecmaStrHandle2).ToCString().c_str()); JSHandle str3 = factory->NewFromASCII("Hello World!"); std::vector args{str3.GetTaggedValue()}; auto ecmaRuntimeCallInfo3 = TestHelper::CreateEcmaRuntimeCallInfo(thread, args, 6); [[maybe_unused]] auto prev3 = TestHelper::SetupFrame(thread, ecmaRuntimeCallInfo3); JSTaggedValue result3 = BuiltinsGlobal::Escape(ecmaRuntimeCallInfo3); TestHelper::TearDownFrame(thread, prev3); EXPECT_TRUE(result3.IsString()); JSHandle ecmaStrHandle3(thread, result3); EXPECT_STREQ("Hello%20World%21", EcmaStringAccessor(ecmaStrHandle3).ToCString().c_str()); } HWTEST_F_L0(BuiltinsGlobalTest, Unescape) { ObjectFactory *factory = thread->GetEcmaVM()->GetFactory(); JSHandle str1 = factory->NewFromASCII(""); auto ecmaRuntimeCallInfo1 = TestHelper::CreateEcmaRuntimeCallInfo(thread, JSTaggedValue::Undefined(), 6); // NOLINT ecmaRuntimeCallInfo1->SetFunction(JSTaggedValue::Undefined()); ecmaRuntimeCallInfo1->SetThis(JSTaggedValue::Undefined()); ecmaRuntimeCallInfo1->SetCallArg(0, str1.GetTaggedValue()); [[maybe_unused]] auto prev1 = TestHelper::SetupFrame(thread, ecmaRuntimeCallInfo1); JSTaggedValue result1 = BuiltinsGlobal::Unescape(ecmaRuntimeCallInfo1); TestHelper::TearDownFrame(thread, prev1); EXPECT_TRUE(result1.IsString()); JSHandle ecmaStrHandle1(thread, result1); EXPECT_STREQ("", EcmaStringAccessor(ecmaStrHandle1).ToCString().c_str()); JSHandle str2 = factory->NewFromASCII("%u%u0%u9%ua%uF%u00%u09%u0f%u0F%u000%u00a%u00F"); auto ecmaRuntimeCallInfo2 = TestHelper::CreateEcmaRuntimeCallInfo(thread, JSTaggedValue::Undefined(), 6); // NOLINT ecmaRuntimeCallInfo2->SetFunction(JSTaggedValue::Undefined()); ecmaRuntimeCallInfo2->SetThis(JSTaggedValue::Undefined()); ecmaRuntimeCallInfo2->SetCallArg(0, str2.GetTaggedValue()); // NOLINT [[maybe_unused]] auto prev2 = TestHelper::SetupFrame(thread, ecmaRuntimeCallInfo2); JSTaggedValue result2 = BuiltinsGlobal::Unescape(ecmaRuntimeCallInfo2); TestHelper::TearDownFrame(thread, prev2); EXPECT_TRUE(result2.IsString()); JSHandle ecmaStrHandle2(thread, result2); EXPECT_STREQ("%u%u0%u9%ua%uF%u00%u09%u0f%u0F%u000%u00a%u00F", EcmaStringAccessor(ecmaStrHandle2).ToCString().c_str()); JSHandle str3 = factory->NewFromASCII("Hello%20World%21"); std::vector args{str3.GetTaggedValue()}; auto ecmaRuntimeCallInfo3 = TestHelper::CreateEcmaRuntimeCallInfo(thread, args, 6); [[maybe_unused]] auto prev3 = TestHelper::SetupFrame(thread, ecmaRuntimeCallInfo3); JSTaggedValue result3 = BuiltinsGlobal::Escape(ecmaRuntimeCallInfo3); TestHelper::TearDownFrame(thread, prev3); EXPECT_TRUE(result3.IsString()); JSHandle ecmaStrHandle3(thread, result3); EXPECT_STREQ("Hello%2520World%2521", EcmaStringAccessor(ecmaStrHandle3).ToCString().c_str()); } HWTEST_F_L0(BuiltinsGlobalTest, isSendable_0) { ObjectFactory *factory = thread->GetEcmaVM()->GetFactory(); JSHandle str1 = factory->NewFromASCII("?!=()#%&"); auto ecmaRuntimeCallInfo1 = TestHelper::CreateEcmaRuntimeCallInfo(thread, JSTaggedValue::Undefined(), 6); ecmaRuntimeCallInfo1->SetFunction(JSTaggedValue::Undefined()); ecmaRuntimeCallInfo1->SetThis(JSTaggedValue::Undefined()); ecmaRuntimeCallInfo1->SetCallArg(0, str1.GetTaggedValue()); [[maybe_unused]] auto prev1 = TestHelper::SetupFrame(thread, ecmaRuntimeCallInfo1); JSTaggedValue result = BuiltinsGlobal::IsSendable(ecmaRuntimeCallInfo1); TestHelper::TearDownFrame(thread, prev1); EXPECT_EQ(JSTaggedValue::True(), result); auto ecmaRuntimeCallInfo2 = TestHelper::CreateEcmaRuntimeCallInfo(thread, JSTaggedValue::Undefined(), 6); ecmaRuntimeCallInfo2->SetFunction(JSTaggedValue::Undefined()); ecmaRuntimeCallInfo2->SetThis(JSTaggedValue::Undefined()); ecmaRuntimeCallInfo2->SetCallArg(0, JSTaggedValue(1)); prev1 = TestHelper::SetupFrame(thread, ecmaRuntimeCallInfo2); result = BuiltinsGlobal::IsSendable(ecmaRuntimeCallInfo2); TestHelper::TearDownFrame(thread, prev1); EXPECT_EQ(JSTaggedValue::True(), result); auto ecmaRuntimeCallInfo3 = TestHelper::CreateEcmaRuntimeCallInfo(thread, JSTaggedValue::Undefined(), 6); ecmaRuntimeCallInfo3->SetFunction(JSTaggedValue::Undefined()); ecmaRuntimeCallInfo3->SetThis(JSTaggedValue::Undefined()); ecmaRuntimeCallInfo3->SetCallArg(0, JSTaggedValue(1.5)); prev1 = TestHelper::SetupFrame(thread, ecmaRuntimeCallInfo3); result = BuiltinsGlobal::IsSendable(ecmaRuntimeCallInfo3); TestHelper::TearDownFrame(thread, prev1); EXPECT_EQ(JSTaggedValue::True(), result); } HWTEST_F_L0(BuiltinsGlobalTest, isSendable_1) { ObjectFactory *factory = thread->GetEcmaVM()->GetFactory(); auto ecmaRuntimeCallInfo5 = TestHelper::CreateEcmaRuntimeCallInfo(thread, JSTaggedValue::Undefined(), 6); ecmaRuntimeCallInfo5->SetFunction(JSTaggedValue::Undefined()); ecmaRuntimeCallInfo5->SetThis(JSTaggedValue::Undefined()); JSFunction *jsFunc = thread->GetEcmaVM()->GetGlobalEnv()->GetObjectFunction().GetObject(); JSHandle jsFunc1(thread, jsFunc); JSHandle jsObj = factory->NewJSObjectByConstructor(JSHandle(jsFunc1), jsFunc1); ecmaRuntimeCallInfo5->SetCallArg(0, jsObj.GetTaggedValue()); [[maybe_unused]] auto prev1 = TestHelper::SetupFrame(thread, ecmaRuntimeCallInfo5); auto result = BuiltinsGlobal::IsSendable(ecmaRuntimeCallInfo5); TestHelper::TearDownFrame(thread, prev1); EXPECT_EQ(JSTaggedValue::False(), result); auto ecmaRuntimeCallInfo6 = TestHelper::CreateEcmaRuntimeCallInfo(thread, JSTaggedValue::Undefined(), 6); ecmaRuntimeCallInfo6->SetFunction(JSTaggedValue::Undefined()); ecmaRuntimeCallInfo6->SetThis(JSTaggedValue::Undefined()); JSHandle sharedArrayFunc(thread->GetEcmaVM()->GetGlobalEnv()->GetSharedArrayFunction()); ecmaRuntimeCallInfo6->SetCallArg(0, sharedArrayFunc.GetTaggedValue()); prev1 = TestHelper::SetupFrame(thread, ecmaRuntimeCallInfo6); result = BuiltinsGlobal::IsSendable(ecmaRuntimeCallInfo6); TestHelper::TearDownFrame(thread, prev1); EXPECT_EQ(JSTaggedValue::True(), result); auto ecmaRuntimeCallInfo8 = TestHelper::CreateEcmaRuntimeCallInfo(thread, JSTaggedValue::Undefined(), 6); ecmaRuntimeCallInfo8->SetFunction(JSTaggedValue::Undefined()); ecmaRuntimeCallInfo8->SetThis(JSTaggedValue::Undefined()); JSArray *arr = JSArray::Cast(JSArray::ArrayCreate(thread, JSTaggedNumber(0)).GetTaggedValue().GetTaggedObject()); EXPECT_TRUE(arr != nullptr); JSHandle arrObj(thread, arr); ecmaRuntimeCallInfo8->SetCallArg(0, arrObj.GetTaggedValue()); prev1 = TestHelper::SetupFrame(thread, ecmaRuntimeCallInfo8); result = BuiltinsGlobal::IsSendable(ecmaRuntimeCallInfo8); TestHelper::TearDownFrame(thread, prev1); EXPECT_EQ(JSTaggedValue::False(), result); } HWTEST_F_L0(BuiltinsGlobalTest, isSendable_2) { auto ecmaRuntimeCallInfo4 = TestHelper::CreateEcmaRuntimeCallInfo(thread, JSTaggedValue::Undefined(), 6); ecmaRuntimeCallInfo4->SetFunction(JSTaggedValue::Undefined()); ecmaRuntimeCallInfo4->SetThis(JSTaggedValue::Undefined()); JSFunction *jsFunc = thread->GetEcmaVM()->GetGlobalEnv()->GetObjectFunction().GetObject(); JSHandle jsFunc1(thread, jsFunc); ecmaRuntimeCallInfo4->SetCallArg(0, jsFunc1.GetTaggedValue()); [[maybe_unused]] auto prev1 = TestHelper::SetupFrame(thread, ecmaRuntimeCallInfo4); auto result = BuiltinsGlobal::IsSendable(ecmaRuntimeCallInfo4); TestHelper::TearDownFrame(thread, prev1); EXPECT_EQ(JSTaggedValue::False(), result); auto ecmaRuntimeCallInfo7 = TestHelper::CreateEcmaRuntimeCallInfo(thread, JSTaggedValue::Undefined(), 6); ecmaRuntimeCallInfo7->SetFunction(JSTaggedValue::Undefined()); ecmaRuntimeCallInfo7->SetThis(JSTaggedValue::Undefined()); JSHandle sharedArrayFunc(thread->GetEcmaVM()->GetGlobalEnv()->GetSharedArrayFunction()); JSSharedArray *arr = JSSharedArray::Cast(JSSharedArray::ArrayCreate(thread, JSTaggedNumber(0))->GetTaggedObject()); EXPECT_TRUE(arr != nullptr); JSHandle sharedObj(thread, arr); ecmaRuntimeCallInfo7->SetCallArg(0, sharedObj.GetTaggedValue()); prev1 = TestHelper::SetupFrame(thread, ecmaRuntimeCallInfo7); result = BuiltinsGlobal::IsSendable(ecmaRuntimeCallInfo7); TestHelper::TearDownFrame(thread, prev1); EXPECT_EQ(JSTaggedValue::True(), result); auto ecmaRuntimeCallInfo9 = TestHelper::CreateEcmaRuntimeCallInfo(thread, JSTaggedValue::Undefined(), 8); ecmaRuntimeCallInfo9->SetFunction(JSTaggedValue::Undefined()); ecmaRuntimeCallInfo9->SetThis(JSTaggedValue::Undefined()); ecmaRuntimeCallInfo9->SetCallArg(0, sharedObj.GetTaggedValue()); ecmaRuntimeCallInfo9->SetCallArg(1, sharedObj.GetTaggedValue()); prev1 = TestHelper::SetupFrame(thread, ecmaRuntimeCallInfo9); result = BuiltinsGlobal::IsSendable(ecmaRuntimeCallInfo9); TestHelper::TearDownFrame(thread, prev1); EXPECT_EQ(JSTaggedValue::False(), result); } HWTEST_F_L0(BuiltinsGlobalTest, isSendable_3) { auto ecmaRuntimeCallInfo4 = TestHelper::CreateEcmaRuntimeCallInfo(thread, JSTaggedValue::Undefined(), 6); ecmaRuntimeCallInfo4->SetFunction(JSTaggedValue::Undefined()); ecmaRuntimeCallInfo4->SetThis(JSTaggedValue::Undefined()); ecmaRuntimeCallInfo4->SetCallArg(0, JSTaggedValue::Undefined()); [[maybe_unused]] auto prev1 = TestHelper::SetupFrame(thread, ecmaRuntimeCallInfo4); auto result = BuiltinsGlobal::IsSendable(ecmaRuntimeCallInfo4); TestHelper::TearDownFrame(thread, prev1); EXPECT_EQ(JSTaggedValue::True(), result); auto ecmaRuntimeCallInfo5 = TestHelper::CreateEcmaRuntimeCallInfo(thread, JSTaggedValue::Undefined(), 6); ecmaRuntimeCallInfo5->SetFunction(JSTaggedValue::Undefined()); ecmaRuntimeCallInfo5->SetThis(JSTaggedValue::Undefined()); ecmaRuntimeCallInfo5->SetCallArg(0, JSTaggedValue::True()); prev1 = TestHelper::SetupFrame(thread, ecmaRuntimeCallInfo5); result = BuiltinsGlobal::IsSendable(ecmaRuntimeCallInfo5); TestHelper::TearDownFrame(thread, prev1); EXPECT_EQ(JSTaggedValue::True(), result); auto ecmaRuntimeCallInfo6 = TestHelper::CreateEcmaRuntimeCallInfo(thread, JSTaggedValue::Undefined(), 8); ecmaRuntimeCallInfo6->SetFunction(JSTaggedValue::Undefined()); ecmaRuntimeCallInfo6->SetThis(JSTaggedValue::Undefined()); ecmaRuntimeCallInfo6->SetCallArg(0, JSTaggedValue::Exception()); prev1 = TestHelper::SetupFrame(thread, ecmaRuntimeCallInfo6); result = BuiltinsGlobal::IsSendable(ecmaRuntimeCallInfo6); TestHelper::TearDownFrame(thread, prev1); EXPECT_EQ(JSTaggedValue::False(), result); auto ecmaRuntimeCallInfo9 = TestHelper::CreateEcmaRuntimeCallInfo(thread, JSTaggedValue::Undefined(), 6); ecmaRuntimeCallInfo9->SetFunction(JSTaggedValue::Undefined()); ecmaRuntimeCallInfo9->SetThis(JSTaggedValue::Undefined()); ecmaRuntimeCallInfo9->SetCallArg(0, JSTaggedValue::Null()); prev1 = TestHelper::SetupFrame(thread, ecmaRuntimeCallInfo9); result = BuiltinsGlobal::IsSendable(ecmaRuntimeCallInfo9); TestHelper::TearDownFrame(thread, prev1); EXPECT_EQ(JSTaggedValue::True(), result); } } // namespace panda::test