arkcompiler_ets_runtime/ecmascript/js_function.cpp
wengchangcheng 9a907ba3c3 optimize primitive type ACCESSORS
Signed-off-by: wengchangcheng <wengchangcheng@huawei.com>
2022-01-25 18:05:12 +08:00

625 lines
28 KiB
C++
Raw Blame History

This file contains ambiguous Unicode characters

This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

/*
* 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 "js_function.h"
#include "ecmascript/base/error_type.h"
#include "ecmascript/class_info_extractor.h"
#include "ecmascript/ecma_macros.h"
#include "ecmascript/ecma_runtime_call_info.h"
#include "ecmascript/global_env.h"
#include "ecmascript/internal_call_params.h"
#include "ecmascript/interpreter/interpreter-inl.h"
#include "ecmascript/js_handle.h"
#include "ecmascript/js_promise.h"
#include "ecmascript/js_proxy.h"
#include "ecmascript/js_tagged_value-inl.h"
#include "ecmascript/mem/c_containers.h"
#include "ecmascript/object_factory.h"
#include "ecmascript/tagged_array.h"
namespace panda::ecmascript {
void JSFunction::InitializeJSFunction(JSThread *thread, const JSHandle<JSFunction> &func, FunctionKind kind,
bool strict)
{
FunctionMode thisMode;
if (IsArrowFunction(kind)) {
thisMode = FunctionMode::LEXICAL;
} else if (strict) {
thisMode = FunctionMode::STRICT;
} else {
thisMode = FunctionMode::GLOBAL;
}
func->SetProtoOrDynClass(thread, JSTaggedValue::Hole(), SKIP_BARRIER);
func->SetHomeObject(thread, JSTaggedValue::Undefined(), SKIP_BARRIER);
func->SetLexicalEnv(thread, JSTaggedValue::Undefined(), SKIP_BARRIER);
func->SetConstantPool(thread, JSTaggedValue::Undefined(), SKIP_BARRIER);
func->SetProfileTypeInfo(thread, JSTaggedValue::Undefined(), SKIP_BARRIER);
func->SetFunctionExtraInfo(thread, JSTaggedValue::Undefined());
func->SetFunctionKind(kind);
func->SetStrict(strict);
func->SetThisMode(thisMode);
func->SetResolved(false);
auto globalConst = thread->GlobalConstants();
if (HasPrototype(kind)) {
JSHandle<JSTaggedValue> accessor = globalConst->GetHandledFunctionPrototypeAccessor();
if (kind == FunctionKind::BASE_CONSTRUCTOR || kind == FunctionKind::GENERATOR_FUNCTION) {
func->SetPropertyInlinedProps(thread, PROTOTYPE_INLINE_PROPERTY_INDEX, accessor.GetTaggedValue());
accessor = globalConst->GetHandledFunctionNameAccessor();
func->SetPropertyInlinedProps(thread, NAME_INLINE_PROPERTY_INDEX, accessor.GetTaggedValue());
} else if (!JSFunction::IsClassConstructor(kind)) { // class ctor do nothing
PropertyDescriptor desc(thread, accessor, kind != FunctionKind::BUILTIN_CONSTRUCTOR, false, false);
[[maybe_unused]] bool success = JSObject::DefineOwnProperty(thread, JSHandle<JSObject>(func),
globalConst->GetHandledPrototypeString(), desc);
ASSERT(success);
}
} else if (HasAccessor(kind)) {
JSHandle<JSTaggedValue> accessor = globalConst->GetHandledFunctionNameAccessor();
func->SetPropertyInlinedProps(thread, NAME_INLINE_PROPERTY_INDEX, accessor.GetTaggedValue());
}
}
JSHandle<JSObject> JSFunction::NewJSFunctionPrototype(JSThread *thread, ObjectFactory *factory,
const JSHandle<JSFunction> &func)
{
JSHandle<GlobalEnv> env = thread->GetEcmaVM()->GetGlobalEnv();
const GlobalEnvConstants *globalConst = thread->GlobalConstants();
JSHandle<JSTaggedValue> objFun = env->GetObjectFunction();
JSHandle<JSObject> funPro = factory->NewJSObjectByConstructor(JSHandle<JSFunction>(objFun), objFun);
func->SetFunctionPrototype(thread, funPro.GetTaggedValue());
// set "constructor" in prototype
JSHandle<JSTaggedValue> constructorKey = globalConst->GetHandledConstructorString();
PropertyDescriptor descriptor(thread, JSHandle<JSTaggedValue>::Cast(func), true, false, true);
JSObject::DefineOwnProperty(thread, funPro, constructorKey, descriptor);
return funPro;
}
JSHClass *JSFunction::GetOrCreateInitialJSHClass(JSThread *thread, const JSHandle<JSFunction> &fun)
{
JSTaggedValue protoOrDyn(fun->GetProtoOrDynClass());
if (protoOrDyn.IsJSHClass()) {
return reinterpret_cast<JSHClass *>(protoOrDyn.GetTaggedObject());
}
ObjectFactory *factory = thread->GetEcmaVM()->GetFactory();
JSHandle<JSTaggedValue> proto;
if (!fun->HasFunctionPrototype()) {
proto = JSHandle<JSTaggedValue>::Cast(NewJSFunctionPrototype(thread, factory, fun));
} else {
proto = JSHandle<JSTaggedValue>(thread, fun->GetProtoOrDynClass());
}
JSHandle<JSHClass> dynclass = factory->NewEcmaDynClass(JSObject::SIZE, JSType::JS_OBJECT, proto);
fun->SetProtoOrDynClass(thread, dynclass);
return *dynclass;
}
JSTaggedValue JSFunction::PrototypeGetter(JSThread *thread, const JSHandle<JSObject> &self)
{
JSHandle<JSFunction> func = JSHandle<JSFunction>::Cast(self);
if (!func->HasFunctionPrototype()) {
ObjectFactory *factory = thread->GetEcmaVM()->GetFactory();
NewJSFunctionPrototype(thread, factory, func);
}
return JSFunction::Cast(*self)->GetFunctionPrototype();
}
bool JSFunction::PrototypeSetter(JSThread *thread, const JSHandle<JSObject> &self, const JSHandle<JSTaggedValue> &value,
[[maybe_unused]] bool mayThrow)
{
JSHandle<JSFunction> func(self);
JSTaggedValue protoOrDyn = func->GetProtoOrDynClass();
if (protoOrDyn.IsJSHClass()) {
// need transtion
JSHandle<JSHClass> dynclass(thread, protoOrDyn);
JSHandle<JSHClass> newDynclass = JSHClass::TransitionProto(thread, dynclass, value);
if (value->IsECMAObject()) {
JSObject::Cast(value->GetTaggedObject())->GetJSHClass()->SetIsPrototype(true);
}
func->SetProtoOrDynClass(thread, newDynclass);
} else {
func->SetFunctionPrototype(thread, value.GetTaggedValue());
}
return true;
}
JSTaggedValue JSFunction::NameGetter(JSThread *thread, const JSHandle<JSObject> &self)
{
JSMethod *target = JSHandle<JSFunction>::Cast(self)->GetCallTarget();
if (target->GetPandaFile() == nullptr) {
return JSTaggedValue::Undefined();
}
CString funcName = target->ParseFunctionName();
if (funcName.empty()) {
return thread->GlobalConstants()->GetEmptyString();
}
if (JSHandle<JSFunction>::Cast(self)->GetFunctionKind() == FunctionKind::GETTER_FUNCTION) {
funcName.insert(0, "get ");
}
if (JSHandle<JSFunction>::Cast(self)->GetFunctionKind() == FunctionKind::SETTER_FUNCTION) {
funcName.insert(0, "set ");
}
ObjectFactory *factory = thread->GetEcmaVM()->GetFactory();
return factory->NewFromString(funcName).GetTaggedValue();
}
bool JSFunction::OrdinaryHasInstance(JSThread *thread, const JSHandle<JSTaggedValue> &constructor,
const JSHandle<JSTaggedValue> &obj)
{
// 1. If IsCallable(C) is false, return false.
if (!constructor->IsCallable()) {
return false;
}
// 2. If C has a [[BoundTargetFunction]] internal slot, then
// a. Let BC be the value of C's [[BoundTargetFunction]] internal slot.
// b. Return InstanceofOperator(O,BC) (see 12.9.4).
if (constructor->IsBoundFunction()) {
JSHandle<JSBoundFunction> boundFunction(thread, JSBoundFunction::Cast(constructor->GetTaggedObject()));
JSTaggedValue boundTarget = boundFunction->GetBoundTarget();
return JSObject::InstanceOf(thread, obj, JSHandle<JSTaggedValue>(thread, boundTarget));
}
// 3. If Type(O) is not Object, return false
if (!obj->IsECMAObject()) {
return false;
}
// 4. Let P be Get(C, "prototype").
const GlobalEnvConstants *globalConst = thread->GlobalConstants();
JSHandle<JSTaggedValue> constructorPrototype =
JSTaggedValue::GetProperty(thread, constructor, globalConst->GetHandledPrototypeString()).GetValue();
// 5. ReturnIfAbrupt(P).
// no throw exception, so needn't return
RETURN_VALUE_IF_ABRUPT_COMPLETION(thread, false);
// 6. If Type(P) is not Object, throw a TypeError exception.
if (!constructorPrototype->IsECMAObject()) {
THROW_TYPE_ERROR_AND_RETURN(thread, "HasInstance: is not Object", false);
}
// 7. Repeat
// a.Let O be O.[[GetPrototypeOf]]().
// b.ReturnIfAbrupt(O).
// c.If O is null, return false.
// d.If SameValue(P, O) is true, return true.
JSTaggedValue objPrototype = obj.GetTaggedValue();
while (!objPrototype.IsNull()) {
if (JSTaggedValue::SameValue(objPrototype, constructorPrototype.GetTaggedValue())) {
return true;
}
objPrototype = JSObject::Cast(objPrototype)->GetPrototype(thread);
RETURN_VALUE_IF_ABRUPT_COMPLETION(thread, false);
}
return false;
}
bool JSFunction::MakeConstructor(JSThread *thread, const JSHandle<JSFunction> &func,
const JSHandle<JSTaggedValue> &proto, bool writable)
{
ASSERT_PRINT(proto->IsHeapObject() || proto->IsUndefined(), "proto must be JSObject or Undefined");
ASSERT_PRINT(func->IsConstructor(), "func must be Constructor type");
JSHandle<GlobalEnv> env = thread->GetEcmaVM()->GetGlobalEnv();
const GlobalEnvConstants *globalConst = thread->GlobalConstants();
JSHandle<JSTaggedValue> constructorKey = globalConst->GetHandledConstructorString();
ASSERT_PRINT(func->GetProtoOrDynClass().IsHole() && func->IsExtensible(),
"function doesn't has proto_type property and is extensible object");
ASSERT_PRINT(JSObject::HasProperty(thread, JSHandle<JSObject>(func), constructorKey),
"function must have constructor");
// proto.constructor = func
bool status = false;
if (proto->IsUndefined()) {
// Let prototype be ObjectCreate(%ObjectPrototype%).
JSHandle<JSTaggedValue> objPrototype = env->GetObjectFunctionPrototype();
PropertyDescriptor constructorDesc(thread, JSHandle<JSTaggedValue>::Cast(func), writable, false, true);
status = JSTaggedValue::DefinePropertyOrThrow(thread, objPrototype, constructorKey, constructorDesc);
} else {
PropertyDescriptor constructorDesc(thread, JSHandle<JSTaggedValue>::Cast(func), writable, false, true);
status = JSTaggedValue::DefinePropertyOrThrow(thread, proto, constructorKey, constructorDesc);
}
ASSERT_PRINT(status, "DefineProperty construct failed");
// func.prototype = proto
// Let status be DefinePropertyOrThrow(F, "prototype", PropertyDescriptor{[[Value]]:
// prototype, [[Writable]]: writablePrototype, [[Enumerable]]: false, [[Configurable]]: false}).
func->SetFunctionPrototype(thread, proto.GetTaggedValue());
ASSERT_PRINT(status, "DefineProperty proto_type failed");
return status;
}
JSTaggedValue JSFunction::Call(JSThread *thread, const JSHandle<JSTaggedValue> &func,
const JSHandle<JSTaggedValue> &thisArg, uint32_t argc, const JSTaggedType argv[])
{
// 1. ReturnIfAbrupt(F).
RETURN_EXCEPTION_IF_ABRUPT_COMPLETION(thread);
// 2. If argumentsList was not passed, let argumentsList be a new empty List.
// 3. If IsCallable(F) is false, throw a TypeError exception.
if (!func->IsCallable()) {
THROW_TYPE_ERROR_AND_RETURN(thread, "Callable is false", JSTaggedValue::Exception());
}
if (func->IsJSFunction()) {
return JSFunction::CallInternal(thread, JSHandle<JSFunction>(func), thisArg, argc, argv);
}
if (func->IsBoundFunction()) {
return JSBoundFunction::CallInternal(thread, JSHandle<JSBoundFunction>(func));
}
if (func->IsJSProxy()) {
return JSProxy::CallInternal(thread, JSHandle<JSProxy>(func), thisArg, argc, argv);
}
THROW_TYPE_ERROR_AND_RETURN(thread, "Call NonCallable", JSTaggedValue::Exception());
}
JSTaggedValue JSFunction::Construct(JSThread *thread, const JSHandle<JSTaggedValue> &func, uint32_t argc,
const JSTaggedType argv[], const JSHandle<JSTaggedValue> &newTarget)
{
JSMutableHandle<JSTaggedValue> target(thread, newTarget.GetTaggedValue());
if (target->IsUndefined()) {
target.Update(func.GetTaggedValue());
}
if (!(func->IsConstructor() && target->IsConstructor())) {
THROW_TYPE_ERROR_AND_RETURN(thread, "Constructor is false", JSTaggedValue::Exception());
}
if (func->IsJSFunction()) {
return JSFunction::ConstructInternal(thread, JSHandle<JSFunction>(func), argc, argv, target);
}
if (func->IsBoundFunction()) {
return JSBoundFunction::ConstructInternal(thread, JSHandle<JSBoundFunction>(func), target);
}
if (func->IsJSProxy()) {
return JSProxy::ConstructInternal(thread, JSHandle<JSProxy>(func), argc, argv, target);
}
THROW_TYPE_ERROR_AND_RETURN(thread, "Constructor NonConstructor", JSTaggedValue::Exception());
}
JSTaggedValue JSFunction::Invoke(JSThread *thread, const JSHandle<JSTaggedValue> &thisArg,
const JSHandle<JSTaggedValue> &key, uint32_t argc, const JSTaggedType argv[])
{
ASSERT(JSTaggedValue::IsPropertyKey(key));
JSHandle<JSTaggedValue> func(JSTaggedValue::GetProperty(thread, thisArg, key).GetValue());
return JSFunction::Call(thread, func, thisArg, argc, argv);
}
// [[Call]]
JSTaggedValue JSFunction::CallInternal(JSThread *thread, const JSHandle<JSFunction> &func,
const JSHandle<JSTaggedValue> &thisArg, uint32_t argc, const JSTaggedType argv[])
{
if (!func->IsBuiltinsConstructor() && func->IsClassConstructor()) {
THROW_TYPE_ERROR_AND_RETURN(thread, "class constructor cannot call", JSTaggedValue::Exception());
}
CallParams params;
params.callTarget = ECMAObject::Cast(*func);
params.newTarget = JSTaggedValue::VALUE_UNDEFINED;
params.thisArg = thisArg.GetTaggedType();
params.argc = argc;
params.argv = argv;
return EcmaInterpreter::Execute(thread, params);
}
// [[Construct]]
JSTaggedValue JSFunction::ConstructInternal(JSThread *thread, const JSHandle<JSFunction> &func, uint32_t argc,
const JSTaggedType argv[], const JSHandle<JSTaggedValue> &newTarget)
{
ASSERT(newTarget->IsECMAObject());
if (!func->IsConstructor()) {
THROW_TYPE_ERROR_AND_RETURN(thread, "Constructor is false", JSTaggedValue::Exception());
}
JSHandle<JSTaggedValue> obj(thread, JSTaggedValue::Undefined());
if (!func->IsBuiltinsConstructor() && func->IsBase()) {
ObjectFactory *factory = thread->GetEcmaVM()->GetFactory();
obj = JSHandle<JSTaggedValue>(factory->NewJSObjectByConstructor(func, newTarget));
}
CallParams params;
params.callTarget = ECMAObject::Cast(*func);
params.newTarget = newTarget.GetTaggedType();
params.thisArg = obj.GetTaggedType();
params.argc = argc;
params.argv = argv;
JSTaggedValue resultValue = EcmaInterpreter::Execute(thread, params);
RETURN_EXCEPTION_IF_ABRUPT_COMPLETION(thread);
// 9.3.2 [[Construct]] (argumentsList, newTarget)
if (func->IsBuiltinsConstructor() || resultValue.IsECMAObject()) {
return resultValue;
}
if (func->IsBase()) {
return obj.GetTaggedValue();
}
// derived ctor(sub class) return the obj which created by base ctor(parent class)
if (func->IsDerivedConstructor()) {
return resultValue;
}
if (!resultValue.IsUndefined()) {
THROW_TYPE_ERROR_AND_RETURN(thread, "function is non-constructor", JSTaggedValue::Exception());
}
return obj.GetTaggedValue();
}
JSHandle<JSTaggedValue> JSFunctionBase::GetFunctionName(JSThread *thread, const JSHandle<JSFunctionBase> &func)
{
JSHandle<JSTaggedValue> nameKey = thread->GlobalConstants()->GetHandledNameString();
return JSObject::GetProperty(thread, JSHandle<JSTaggedValue>(func), nameKey).GetValue();
}
bool JSFunctionBase::SetFunctionName(JSThread *thread, const JSHandle<JSFunctionBase> &func,
const JSHandle<JSTaggedValue> &name, const JSHandle<JSTaggedValue> &prefix)
{
ASSERT_PRINT(func->IsExtensible(), "Function must be extensible");
ASSERT_PRINT(name->IsStringOrSymbol(), "name must be string or symbol");
bool needPrefix = false;
ObjectFactory *factory = thread->GetEcmaVM()->GetFactory();
if (!prefix->IsUndefined()) {
ASSERT_PRINT(prefix->IsString(), "prefix must be string");
needPrefix = true;
}
// If Type(name) is Symbol, then
// Let description be names [[Description]] value.
// If description is undefined, let name be the empty String.
// Else, let name be the concatenation of "[", description, and "]".
JSHandle<EcmaString> functionName;
if (name->IsSymbol()) {
JSTaggedValue description = JSHandle<JSSymbol>::Cast(name)->GetDescription();
JSHandle<EcmaString> descriptionHandle(thread, description);
if (description.IsUndefined()) {
functionName = factory->GetEmptyString();
} else {
JSHandle<EcmaString> leftBrackets = factory->NewFromCanBeCompressString("[");
JSHandle<EcmaString> rightBrackets = factory->NewFromCanBeCompressString("]");
functionName = factory->ConcatFromString(leftBrackets, descriptionHandle);
functionName = factory->ConcatFromString(functionName, rightBrackets);
}
} else {
functionName = JSHandle<EcmaString>::Cast(name);
}
EcmaString *newString;
if (needPrefix) {
JSHandle<EcmaString> handlePrefixString = JSTaggedValue::ToString(thread, prefix);
JSHandle<EcmaString> spaceString(factory->NewFromCanBeCompressString(" "));
JSHandle<EcmaString> concatString = factory->ConcatFromString(handlePrefixString, spaceString);
newString = *factory->ConcatFromString(concatString, functionName);
} else {
newString = *functionName;
}
JSHandle<JSTaggedValue> nameHandle(thread, newString);
JSHandle<JSTaggedValue> nameKey = thread->GlobalConstants()->GetHandledNameString();
PropertyDescriptor nameDesc(thread, nameHandle, false, false, true);
JSHandle<JSTaggedValue> funcHandle(func);
return JSTaggedValue::DefinePropertyOrThrow(thread, funcHandle, nameKey, nameDesc);
}
bool JSFunction::SetFunctionLength(JSThread *thread, const JSHandle<JSFunction> &func, JSTaggedValue length, bool cfg)
{
ASSERT_PRINT(func->IsExtensible(), "Function must be extensible");
ASSERT_PRINT(length.IsInteger(), "length must be integer");
JSHandle<JSTaggedValue> lengthKeyHandle = thread->GlobalConstants()->GetHandledLengthString();
ASSERT_PRINT(!JSTaggedValue::Less(thread, JSHandle<JSTaggedValue>(thread, length),
JSHandle<JSTaggedValue>(thread, JSTaggedValue(0))),
"length must be non negtive integer");
PropertyDescriptor lengthDesc(thread, JSHandle<JSTaggedValue>(thread, length), false, false, cfg);
JSHandle<JSTaggedValue> funcHandle(func);
return JSTaggedValue::DefinePropertyOrThrow(thread, funcHandle, lengthKeyHandle, lengthDesc);
}
JSTaggedValue JSBoundFunction::CallInternal(JSThread *thread, const JSHandle<JSBoundFunction> &func)
{
JSHandle<JSTaggedValue> target(thread, func->GetBoundTarget());
JSHandle<JSTaggedValue> boundThis(thread, func->GetBoundThis());
InternalCallParams *params = thread->GetInternalCallParams();
params->MakeBoundArgv(thread, func);
return JSFunction::Call(thread, target, boundThis, params->GetLength(), params->GetArgv());
}
// 9.4.1.2[[Construct]](argumentsList, newTarget)
JSTaggedValue JSBoundFunction::ConstructInternal(JSThread *thread, const JSHandle<JSBoundFunction> &func,
const JSHandle<JSTaggedValue> &newTarget)
{
JSHandle<JSTaggedValue> target(thread, func->GetBoundTarget());
ASSERT(target->IsConstructor());
JSMutableHandle<JSTaggedValue> newTargetMutable(thread, newTarget.GetTaggedValue());
if (JSTaggedValue::SameValue(func.GetTaggedValue(), newTarget.GetTaggedValue())) {
newTargetMutable.Update(target.GetTaggedValue());
}
InternalCallParams *params = thread->GetInternalCallParams();
params->MakeBoundArgv(thread, func);
return JSFunction::Construct(thread, target, params->GetLength(), params->GetArgv(), newTargetMutable);
}
void JSProxyRevocFunction::ProxyRevocFunctions(const JSThread *thread, const JSHandle<JSProxyRevocFunction> &revoker)
{
// 1.Let p be the value of Fs [[RevocableProxy]] internal slot.
JSTaggedValue proxy = revoker->GetRevocableProxy();
// 2.If p is null, return undefined.
if (proxy.IsNull()) {
return;
}
// 3.Set the value of Fs [[RevocableProxy]] internal slot to null.
revoker->SetRevocableProxy(thread, JSTaggedValue::Null());
// 4.Assert: p is a Proxy object.
ASSERT(proxy.IsJSProxy());
JSHandle<JSProxy> proxyHandle(thread, proxy);
// 5 ~ 6 Set internal slot of p to null.
proxyHandle->SetTarget(thread, JSTaggedValue::Null());
proxyHandle->SetHandler(thread, JSTaggedValue::Null());
}
JSTaggedValue JSFunction::AccessCallerArgumentsThrowTypeError([[maybe_unused]] EcmaRuntimeCallInfo *argv)
{
THROW_TYPE_ERROR_AND_RETURN(argv->GetThread(),
"Under strict mode, 'caller' and 'arguments' properties must not be accessed.",
JSTaggedValue::Exception());
}
JSTaggedValue JSIntlBoundFunction::IntlNameGetter(JSThread *thread, [[maybe_unused]] const JSHandle<JSObject> &self)
{
return thread->GlobalConstants()->GetEmptyString();
}
void JSFunction::SetFunctionNameNoPrefix(JSThread *thread, JSFunction *func, JSTaggedValue name)
{
ASSERT_PRINT(func->IsExtensible(), "Function must be extensible");
ObjectFactory *factory = thread->GetEcmaVM()->GetFactory();
JSHandle<JSTaggedValue> funcHandle(thread, func);
{
JSMutableHandle<JSTaggedValue> nameHandle(thread, JSTaggedValue::Undefined());
if (!name.IsSymbol()) {
nameHandle.Update(name);
} else {
JSHandle<JSTaggedValue> nameBegin(thread, name);
JSTaggedValue description = JSSymbol::Cast(name.GetTaggedObject())->GetDescription();
if (description.IsUndefined()) {
nameHandle.Update(thread->GlobalConstants()->GetEmptyString());
} else {
JSHandle<EcmaString> concatName;
JSHandle<EcmaString> leftBrackets = factory->NewFromCanBeCompressString("[");
JSHandle<EcmaString> rightBrackets = factory->NewFromCanBeCompressString("]");
concatName = factory->ConcatFromString(
leftBrackets,
JSHandle<EcmaString>(thread, JSSymbol::Cast(nameBegin->GetHeapObject())->GetDescription()));
concatName = factory->ConcatFromString(concatName, rightBrackets);
nameHandle.Update(concatName.GetTaggedValue());
}
}
PropertyDescriptor nameDesc(thread, nameHandle, false, false, true);
JSTaggedValue::DefinePropertyOrThrow(thread, funcHandle, thread->GlobalConstants()->GetHandledNameString(),
nameDesc);
}
}
JSHandle<JSHClass> JSFunction::GetInstanceJSHClass(JSThread *thread, JSHandle<JSFunction> constructor,
JSHandle<JSTaggedValue> newTarget)
{
JSHandle<JSHClass> ctorInitialJSHClass(thread, JSFunction::GetOrCreateInitialJSHClass(thread, constructor));
// newTarget is construct itself
if (newTarget.GetTaggedValue() == constructor.GetTaggedValue()) {
return ctorInitialJSHClass;
}
// newTarget is derived-class of constructor
if (newTarget->IsJSFunction()) {
JSHandle<JSFunction> newTargetFunc = JSHandle<JSFunction>::Cast(newTarget);
if (newTargetFunc->IsDerivedConstructor()) {
JSTaggedValue newTargetProto = JSHandle<JSObject>::Cast(newTarget)->GetPrototype(thread);
if (newTargetProto == constructor.GetTaggedValue()) {
return GetOrCreateDerivedJSHClass(thread, newTargetFunc, constructor, ctorInitialJSHClass);
}
}
}
// ECMA2015 9.1.15 3.Let proto be Get(constructor, "prototype").
JSMutableHandle<JSTaggedValue> prototype(thread, JSTaggedValue::Undefined());
if (newTarget->IsJSFunction()) {
JSHandle<JSFunction> newTargetFunc = JSHandle<JSFunction>::Cast(newTarget);
FunctionKind kind = newTargetFunc->GetFunctionKind();
if (HasPrototype(kind)) {
prototype.Update(PrototypeGetter(thread, JSHandle<JSObject>::Cast(newTargetFunc)));
}
} else {
// Such case: bound function and define a "prototype" property.
JSHandle<JSTaggedValue> customizePrototype =
JSTaggedValue::GetProperty(thread, newTarget, thread->GlobalConstants()->GetHandledPrototypeString())
.GetValue();
RETURN_HANDLE_IF_ABRUPT_COMPLETION(JSHClass, thread);
prototype.Update(customizePrototype.GetTaggedValue());
// Reload JSHClass of constructor, where the lookup of 'prototype' property may change it.
ctorInitialJSHClass = JSHandle<JSHClass>(thread, JSFunction::GetOrCreateInitialJSHClass(thread, constructor));
}
if (!prototype->IsECMAObject()) {
prototype.Update(constructor->GetFunctionPrototype());
}
JSHandle<JSHClass> newJSHClass = JSHClass::Clone(thread, ctorInitialJSHClass);
newJSHClass->SetPrototype(thread, prototype);
return newJSHClass;
}
JSHandle<JSHClass> JSFunction::GetOrCreateDerivedJSHClass(JSThread *thread, JSHandle<JSFunction> derived,
JSHandle<JSFunction> constructor,
JSHandle<JSHClass> ctorInitialJSHClass)
{
JSTaggedValue protoOrDyn(derived->GetProtoOrDynClass());
// has cached JSHClass, return directly
if (protoOrDyn.IsJSHClass()) {
return JSHandle<JSHClass>(thread, protoOrDyn);
}
JSHandle<JSHClass> newJSHClass = JSHClass::Clone(thread, ctorInitialJSHClass);
// guarante derived has function prototype
JSHandle<JSTaggedValue> prototype(thread, derived->GetProtoOrDynClass());
ASSERT(!prototype->IsHole());
newJSHClass->SetPrototype(thread, prototype);
derived->SetProtoOrDynClass(thread, newJSHClass);
return newJSHClass;
}
// Those interface below is discarded
void JSFunction::InitializeJSFunction(JSThread *thread, [[maybe_unused]] const JSHandle<GlobalEnv> &env,
const JSHandle<JSFunction> &func, FunctionKind kind, bool strict)
{
InitializeJSFunction(thread, func, kind, strict);
}
bool JSFunction::IsDynClass(JSTaggedValue object)
{
return object.IsJSHClass();
}
DynClass *JSFunction::GetOrCreateInitialDynClass(JSThread *thread, const JSHandle<JSFunction> &fun)
{
return reinterpret_cast<DynClass *>(JSFunction::GetOrCreateInitialJSHClass(thread, fun));
}
JSHandle<DynClass> JSFunction::GetInstanceDynClass(JSThread *thread, JSHandle<JSFunction> constructor,
JSHandle<JSTaggedValue> newTarget)
{
return JSHandle<DynClass>(JSFunction::GetInstanceJSHClass(thread, constructor, newTarget));
}
bool JSFunction::NameSetter(JSThread *thread, const JSHandle<JSObject> &self, const JSHandle<JSTaggedValue> &value,
[[maybe_unused]] bool mayThrow)
{
if (self->IsPropertiesDict()) {
// replace setter with value
JSHandle<JSTaggedValue> nameString = thread->GlobalConstants()->GetHandledNameString();
return self->UpdatePropertyInDictionary(thread, nameString.GetTaggedValue(), value.GetTaggedValue());
}
self->SetPropertyInlinedProps(thread, NAME_INLINE_PROPERTY_INDEX, value.GetTaggedValue());
return true;
}
} // namespace panda::ecmascript