2012-01-24 08:43:12 +00:00
|
|
|
// Copyright 2012 the V8 project authors. All rights reserved.
|
2014-04-29 06:42:26 +00:00
|
|
|
// Use of this source code is governed by a BSD-style license that can be
|
|
|
|
// found in the LICENSE file.
|
2008-07-03 15:10:15 +00:00
|
|
|
|
2014-06-20 08:40:11 +00:00
|
|
|
#include "src/accessors.h"
|
2015-08-14 09:41:32 +00:00
|
|
|
|
2018-07-23 11:42:37 +00:00
|
|
|
#include "src/api-inl.h"
|
2014-06-03 08:12:43 +00:00
|
|
|
#include "src/contexts.h"
|
2018-11-13 12:40:24 +00:00
|
|
|
#include "src/counters.h"
|
2014-06-03 08:12:43 +00:00
|
|
|
#include "src/deoptimizer.h"
|
|
|
|
#include "src/execution.h"
|
|
|
|
#include "src/frames-inl.h"
|
2018-04-09 19:11:22 +00:00
|
|
|
#include "src/heap/factory.h"
|
2015-09-01 09:25:19 +00:00
|
|
|
#include "src/isolate-inl.h"
|
2015-05-06 07:51:56 +00:00
|
|
|
#include "src/messages.h"
|
2018-04-30 13:27:37 +00:00
|
|
|
#include "src/objects/api-callbacks.h"
|
2018-07-25 14:11:56 +00:00
|
|
|
#include "src/objects/js-array-inl.h"
|
2018-05-23 13:29:02 +00:00
|
|
|
#include "src/objects/module-inl.h"
|
2014-06-03 08:12:43 +00:00
|
|
|
#include "src/property-details.h"
|
2014-07-14 07:19:49 +00:00
|
|
|
#include "src/prototype.h"
|
2008-07-03 15:10:15 +00:00
|
|
|
|
2009-05-25 10:05:56 +00:00
|
|
|
namespace v8 {
|
|
|
|
namespace internal {
|
2008-07-03 15:10:15 +00:00
|
|
|
|
2014-04-29 10:59:14 +00:00
|
|
|
Handle<AccessorInfo> Accessors::MakeAccessor(
|
2016-10-26 08:59:24 +00:00
|
|
|
Isolate* isolate, Handle<Name> name, AccessorNameGetterCallback getter,
|
2017-10-26 15:18:09 +00:00
|
|
|
AccessorNameBooleanSetterCallback setter) {
|
2014-04-15 13:25:17 +00:00
|
|
|
Factory* factory = isolate->factory();
|
2016-01-18 15:08:36 +00:00
|
|
|
Handle<AccessorInfo> info = factory->NewAccessorInfo();
|
2014-04-28 13:41:12 +00:00
|
|
|
info->set_all_can_read(false);
|
|
|
|
info->set_all_can_write(false);
|
2015-05-18 12:36:49 +00:00
|
|
|
info->set_is_special_data_property(true);
|
2016-05-06 14:10:12 +00:00
|
|
|
info->set_is_sloppy(false);
|
2016-10-28 14:39:33 +00:00
|
|
|
info->set_replace_on_access(false);
|
2018-09-04 08:53:13 +00:00
|
|
|
info->set_getter_side_effect_type(SideEffectType::kHasSideEffect);
|
|
|
|
info->set_setter_side_effect_type(SideEffectType::kHasSideEffect);
|
2016-02-18 15:01:01 +00:00
|
|
|
name = factory->InternalizeName(name);
|
|
|
|
info->set_name(*name);
|
2014-04-15 13:25:17 +00:00
|
|
|
Handle<Object> get = v8::FromCData(isolate, getter);
|
2016-01-27 13:21:53 +00:00
|
|
|
if (setter == nullptr) setter = &ReconfigureToDataProperty;
|
2014-04-15 13:25:17 +00:00
|
|
|
Handle<Object> set = v8::FromCData(isolate, setter);
|
|
|
|
info->set_getter(*get);
|
|
|
|
info->set_setter(*set);
|
2016-04-20 08:01:08 +00:00
|
|
|
Address redirected = info->redirected_getter();
|
2018-04-13 22:28:05 +00:00
|
|
|
if (redirected != kNullAddress) {
|
2016-04-20 08:01:08 +00:00
|
|
|
Handle<Object> js_get = v8::FromCData(isolate, redirected);
|
|
|
|
info->set_js_getter(*js_get);
|
|
|
|
}
|
2014-04-15 13:25:17 +00:00
|
|
|
return info;
|
|
|
|
}
|
|
|
|
|
2018-06-20 16:32:59 +00:00
|
|
|
static V8_INLINE bool CheckForName(Isolate* isolate, Handle<Name> name,
|
2017-11-14 14:15:00 +00:00
|
|
|
Handle<String> property_name, int offset,
|
|
|
|
FieldIndex::Encoding encoding,
|
|
|
|
FieldIndex* index) {
|
2018-06-20 16:32:59 +00:00
|
|
|
if (Name::Equals(isolate, name, property_name)) {
|
2017-11-14 14:15:00 +00:00
|
|
|
*index = FieldIndex::ForInObjectOffset(offset, encoding);
|
2013-09-25 08:19:35 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-02-04 12:44:15 +00:00
|
|
|
// Returns true for properties that are accessors to object fields.
|
|
|
|
// If true, *object_offset contains offset of object field.
|
2018-05-31 14:47:33 +00:00
|
|
|
bool Accessors::IsJSObjectFieldAccessor(Isolate* isolate, Handle<Map> map,
|
|
|
|
Handle<Name> name, FieldIndex* index) {
|
2013-09-25 08:19:35 +00:00
|
|
|
switch (map->instance_type()) {
|
|
|
|
case JS_ARRAY_TYPE:
|
2018-06-20 16:32:59 +00:00
|
|
|
return CheckForName(isolate, name, isolate->factory()->length_string(),
|
2017-11-14 14:15:00 +00:00
|
|
|
JSArray::kLengthOffset, FieldIndex::kTagged, index);
|
2015-04-27 09:28:16 +00:00
|
|
|
default:
|
|
|
|
if (map->instance_type() < FIRST_NONSTRING_TYPE) {
|
2018-06-20 16:32:59 +00:00
|
|
|
return CheckForName(isolate, name, isolate->factory()->length_string(),
|
2018-09-13 12:08:32 +00:00
|
|
|
String::kLengthOffset, FieldIndex::kWord32, index);
|
2015-04-27 09:28:16 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-04-19 18:08:02 +00:00
|
|
|
V8_WARN_UNUSED_RESULT MaybeHandle<Object>
|
2018-07-18 16:05:36 +00:00
|
|
|
Accessors::ReplaceAccessorWithDataProperty(Handle<Object> receiver,
|
2018-04-19 18:08:02 +00:00
|
|
|
Handle<JSObject> holder,
|
|
|
|
Handle<Name> name,
|
|
|
|
Handle<Object> value) {
|
2016-01-27 13:21:53 +00:00
|
|
|
LookupIterator it(receiver, name, holder,
|
|
|
|
LookupIterator::OWN_SKIP_INTERCEPTOR);
|
|
|
|
// Skip any access checks we might hit. This accessor should never hit in a
|
|
|
|
// situation where the caller does not have access.
|
|
|
|
if (it.state() == LookupIterator::ACCESS_CHECK) {
|
|
|
|
CHECK(it.HasAccess());
|
|
|
|
it.Next();
|
|
|
|
}
|
2016-05-13 11:33:17 +00:00
|
|
|
DCHECK(holder.is_identical_to(it.GetHolder<JSObject>()));
|
2016-01-27 13:21:53 +00:00
|
|
|
CHECK_EQ(LookupIterator::ACCESSOR, it.state());
|
|
|
|
it.ReconfigureDataProperty(value, it.property_attributes());
|
|
|
|
return value;
|
|
|
|
}
|
|
|
|
|
2016-05-09 11:38:07 +00:00
|
|
|
|
2018-02-27 16:13:33 +00:00
|
|
|
//
|
|
|
|
// Accessors::ReconfigureToDataProperty
|
|
|
|
//
|
2016-01-27 13:21:53 +00:00
|
|
|
void Accessors::ReconfigureToDataProperty(
|
|
|
|
v8::Local<v8::Name> key, v8::Local<v8::Value> val,
|
2016-10-26 08:59:24 +00:00
|
|
|
const v8::PropertyCallbackInfo<v8::Boolean>& info) {
|
2016-01-27 13:21:53 +00:00
|
|
|
i::Isolate* isolate = reinterpret_cast<i::Isolate*>(info.GetIsolate());
|
2017-07-25 15:22:43 +00:00
|
|
|
RuntimeCallTimerScope stats_scope(
|
2017-11-29 15:50:22 +00:00
|
|
|
isolate, RuntimeCallCounterId::kReconfigureToDataProperty);
|
2016-01-27 13:21:53 +00:00
|
|
|
HandleScope scope(isolate);
|
2016-05-09 11:38:07 +00:00
|
|
|
Handle<Object> receiver = Utils::OpenHandle(*info.This());
|
2016-01-27 13:21:53 +00:00
|
|
|
Handle<JSObject> holder =
|
|
|
|
Handle<JSObject>::cast(Utils::OpenHandle(*info.Holder()));
|
|
|
|
Handle<Name> name = Utils::OpenHandle(*key);
|
|
|
|
Handle<Object> value = Utils::OpenHandle(*val);
|
2018-07-18 16:05:36 +00:00
|
|
|
MaybeHandle<Object> result =
|
|
|
|
Accessors::ReplaceAccessorWithDataProperty(receiver, holder, name, value);
|
2016-10-26 08:59:24 +00:00
|
|
|
if (result.is_null()) {
|
|
|
|
isolate->OptionalRescheduleException(false);
|
|
|
|
} else {
|
|
|
|
info.GetReturnValue().Set(true);
|
|
|
|
}
|
2016-01-27 13:21:53 +00:00
|
|
|
}
|
2013-09-25 08:19:35 +00:00
|
|
|
|
2018-02-27 16:13:33 +00:00
|
|
|
|
2014-08-25 09:12:22 +00:00
|
|
|
//
|
|
|
|
// Accessors::ArgumentsIterator
|
|
|
|
//
|
|
|
|
|
|
|
|
|
|
|
|
void Accessors::ArgumentsIteratorGetter(
|
|
|
|
v8::Local<v8::Name> name, const v8::PropertyCallbackInfo<v8::Value>& info) {
|
|
|
|
i::Isolate* isolate = reinterpret_cast<i::Isolate*>(info.GetIsolate());
|
|
|
|
DisallowHeapAllocation no_allocation;
|
|
|
|
HandleScope scope(isolate);
|
|
|
|
Object* result = isolate->native_context()->array_values_iterator();
|
|
|
|
info.GetReturnValue().Set(Utils::ToLocal(Handle<Object>(result, isolate)));
|
|
|
|
}
|
|
|
|
|
2017-10-27 09:06:44 +00:00
|
|
|
Handle<AccessorInfo> Accessors::MakeArgumentsIteratorInfo(Isolate* isolate) {
|
2014-11-13 08:47:52 +00:00
|
|
|
Handle<Name> name = isolate->factory()->iterator_symbol();
|
2017-10-26 15:18:09 +00:00
|
|
|
return MakeAccessor(isolate, name, &ArgumentsIteratorGetter, nullptr);
|
2014-08-25 09:12:22 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
//
|
|
|
|
// Accessors::ArrayLength
|
|
|
|
//
|
|
|
|
|
|
|
|
|
2014-04-28 14:59:29 +00:00
|
|
|
void Accessors::ArrayLengthGetter(
|
2014-08-20 15:25:13 +00:00
|
|
|
v8::Local<v8::Name> name,
|
2014-04-28 14:59:29 +00:00
|
|
|
const v8::PropertyCallbackInfo<v8::Value>& info) {
|
|
|
|
i::Isolate* isolate = reinterpret_cast<i::Isolate*>(info.GetIsolate());
|
2017-11-29 15:50:22 +00:00
|
|
|
RuntimeCallTimerScope timer(isolate,
|
|
|
|
RuntimeCallCounterId::kArrayLengthGetter);
|
2014-04-28 14:59:29 +00:00
|
|
|
DisallowHeapAllocation no_allocation;
|
2013-11-05 13:47:51 +00:00
|
|
|
HandleScope scope(isolate);
|
2014-07-23 20:11:33 +00:00
|
|
|
JSArray* holder = JSArray::cast(*Utils::OpenHandle(*info.Holder()));
|
|
|
|
Object* result = holder->length();
|
2014-04-28 14:59:29 +00:00
|
|
|
info.GetReturnValue().Set(Utils::ToLocal(Handle<Object>(result, isolate)));
|
|
|
|
}
|
2013-11-05 13:47:51 +00:00
|
|
|
|
2014-04-28 14:59:29 +00:00
|
|
|
void Accessors::ArrayLengthSetter(
|
2016-10-26 08:59:24 +00:00
|
|
|
v8::Local<v8::Name> name, v8::Local<v8::Value> val,
|
|
|
|
const v8::PropertyCallbackInfo<v8::Boolean>& info) {
|
2014-04-28 14:59:29 +00:00
|
|
|
i::Isolate* isolate = reinterpret_cast<i::Isolate*>(info.GetIsolate());
|
2017-11-29 15:50:22 +00:00
|
|
|
RuntimeCallTimerScope timer(isolate,
|
|
|
|
RuntimeCallCounterId::kArrayLengthSetter);
|
2014-04-28 14:59:29 +00:00
|
|
|
HandleScope scope(isolate);
|
2008-07-03 15:10:15 +00:00
|
|
|
|
2018-07-04 09:10:05 +00:00
|
|
|
DCHECK(Utils::OpenHandle(*name)->SameValue(
|
|
|
|
ReadOnlyRoots(isolate).length_string()));
|
2016-11-30 19:13:18 +00:00
|
|
|
|
2016-05-09 11:38:07 +00:00
|
|
|
Handle<JSReceiver> object = Utils::OpenHandle(*info.Holder());
|
2015-06-19 15:27:40 +00:00
|
|
|
Handle<JSArray> array = Handle<JSArray>::cast(object);
|
|
|
|
Handle<Object> length_obj = Utils::OpenHandle(*val);
|
|
|
|
|
2016-11-30 19:13:18 +00:00
|
|
|
bool was_readonly = JSArray::HasReadOnlyLength(array);
|
|
|
|
|
2015-06-19 15:27:40 +00:00
|
|
|
uint32_t length = 0;
|
2015-10-15 16:03:20 +00:00
|
|
|
if (!JSArray::AnythingToArrayLength(isolate, length_obj, &length)) {
|
|
|
|
isolate->OptionalRescheduleException(false);
|
|
|
|
return;
|
2014-04-28 14:59:29 +00:00
|
|
|
}
|
2008-07-03 15:10:15 +00:00
|
|
|
|
2016-11-30 19:13:18 +00:00
|
|
|
if (!was_readonly && V8_UNLIKELY(JSArray::HasReadOnlyLength(array)) &&
|
|
|
|
length != array->length()->Number()) {
|
|
|
|
// AnythingToArrayLength() may have called setter re-entrantly and modified
|
|
|
|
// its property descriptor. Don't perform this check if "length" was
|
|
|
|
// previously readonly, as this may have been called during
|
|
|
|
// DefineOwnPropertyIgnoreAttributes().
|
|
|
|
if (info.ShouldThrowOnError()) {
|
|
|
|
Factory* factory = isolate->factory();
|
|
|
|
isolate->Throw(*factory->NewTypeError(
|
|
|
|
MessageTemplate::kStrictReadOnlyProperty, Utils::OpenHandle(*name),
|
|
|
|
i::Object::TypeOf(isolate, object), object));
|
|
|
|
isolate->OptionalRescheduleException(false);
|
|
|
|
} else {
|
|
|
|
info.GetReturnValue().Set(false);
|
|
|
|
}
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2016-04-22 09:01:36 +00:00
|
|
|
JSArray::SetLength(array, length);
|
2016-01-22 09:53:29 +00:00
|
|
|
|
2016-10-26 08:59:24 +00:00
|
|
|
uint32_t actual_new_len = 0;
|
|
|
|
CHECK(array->length()->ToArrayLength(&actual_new_len));
|
|
|
|
// Fail if there were non-deletable elements.
|
|
|
|
if (actual_new_len != length) {
|
|
|
|
if (info.ShouldThrowOnError()) {
|
2016-01-22 09:53:29 +00:00
|
|
|
Factory* factory = isolate->factory();
|
|
|
|
isolate->Throw(*factory->NewTypeError(
|
|
|
|
MessageTemplate::kStrictDeleteProperty,
|
|
|
|
factory->NewNumberFromUint(actual_new_len - 1), array));
|
|
|
|
isolate->OptionalRescheduleException(false);
|
2016-10-26 08:59:24 +00:00
|
|
|
} else {
|
|
|
|
info.GetReturnValue().Set(false);
|
2016-01-22 09:53:29 +00:00
|
|
|
}
|
2016-10-26 08:59:24 +00:00
|
|
|
} else {
|
|
|
|
info.GetReturnValue().Set(true);
|
2016-01-22 09:53:29 +00:00
|
|
|
}
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
2017-10-27 09:06:44 +00:00
|
|
|
Handle<AccessorInfo> Accessors::MakeArrayLengthInfo(Isolate* isolate) {
|
2017-10-26 15:18:09 +00:00
|
|
|
return MakeAccessor(isolate, isolate->factory()->length_string(),
|
|
|
|
&ArrayLengthGetter, &ArrayLengthSetter);
|
2016-10-07 19:37:04 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
//
|
|
|
|
// Accessors::ModuleNamespaceEntry
|
|
|
|
//
|
|
|
|
|
|
|
|
void Accessors::ModuleNamespaceEntryGetter(
|
|
|
|
v8::Local<v8::Name> name, const v8::PropertyCallbackInfo<v8::Value>& info) {
|
|
|
|
i::Isolate* isolate = reinterpret_cast<i::Isolate*>(info.GetIsolate());
|
|
|
|
HandleScope scope(isolate);
|
|
|
|
JSModuleNamespace* holder =
|
|
|
|
JSModuleNamespace::cast(*Utils::OpenHandle(*info.Holder()));
|
|
|
|
Handle<Object> result;
|
2018-06-20 12:44:39 +00:00
|
|
|
if (!holder
|
|
|
|
->GetExport(isolate, Handle<String>::cast(Utils::OpenHandle(*name)))
|
2016-10-07 19:37:04 +00:00
|
|
|
.ToHandle(&result)) {
|
|
|
|
isolate->OptionalRescheduleException(false);
|
|
|
|
} else {
|
|
|
|
info.GetReturnValue().Set(Utils::ToLocal(result));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void Accessors::ModuleNamespaceEntrySetter(
|
|
|
|
v8::Local<v8::Name> name, v8::Local<v8::Value> val,
|
2016-10-26 08:59:24 +00:00
|
|
|
const v8::PropertyCallbackInfo<v8::Boolean>& info) {
|
2016-10-07 19:37:04 +00:00
|
|
|
i::Isolate* isolate = reinterpret_cast<i::Isolate*>(info.GetIsolate());
|
|
|
|
HandleScope scope(isolate);
|
|
|
|
Factory* factory = isolate->factory();
|
|
|
|
Handle<JSModuleNamespace> holder =
|
|
|
|
Handle<JSModuleNamespace>::cast(Utils::OpenHandle(*info.Holder()));
|
|
|
|
|
|
|
|
if (info.ShouldThrowOnError()) {
|
|
|
|
isolate->Throw(*factory->NewTypeError(
|
|
|
|
MessageTemplate::kStrictReadOnlyProperty, Utils::OpenHandle(*name),
|
|
|
|
i::Object::TypeOf(isolate, holder), holder));
|
|
|
|
isolate->OptionalRescheduleException(false);
|
|
|
|
} else {
|
2016-10-26 08:59:24 +00:00
|
|
|
info.GetReturnValue().Set(false);
|
2016-10-07 19:37:04 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-10-27 09:06:44 +00:00
|
|
|
Handle<AccessorInfo> Accessors::MakeModuleNamespaceEntryInfo(
|
|
|
|
Isolate* isolate, Handle<String> name) {
|
2016-10-07 19:37:04 +00:00
|
|
|
return MakeAccessor(isolate, name, &ModuleNamespaceEntryGetter,
|
2017-10-26 15:18:09 +00:00
|
|
|
&ModuleNamespaceEntrySetter);
|
2014-04-28 14:59:29 +00:00
|
|
|
}
|
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
//
|
|
|
|
// Accessors::StringLength
|
|
|
|
//
|
|
|
|
|
2014-04-15 13:25:17 +00:00
|
|
|
void Accessors::StringLengthGetter(
|
2014-08-20 15:25:13 +00:00
|
|
|
v8::Local<v8::Name> name,
|
2014-04-15 13:25:17 +00:00
|
|
|
const v8::PropertyCallbackInfo<v8::Value>& info) {
|
|
|
|
i::Isolate* isolate = reinterpret_cast<i::Isolate*>(info.GetIsolate());
|
2017-11-29 15:50:22 +00:00
|
|
|
RuntimeCallTimerScope timer(isolate,
|
|
|
|
RuntimeCallCounterId::kStringLengthGetter);
|
2014-04-15 13:25:17 +00:00
|
|
|
DisallowHeapAllocation no_allocation;
|
|
|
|
HandleScope scope(isolate);
|
2014-07-23 20:11:33 +00:00
|
|
|
|
|
|
|
// We have a slight impedance mismatch between the external API and the way we
|
|
|
|
// use callbacks internally: Externally, callbacks can only be used with
|
|
|
|
// v8::Object, but internally we have callbacks on entities which are higher
|
|
|
|
// in the hierarchy, in this case for String values.
|
|
|
|
|
|
|
|
Object* value = *Utils::OpenHandle(*v8::Local<v8::Value>(info.This()));
|
|
|
|
if (!value->IsString()) {
|
|
|
|
// Not a string value. That means that we either got a String wrapper or
|
|
|
|
// a Value with a String wrapper in its prototype chain.
|
|
|
|
value = JSValue::cast(*Utils::OpenHandle(*info.Holder()))->value();
|
2014-04-15 13:25:17 +00:00
|
|
|
}
|
2014-07-23 20:11:33 +00:00
|
|
|
Object* result = Smi::FromInt(String::cast(value)->length());
|
2014-04-15 13:25:17 +00:00
|
|
|
info.GetReturnValue().Set(Utils::ToLocal(Handle<Object>(result, isolate)));
|
|
|
|
}
|
2014-04-10 12:00:36 +00:00
|
|
|
|
2017-10-27 09:06:44 +00:00
|
|
|
Handle<AccessorInfo> Accessors::MakeStringLengthInfo(Isolate* isolate) {
|
2016-01-27 13:21:53 +00:00
|
|
|
return MakeAccessor(isolate, isolate->factory()->length_string(),
|
2017-10-26 15:18:09 +00:00
|
|
|
&StringLengthGetter, nullptr);
|
2014-04-15 13:25:17 +00:00
|
|
|
}
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
//
|
|
|
|
// Accessors::FunctionPrototype
|
|
|
|
//
|
|
|
|
|
2016-06-27 18:03:59 +00:00
|
|
|
static Handle<Object> GetFunctionPrototype(Isolate* isolate,
|
|
|
|
Handle<JSFunction> function) {
|
|
|
|
if (!function->has_prototype()) {
|
|
|
|
Handle<Object> proto = isolate->factory()->NewFunctionPrototype(function);
|
|
|
|
JSFunction::SetPrototype(function, proto);
|
|
|
|
}
|
|
|
|
return Handle<Object>(function->prototype(), isolate);
|
|
|
|
}
|
|
|
|
|
2014-04-24 08:35:53 +00:00
|
|
|
void Accessors::FunctionPrototypeGetter(
|
2014-08-20 15:25:13 +00:00
|
|
|
v8::Local<v8::Name> name,
|
2014-04-24 08:35:53 +00:00
|
|
|
const v8::PropertyCallbackInfo<v8::Value>& info) {
|
|
|
|
i::Isolate* isolate = reinterpret_cast<i::Isolate*>(info.GetIsolate());
|
2017-07-26 22:34:21 +00:00
|
|
|
RuntimeCallTimerScope timer(isolate,
|
2017-11-29 15:50:22 +00:00
|
|
|
RuntimeCallCounterId::kFunctionPrototypeGetter);
|
2014-04-17 09:12:19 +00:00
|
|
|
HandleScope scope(isolate);
|
2014-07-23 20:11:33 +00:00
|
|
|
Handle<JSFunction> function =
|
|
|
|
Handle<JSFunction>::cast(Utils::OpenHandle(*info.Holder()));
|
2018-07-26 11:32:20 +00:00
|
|
|
DCHECK(function->has_prototype_property());
|
2016-06-27 18:03:59 +00:00
|
|
|
Handle<Object> result = GetFunctionPrototype(isolate, function);
|
2014-04-24 08:35:53 +00:00
|
|
|
info.GetReturnValue().Set(Utils::ToLocal(result));
|
2014-04-17 09:12:19 +00:00
|
|
|
}
|
|
|
|
|
2014-04-24 08:35:53 +00:00
|
|
|
void Accessors::FunctionPrototypeSetter(
|
2016-10-26 08:59:24 +00:00
|
|
|
v8::Local<v8::Name> name, v8::Local<v8::Value> val,
|
|
|
|
const v8::PropertyCallbackInfo<v8::Boolean>& info) {
|
2014-04-24 08:35:53 +00:00
|
|
|
i::Isolate* isolate = reinterpret_cast<i::Isolate*>(info.GetIsolate());
|
2017-07-26 22:34:21 +00:00
|
|
|
RuntimeCallTimerScope timer(isolate,
|
2017-11-29 15:50:22 +00:00
|
|
|
RuntimeCallCounterId::kFunctionPrototypeSetter);
|
2014-04-24 08:35:53 +00:00
|
|
|
HandleScope scope(isolate);
|
|
|
|
Handle<Object> value = Utils::OpenHandle(*val);
|
2014-07-24 16:42:54 +00:00
|
|
|
Handle<JSFunction> object =
|
|
|
|
Handle<JSFunction>::cast(Utils::OpenHandle(*info.Holder()));
|
2018-07-26 11:32:20 +00:00
|
|
|
DCHECK(object->has_prototype_property());
|
2017-04-19 21:55:04 +00:00
|
|
|
JSFunction::SetPrototype(object, value);
|
|
|
|
info.GetReturnValue().Set(true);
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
2017-10-27 09:06:44 +00:00
|
|
|
Handle<AccessorInfo> Accessors::MakeFunctionPrototypeInfo(Isolate* isolate) {
|
2017-10-26 15:18:09 +00:00
|
|
|
return MakeAccessor(isolate, isolate->factory()->prototype_string(),
|
|
|
|
&FunctionPrototypeGetter, &FunctionPrototypeSetter);
|
2014-04-24 08:35:53 +00:00
|
|
|
}
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
|
|
|
|
//
|
|
|
|
// Accessors::FunctionLength
|
|
|
|
//
|
|
|
|
|
|
|
|
|
2014-04-24 11:24:13 +00:00
|
|
|
void Accessors::FunctionLengthGetter(
|
2014-08-20 15:25:13 +00:00
|
|
|
v8::Local<v8::Name> name,
|
2014-04-24 11:24:13 +00:00
|
|
|
const v8::PropertyCallbackInfo<v8::Value>& info) {
|
|
|
|
i::Isolate* isolate = reinterpret_cast<i::Isolate*>(info.GetIsolate());
|
2017-11-29 15:50:22 +00:00
|
|
|
RuntimeCallTimerScope timer(isolate,
|
|
|
|
RuntimeCallCounterId::kFunctionLengthGetter);
|
2013-02-27 13:22:29 +00:00
|
|
|
HandleScope scope(isolate);
|
2014-07-23 20:11:33 +00:00
|
|
|
Handle<JSFunction> function =
|
|
|
|
Handle<JSFunction>::cast(Utils::OpenHandle(*info.Holder()));
|
2017-10-04 11:38:34 +00:00
|
|
|
int length = 0;
|
|
|
|
if (!JSFunction::GetLength(isolate, function).To(&length)) {
|
2016-04-07 13:36:40 +00:00
|
|
|
isolate->OptionalRescheduleException(false);
|
2014-04-24 11:24:13 +00:00
|
|
|
}
|
2017-10-04 11:38:34 +00:00
|
|
|
Handle<Object> result(Smi::FromInt(length), isolate);
|
2014-04-24 11:24:13 +00:00
|
|
|
info.GetReturnValue().Set(Utils::ToLocal(result));
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
2017-10-27 09:06:44 +00:00
|
|
|
Handle<AccessorInfo> Accessors::MakeFunctionLengthInfo(Isolate* isolate) {
|
2016-01-27 13:21:53 +00:00
|
|
|
return MakeAccessor(isolate, isolate->factory()->length_string(),
|
2017-10-26 15:18:09 +00:00
|
|
|
&FunctionLengthGetter, &ReconfigureToDataProperty);
|
2014-04-24 11:24:13 +00:00
|
|
|
}
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
|
|
|
|
//
|
|
|
|
// Accessors::FunctionName
|
|
|
|
//
|
|
|
|
|
|
|
|
|
2014-04-28 08:26:35 +00:00
|
|
|
void Accessors::FunctionNameGetter(
|
2014-08-20 15:25:13 +00:00
|
|
|
v8::Local<v8::Name> name,
|
2014-04-28 08:26:35 +00:00
|
|
|
const v8::PropertyCallbackInfo<v8::Value>& info) {
|
|
|
|
i::Isolate* isolate = reinterpret_cast<i::Isolate*>(info.GetIsolate());
|
|
|
|
HandleScope scope(isolate);
|
2014-07-23 20:11:33 +00:00
|
|
|
Handle<JSFunction> function =
|
|
|
|
Handle<JSFunction>::cast(Utils::OpenHandle(*info.Holder()));
|
2016-04-07 13:36:40 +00:00
|
|
|
Handle<Object> result = JSFunction::GetName(isolate, function);
|
2014-04-28 08:26:35 +00:00
|
|
|
info.GetReturnValue().Set(Utils::ToLocal(result));
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
2017-10-27 09:06:44 +00:00
|
|
|
Handle<AccessorInfo> Accessors::MakeFunctionNameInfo(Isolate* isolate) {
|
2016-01-27 13:21:53 +00:00
|
|
|
return MakeAccessor(isolate, isolate->factory()->name_string(),
|
2017-10-26 15:18:09 +00:00
|
|
|
&FunctionNameGetter, &ReconfigureToDataProperty);
|
2014-04-28 08:26:35 +00:00
|
|
|
}
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
|
|
|
|
//
|
|
|
|
// Accessors::FunctionArguments
|
|
|
|
//
|
|
|
|
|
2017-11-29 13:18:33 +00:00
|
|
|
namespace {
|
2010-12-07 11:31:57 +00:00
|
|
|
|
2017-11-29 13:18:33 +00:00
|
|
|
Handle<JSObject> ArgumentsForInlinedFunction(JavaScriptFrame* frame,
|
|
|
|
int inlined_frame_index) {
|
|
|
|
Isolate* isolate = frame->isolate();
|
2013-02-25 14:46:09 +00:00
|
|
|
Factory* factory = isolate->factory();
|
The current
version is passing all the existing test + a bunch of new tests
(packaged in the change list, too).
The patch extends the SlotRef object to describe captured and duplicated
objects. Since the SlotRefs are not independent of each other anymore,
there is a new SlotRefValueBuilder class that stores the SlotRefs and
later materializes the objects from the SlotRefs.
Note that unlike the previous implementation of SlotRefs, we now build
the SlotRef entries for the entire frame, not just the particular
function. This is because duplicate objects might refer to previous
captured objects (that might live inside other inlined function's part
of the frame).
We also need to store the materialized objects between other potential
invocations of the same arguments object so that we materialize each
captured object at most once. The materialized objects of frames live
in the new MaterielizedObjectStore object (contained in Isolate),
indexed by the frame's FP address. Each argument materialization (and
deoptimization) tries to lookup its captured objects in the store before
building new ones. Deoptimization also removes the materialized objects
from the store. We also schedule a lazy deopt to be sure that we always
get rid of the materialized objects and that the optmized function
adopts the materialized objects (instead of happily computing with its
captured representations).
Concerns:
- Is the FP address the right key for a frame? (Note that deoptimizer's
representation of frame is different from the argument object
materializer's one - it is not easy to find common ground.)
- Performance is suboptimal in several places, but a quick local run of
benchmarks does not seem to show a perf hit. Examples of possible
improvements: smarter generation of SlotRefs (build other functions'
SlotRefs only for captured objects and only if necessary), smarter
lookup of stored materialized objects.
- Ideally, we would like to share the code for argument materialization
with deoptimizer's materializer. However, the supporting data structures
(mainly the frame descriptor) are quite different in each case, so it
looks more like a separate project.
Thanks for any feedback.
R=danno@chromium.org, mstarzinger@chromium.org
LOG=N
BUG=
Committed: https://code.google.com/p/v8/source/detail?r=18918
Review URL: https://codereview.chromium.org/103243005
git-svn-id: http://v8.googlecode.com/svn/branches/bleeding_edge@18936 ce2b1a6d-e550-0410-aec6-3dcde31c8c00
2014-01-30 10:33:53 +00:00
|
|
|
|
2015-06-08 10:04:51 +00:00
|
|
|
TranslatedState translated_values(frame);
|
2017-08-01 12:10:32 +00:00
|
|
|
translated_values.Prepare(frame->fp());
|
2015-06-08 10:04:51 +00:00
|
|
|
|
|
|
|
int argument_count = 0;
|
|
|
|
TranslatedFrame* translated_frame =
|
|
|
|
translated_values.GetArgumentsInfoFromJSFrameIndex(inlined_frame_index,
|
|
|
|
&argument_count);
|
|
|
|
TranslatedFrame::iterator iter = translated_frame->begin();
|
|
|
|
|
2017-11-29 13:18:33 +00:00
|
|
|
// Materialize the function.
|
|
|
|
bool should_deoptimize = iter->IsMaterializedObject();
|
|
|
|
Handle<JSFunction> function = Handle<JSFunction>::cast(iter->GetValue());
|
2015-06-10 11:52:35 +00:00
|
|
|
iter++;
|
|
|
|
|
2015-06-08 10:04:51 +00:00
|
|
|
// Skip the receiver.
|
|
|
|
iter++;
|
|
|
|
argument_count--;
|
|
|
|
|
2010-12-07 11:31:57 +00:00
|
|
|
Handle<JSObject> arguments =
|
2017-11-29 13:18:33 +00:00
|
|
|
factory->NewArgumentsObject(function, argument_count);
|
2015-06-08 10:04:51 +00:00
|
|
|
Handle<FixedArray> array = factory->NewFixedArray(argument_count);
|
|
|
|
for (int i = 0; i < argument_count; ++i) {
|
2016-11-29 11:34:07 +00:00
|
|
|
// If we materialize any object, we should deoptimize the frame because we
|
|
|
|
// might alias an object that was eliminated by escape analysis.
|
2015-06-08 10:04:51 +00:00
|
|
|
should_deoptimize = should_deoptimize || iter->IsMaterializedObject();
|
|
|
|
Handle<Object> value = iter->GetValue();
|
2010-12-07 11:31:57 +00:00
|
|
|
array->set(i, *value);
|
2015-06-08 10:04:51 +00:00
|
|
|
iter++;
|
2010-12-07 11:31:57 +00:00
|
|
|
}
|
|
|
|
arguments->set_elements(*array);
|
|
|
|
|
2015-06-08 10:04:51 +00:00
|
|
|
if (should_deoptimize) {
|
2016-11-29 11:34:07 +00:00
|
|
|
translated_values.StoreMaterializedValuesAndDeopt(frame);
|
2015-06-08 10:04:51 +00:00
|
|
|
}
|
|
|
|
|
2010-12-07 11:31:57 +00:00
|
|
|
// Return the freshly allocated arguments object.
|
2014-04-24 14:23:15 +00:00
|
|
|
return arguments;
|
2010-12-07 11:31:57 +00:00
|
|
|
}
|
|
|
|
|
2017-11-29 13:18:33 +00:00
|
|
|
int FindFunctionInFrame(JavaScriptFrame* frame, Handle<JSFunction> function) {
|
2017-08-28 12:10:52 +00:00
|
|
|
std::vector<FrameSummary> frames;
|
2017-01-13 10:29:13 +00:00
|
|
|
frame->Summarize(&frames);
|
2017-08-28 12:10:52 +00:00
|
|
|
for (size_t i = frames.size(); i != 0; i--) {
|
|
|
|
if (*frames[i - 1].AsJavaScript().function() == *function) {
|
|
|
|
return static_cast<int>(i) - 1;
|
|
|
|
}
|
2014-04-24 14:23:15 +00:00
|
|
|
}
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2017-11-29 13:18:33 +00:00
|
|
|
Handle<JSObject> GetFrameArguments(Isolate* isolate,
|
|
|
|
JavaScriptFrameIterator* it,
|
|
|
|
int function_index) {
|
|
|
|
JavaScriptFrame* frame = it->frame();
|
2014-04-24 14:23:15 +00:00
|
|
|
|
2017-11-29 13:18:33 +00:00
|
|
|
if (function_index > 0) {
|
|
|
|
// The function in question was inlined. Inlined functions have the
|
|
|
|
// correct number of arguments and no allocated arguments object, so
|
|
|
|
// we can construct a fresh one by interpreting the function's
|
|
|
|
// deoptimization input data.
|
|
|
|
return ArgumentsForInlinedFunction(frame, function_index);
|
|
|
|
}
|
2016-02-11 07:12:49 +00:00
|
|
|
|
2017-11-29 13:18:33 +00:00
|
|
|
// Find the frame that holds the actual arguments passed to the function.
|
|
|
|
if (it->frame()->has_adapted_arguments()) {
|
|
|
|
it->AdvanceOneFrame();
|
|
|
|
DCHECK(it->frame()->is_arguments_adaptor());
|
|
|
|
}
|
|
|
|
frame = it->frame();
|
2011-02-02 15:08:29 +00:00
|
|
|
|
2017-11-29 13:18:33 +00:00
|
|
|
// Get the number of arguments and construct an arguments object
|
|
|
|
// mirror for the right frame and the underlying function.
|
|
|
|
const int length = frame->ComputeParametersCount();
|
|
|
|
Handle<JSFunction> function(frame->function(), isolate);
|
|
|
|
Handle<JSObject> arguments =
|
|
|
|
isolate->factory()->NewArgumentsObject(function, length);
|
|
|
|
Handle<FixedArray> array = isolate->factory()->NewFixedArray(length);
|
|
|
|
|
|
|
|
// Copy the parameters to the arguments object.
|
|
|
|
DCHECK(array->length() == length);
|
|
|
|
for (int i = 0; i < length; i++) {
|
|
|
|
Object* value = frame->GetParameter(i);
|
|
|
|
if (value->IsTheHole(isolate)) {
|
|
|
|
// Generators currently use holes as dummy arguments when resuming. We
|
|
|
|
// must not leak those.
|
|
|
|
DCHECK(IsResumableFunction(function->shared()->kind()));
|
2018-07-04 09:10:05 +00:00
|
|
|
value = ReadOnlyRoots(isolate).undefined_value();
|
2016-07-06 08:10:08 +00:00
|
|
|
}
|
2017-11-29 13:18:33 +00:00
|
|
|
array->set(i, value);
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
2017-11-29 13:18:33 +00:00
|
|
|
arguments->set_elements(*array);
|
2008-07-03 15:10:15 +00:00
|
|
|
|
2017-11-29 13:18:33 +00:00
|
|
|
// Return the freshly allocated arguments object.
|
|
|
|
return arguments;
|
2014-04-24 14:23:15 +00:00
|
|
|
}
|
|
|
|
|
2016-02-11 07:12:49 +00:00
|
|
|
} // namespace
|
|
|
|
|
2017-11-29 13:18:33 +00:00
|
|
|
Handle<JSObject> Accessors::FunctionGetArguments(JavaScriptFrame* frame,
|
|
|
|
int inlined_jsframe_index) {
|
|
|
|
Isolate* isolate = frame->isolate();
|
|
|
|
Address requested_frame_fp = frame->fp();
|
|
|
|
// Forward a frame iterator to the requested frame. This is needed because we
|
|
|
|
// potentially need for advance it to the arguments adaptor frame later.
|
|
|
|
for (JavaScriptFrameIterator it(isolate); !it.done(); it.Advance()) {
|
|
|
|
if (it.frame()->fp() != requested_frame_fp) continue;
|
|
|
|
return GetFrameArguments(isolate, &it, inlined_jsframe_index);
|
|
|
|
}
|
|
|
|
UNREACHABLE(); // Requested frame not found.
|
|
|
|
return Handle<JSObject>();
|
2014-04-24 14:23:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-04-28 12:02:11 +00:00
|
|
|
void Accessors::FunctionArgumentsGetter(
|
2014-08-20 15:25:13 +00:00
|
|
|
v8::Local<v8::Name> name,
|
2014-04-28 12:02:11 +00:00
|
|
|
const v8::PropertyCallbackInfo<v8::Value>& info) {
|
|
|
|
i::Isolate* isolate = reinterpret_cast<i::Isolate*>(info.GetIsolate());
|
2014-04-24 14:23:15 +00:00
|
|
|
HandleScope scope(isolate);
|
2014-07-23 20:11:33 +00:00
|
|
|
Handle<JSFunction> function =
|
|
|
|
Handle<JSFunction>::cast(Utils::OpenHandle(*info.Holder()));
|
2017-11-29 13:18:33 +00:00
|
|
|
Handle<Object> result = isolate->factory()->null_value();
|
|
|
|
if (!function->shared()->native()) {
|
|
|
|
// Find the top invocation of the function by traversing frames.
|
|
|
|
for (JavaScriptFrameIterator it(isolate); !it.done(); it.Advance()) {
|
|
|
|
JavaScriptFrame* frame = it.frame();
|
|
|
|
int function_index = FindFunctionInFrame(frame, function);
|
|
|
|
if (function_index >= 0) {
|
|
|
|
result = GetFrameArguments(isolate, &it, function_index);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2014-04-28 12:02:11 +00:00
|
|
|
info.GetReturnValue().Set(Utils::ToLocal(result));
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
2017-10-27 09:06:44 +00:00
|
|
|
Handle<AccessorInfo> Accessors::MakeFunctionArgumentsInfo(Isolate* isolate) {
|
2016-01-27 13:21:53 +00:00
|
|
|
return MakeAccessor(isolate, isolate->factory()->arguments_string(),
|
2017-10-26 15:18:09 +00:00
|
|
|
&FunctionArgumentsGetter, nullptr);
|
2014-04-28 12:02:11 +00:00
|
|
|
}
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
|
|
|
|
//
|
|
|
|
// Accessors::FunctionCaller
|
|
|
|
//
|
|
|
|
|
|
|
|
|
2014-05-19 13:45:45 +00:00
|
|
|
static inline bool AllowAccessToFunction(Context* current_context,
|
|
|
|
JSFunction* function) {
|
|
|
|
return current_context->HasSameSecurityTokenAs(function->context());
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-08-25 13:38:58 +00:00
|
|
|
class FrameFunctionIterator {
|
|
|
|
public:
|
2017-08-22 13:38:31 +00:00
|
|
|
explicit FrameFunctionIterator(Isolate* isolate)
|
2017-09-27 11:47:10 +00:00
|
|
|
: isolate_(isolate), frame_iterator_(isolate), inlined_frame_index_(-1) {
|
2017-01-13 10:29:13 +00:00
|
|
|
GetFrames();
|
2011-08-25 13:38:58 +00:00
|
|
|
}
|
2017-09-27 10:02:27 +00:00
|
|
|
|
|
|
|
// Iterate through functions until the first occurrence of 'function'.
|
2017-09-27 11:47:10 +00:00
|
|
|
// Returns true if one is found, and false if the iterator ends before.
|
2017-09-27 10:02:27 +00:00
|
|
|
bool Find(Handle<JSFunction> function) {
|
|
|
|
do {
|
2017-09-27 11:47:10 +00:00
|
|
|
if (!next().ToHandle(&function_)) return false;
|
|
|
|
} while (!function_.is_identical_to(function));
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Iterate through functions until the next non-toplevel one is found.
|
|
|
|
// Returns true if one is found, and false if the iterator ends before.
|
|
|
|
bool FindNextNonTopLevel() {
|
|
|
|
do {
|
|
|
|
if (!next().ToHandle(&function_)) return false;
|
|
|
|
} while (function_->shared()->is_toplevel());
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Iterate through function until the first native or user-provided function
|
|
|
|
// is found. Functions not defined in user-provided scripts are not visible
|
|
|
|
// unless directly exposed, in which case the native flag is set on them.
|
|
|
|
// Returns true if one is found, and false if the iterator ends before.
|
|
|
|
bool FindFirstNativeOrUserJavaScript() {
|
|
|
|
while (!function_->shared()->native() &&
|
|
|
|
!function_->shared()->IsUserJavaScript()) {
|
|
|
|
if (!next().ToHandle(&function_)) return false;
|
|
|
|
}
|
2017-09-27 10:02:27 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2017-09-27 11:47:10 +00:00
|
|
|
// In case of inlined frames the function could have been materialized from
|
|
|
|
// deoptimization information. If that is the case we need to make sure that
|
|
|
|
// subsequent call will see the same function, since we are about to hand out
|
|
|
|
// the value to JavaScript. Make sure to store the materialized value and
|
|
|
|
// trigger a deoptimization of the underlying frame.
|
|
|
|
Handle<JSFunction> MaterializeFunction() {
|
|
|
|
if (inlined_frame_index_ == 0) return function_;
|
|
|
|
|
|
|
|
JavaScriptFrame* frame = frame_iterator_.frame();
|
|
|
|
TranslatedState translated_values(frame);
|
|
|
|
translated_values.Prepare(frame->fp());
|
|
|
|
|
|
|
|
TranslatedFrame* translated_frame =
|
|
|
|
translated_values.GetFrameFromJSFrameIndex(inlined_frame_index_);
|
|
|
|
TranslatedFrame::iterator iter = translated_frame->begin();
|
|
|
|
|
|
|
|
// First value is the function.
|
|
|
|
bool should_deoptimize = iter->IsMaterializedObject();
|
|
|
|
Handle<Object> value = iter->GetValue();
|
|
|
|
if (should_deoptimize) {
|
|
|
|
translated_values.StoreMaterializedValuesAndDeopt(frame);
|
|
|
|
}
|
|
|
|
|
|
|
|
return Handle<JSFunction>::cast(value);
|
|
|
|
}
|
|
|
|
|
2017-09-27 10:02:27 +00:00
|
|
|
private:
|
2017-09-27 11:47:10 +00:00
|
|
|
MaybeHandle<JSFunction> next() {
|
|
|
|
while (true) {
|
2017-12-18 21:13:46 +00:00
|
|
|
if (inlined_frame_index_ <= 0) {
|
2017-09-27 11:47:10 +00:00
|
|
|
if (!frame_iterator_.done()) {
|
|
|
|
frame_iterator_.Advance();
|
|
|
|
frames_.clear();
|
2017-12-18 21:13:46 +00:00
|
|
|
inlined_frame_index_ = -1;
|
2017-09-27 11:47:10 +00:00
|
|
|
GetFrames();
|
|
|
|
}
|
|
|
|
if (inlined_frame_index_ == -1) return MaybeHandle<JSFunction>();
|
|
|
|
}
|
2017-12-18 21:13:46 +00:00
|
|
|
|
|
|
|
--inlined_frame_index_;
|
2017-09-27 11:47:10 +00:00
|
|
|
Handle<JSFunction> next_function =
|
|
|
|
frames_[inlined_frame_index_].AsJavaScript().function();
|
|
|
|
// Skip functions from other origins.
|
|
|
|
if (!AllowAccessToFunction(isolate_->context(), *next_function)) continue;
|
|
|
|
return next_function;
|
|
|
|
}
|
|
|
|
}
|
2017-01-13 10:29:13 +00:00
|
|
|
void GetFrames() {
|
2017-09-27 11:47:10 +00:00
|
|
|
DCHECK_EQ(-1, inlined_frame_index_);
|
2011-08-25 13:38:58 +00:00
|
|
|
if (frame_iterator_.done()) return;
|
|
|
|
JavaScriptFrame* frame = frame_iterator_.frame();
|
2017-01-13 10:29:13 +00:00
|
|
|
frame->Summarize(&frames_);
|
2017-09-27 11:47:10 +00:00
|
|
|
inlined_frame_index_ = static_cast<int>(frames_.size());
|
|
|
|
DCHECK_LT(0, inlined_frame_index_);
|
2011-08-25 13:38:58 +00:00
|
|
|
}
|
2014-05-19 13:45:45 +00:00
|
|
|
Isolate* isolate_;
|
2017-09-27 11:47:10 +00:00
|
|
|
Handle<JSFunction> function_;
|
2011-08-25 13:38:58 +00:00
|
|
|
JavaScriptFrameIterator frame_iterator_;
|
2017-08-28 12:10:52 +00:00
|
|
|
std::vector<FrameSummary> frames_;
|
2017-09-27 11:47:10 +00:00
|
|
|
int inlined_frame_index_;
|
2011-08-25 13:38:58 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
|
2014-04-28 13:41:12 +00:00
|
|
|
MaybeHandle<JSFunction> FindCaller(Isolate* isolate,
|
|
|
|
Handle<JSFunction> function) {
|
2017-08-22 13:38:31 +00:00
|
|
|
FrameFunctionIterator it(isolate);
|
2014-04-28 13:41:12 +00:00
|
|
|
if (function->shared()->native()) {
|
|
|
|
return MaybeHandle<JSFunction>();
|
|
|
|
}
|
2017-09-27 11:47:10 +00:00
|
|
|
// Find the function from the frames. Return null in case no frame
|
|
|
|
// corresponding to the given function was found.
|
2017-08-22 13:38:31 +00:00
|
|
|
if (!it.Find(function)) {
|
2014-04-28 13:41:12 +00:00
|
|
|
return MaybeHandle<JSFunction>();
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
2011-08-25 13:38:58 +00:00
|
|
|
// Find previously called non-toplevel function.
|
2017-09-27 11:47:10 +00:00
|
|
|
if (!it.FindNextNonTopLevel()) {
|
|
|
|
return MaybeHandle<JSFunction>();
|
2011-08-25 13:38:58 +00:00
|
|
|
}
|
2017-09-27 11:47:10 +00:00
|
|
|
// Find the first user-land JavaScript function (or the entry point into
|
|
|
|
// native JavaScript builtins in case such a builtin was the caller).
|
|
|
|
if (!it.FindFirstNativeOrUserJavaScript()) {
|
|
|
|
return MaybeHandle<JSFunction>();
|
2012-09-05 08:19:49 +00:00
|
|
|
}
|
2017-09-27 11:47:10 +00:00
|
|
|
|
|
|
|
// Materialize the function that the iterator is currently sitting on. Note
|
|
|
|
// that this might trigger deoptimization in case the function was actually
|
|
|
|
// materialized. Identity of the function must be preserved because we are
|
|
|
|
// going to return it to JavaScript after this point.
|
|
|
|
Handle<JSFunction> caller = it.MaterializeFunction();
|
|
|
|
|
2014-03-11 14:39:08 +00:00
|
|
|
// Censor if the caller is not a sloppy mode function.
|
2012-11-23 15:47:58 +00:00
|
|
|
// Change from ES5, which used to throw, see:
|
|
|
|
// https://bugs.ecmascript.org/show_bug.cgi?id=310
|
2015-02-04 09:34:05 +00:00
|
|
|
if (is_strict(caller->shared()->language_mode())) {
|
2014-04-28 13:41:12 +00:00
|
|
|
return MaybeHandle<JSFunction>();
|
|
|
|
}
|
2014-05-19 13:45:45 +00:00
|
|
|
// Don't return caller from another security context.
|
2017-08-22 13:38:31 +00:00
|
|
|
if (!AllowAccessToFunction(isolate->context(), *caller)) {
|
2014-05-19 13:45:45 +00:00
|
|
|
return MaybeHandle<JSFunction>();
|
|
|
|
}
|
2017-08-22 13:38:31 +00:00
|
|
|
return caller;
|
2014-04-28 13:41:12 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void Accessors::FunctionCallerGetter(
|
2014-08-20 15:25:13 +00:00
|
|
|
v8::Local<v8::Name> name,
|
2014-04-28 13:41:12 +00:00
|
|
|
const v8::PropertyCallbackInfo<v8::Value>& info) {
|
|
|
|
i::Isolate* isolate = reinterpret_cast<i::Isolate*>(info.GetIsolate());
|
|
|
|
HandleScope scope(isolate);
|
2014-07-23 20:11:33 +00:00
|
|
|
Handle<JSFunction> function =
|
|
|
|
Handle<JSFunction>::cast(Utils::OpenHandle(*info.Holder()));
|
2014-04-28 13:41:12 +00:00
|
|
|
Handle<Object> result;
|
2014-07-23 20:11:33 +00:00
|
|
|
MaybeHandle<JSFunction> maybe_caller;
|
|
|
|
maybe_caller = FindCaller(isolate, function);
|
|
|
|
Handle<JSFunction> caller;
|
|
|
|
if (maybe_caller.ToHandle(&caller)) {
|
|
|
|
result = caller;
|
2014-04-28 13:41:12 +00:00
|
|
|
} else {
|
2014-07-23 20:11:33 +00:00
|
|
|
result = isolate->factory()->null_value();
|
2012-11-23 15:47:58 +00:00
|
|
|
}
|
2014-04-28 13:41:12 +00:00
|
|
|
info.GetReturnValue().Set(Utils::ToLocal(result));
|
|
|
|
}
|
|
|
|
|
2017-10-27 09:06:44 +00:00
|
|
|
Handle<AccessorInfo> Accessors::MakeFunctionCallerInfo(Isolate* isolate) {
|
2016-01-27 13:21:53 +00:00
|
|
|
return MakeAccessor(isolate, isolate->factory()->caller_string(),
|
2017-10-26 15:18:09 +00:00
|
|
|
&FunctionCallerGetter, nullptr);
|
2014-04-28 13:41:12 +00:00
|
|
|
}
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
|
2016-04-07 13:36:40 +00:00
|
|
|
//
|
|
|
|
// Accessors::BoundFunctionLength
|
|
|
|
//
|
|
|
|
|
|
|
|
void Accessors::BoundFunctionLengthGetter(
|
|
|
|
v8::Local<v8::Name> name, const v8::PropertyCallbackInfo<v8::Value>& info) {
|
|
|
|
i::Isolate* isolate = reinterpret_cast<i::Isolate*>(info.GetIsolate());
|
2017-07-26 22:34:21 +00:00
|
|
|
RuntimeCallTimerScope timer(isolate,
|
2017-11-29 15:50:22 +00:00
|
|
|
RuntimeCallCounterId::kBoundFunctionLengthGetter);
|
2016-04-07 13:36:40 +00:00
|
|
|
HandleScope scope(isolate);
|
|
|
|
Handle<JSBoundFunction> function =
|
|
|
|
Handle<JSBoundFunction>::cast(Utils::OpenHandle(*info.Holder()));
|
|
|
|
|
2017-10-04 11:38:34 +00:00
|
|
|
int length = 0;
|
|
|
|
if (!JSBoundFunction::GetLength(isolate, function).To(&length)) {
|
2016-04-07 13:36:40 +00:00
|
|
|
isolate->OptionalRescheduleException(false);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
Handle<Object> result(Smi::FromInt(length), isolate);
|
|
|
|
info.GetReturnValue().Set(Utils::ToLocal(result));
|
|
|
|
}
|
|
|
|
|
2017-10-27 09:06:44 +00:00
|
|
|
Handle<AccessorInfo> Accessors::MakeBoundFunctionLengthInfo(Isolate* isolate) {
|
2016-04-07 13:36:40 +00:00
|
|
|
return MakeAccessor(isolate, isolate->factory()->length_string(),
|
2017-10-26 15:18:09 +00:00
|
|
|
&BoundFunctionLengthGetter, &ReconfigureToDataProperty);
|
2016-04-07 13:36:40 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
//
|
|
|
|
// Accessors::BoundFunctionName
|
|
|
|
//
|
|
|
|
|
|
|
|
void Accessors::BoundFunctionNameGetter(
|
|
|
|
v8::Local<v8::Name> name, const v8::PropertyCallbackInfo<v8::Value>& info) {
|
|
|
|
i::Isolate* isolate = reinterpret_cast<i::Isolate*>(info.GetIsolate());
|
2017-07-26 22:34:21 +00:00
|
|
|
RuntimeCallTimerScope timer(isolate,
|
2017-11-29 15:50:22 +00:00
|
|
|
RuntimeCallCounterId::kBoundFunctionNameGetter);
|
2016-04-07 13:36:40 +00:00
|
|
|
HandleScope scope(isolate);
|
|
|
|
Handle<JSBoundFunction> function =
|
|
|
|
Handle<JSBoundFunction>::cast(Utils::OpenHandle(*info.Holder()));
|
|
|
|
Handle<Object> result;
|
|
|
|
if (!JSBoundFunction::GetName(isolate, function).ToHandle(&result)) {
|
|
|
|
isolate->OptionalRescheduleException(false);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
info.GetReturnValue().Set(Utils::ToLocal(result));
|
|
|
|
}
|
|
|
|
|
2017-10-27 09:06:44 +00:00
|
|
|
Handle<AccessorInfo> Accessors::MakeBoundFunctionNameInfo(Isolate* isolate) {
|
2016-04-07 13:36:40 +00:00
|
|
|
return MakeAccessor(isolate, isolate->factory()->name_string(),
|
2017-10-26 15:18:09 +00:00
|
|
|
&BoundFunctionNameGetter, &ReconfigureToDataProperty);
|
2016-04-07 13:36:40 +00:00
|
|
|
}
|
|
|
|
|
2016-07-20 13:02:36 +00:00
|
|
|
//
|
|
|
|
// Accessors::ErrorStack
|
|
|
|
//
|
|
|
|
|
|
|
|
namespace {
|
|
|
|
|
|
|
|
MaybeHandle<JSReceiver> ClearInternalStackTrace(Isolate* isolate,
|
|
|
|
Handle<JSObject> error) {
|
|
|
|
RETURN_ON_EXCEPTION(
|
|
|
|
isolate,
|
2018-07-17 08:49:20 +00:00
|
|
|
JSReceiver::SetProperty(
|
|
|
|
isolate, error, isolate->factory()->stack_trace_symbol(),
|
|
|
|
isolate->factory()->undefined_value(), LanguageMode::kStrict),
|
2016-07-20 13:02:36 +00:00
|
|
|
JSReceiver);
|
|
|
|
return error;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool IsAccessor(Handle<Object> receiver, Handle<Name> name,
|
|
|
|
Handle<JSObject> holder) {
|
|
|
|
LookupIterator it(receiver, name, holder,
|
|
|
|
LookupIterator::OWN_SKIP_INTERCEPTOR);
|
|
|
|
// Skip any access checks we might hit. This accessor should never hit in a
|
|
|
|
// situation where the caller does not have access.
|
|
|
|
if (it.state() == LookupIterator::ACCESS_CHECK) {
|
|
|
|
CHECK(it.HasAccess());
|
|
|
|
it.Next();
|
|
|
|
}
|
|
|
|
return (it.state() == LookupIterator::ACCESSOR);
|
|
|
|
}
|
|
|
|
|
|
|
|
} // namespace
|
|
|
|
|
|
|
|
void Accessors::ErrorStackGetter(
|
|
|
|
v8::Local<v8::Name> key, const v8::PropertyCallbackInfo<v8::Value>& info) {
|
|
|
|
i::Isolate* isolate = reinterpret_cast<i::Isolate*>(info.GetIsolate());
|
|
|
|
HandleScope scope(isolate);
|
|
|
|
Handle<JSObject> holder =
|
|
|
|
Handle<JSObject>::cast(Utils::OpenHandle(*info.Holder()));
|
|
|
|
|
|
|
|
// Retrieve the structured stack trace.
|
|
|
|
|
|
|
|
Handle<Object> stack_trace;
|
|
|
|
Handle<Symbol> stack_trace_symbol = isolate->factory()->stack_trace_symbol();
|
|
|
|
MaybeHandle<Object> maybe_stack_trace =
|
2018-06-19 09:00:37 +00:00
|
|
|
JSObject::GetProperty(isolate, holder, stack_trace_symbol);
|
2016-07-20 13:02:36 +00:00
|
|
|
if (!maybe_stack_trace.ToHandle(&stack_trace) ||
|
|
|
|
stack_trace->IsUndefined(isolate)) {
|
|
|
|
Handle<Object> result = isolate->factory()->undefined_value();
|
|
|
|
info.GetReturnValue().Set(Utils::ToLocal(result));
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Format it, clear the internal structured trace and reconfigure as a data
|
|
|
|
// property.
|
|
|
|
|
|
|
|
Handle<Object> formatted_stack_trace;
|
2016-08-03 13:48:43 +00:00
|
|
|
if (!ErrorUtils::FormatStackTrace(isolate, holder, stack_trace)
|
2016-07-20 13:02:36 +00:00
|
|
|
.ToHandle(&formatted_stack_trace)) {
|
|
|
|
isolate->OptionalRescheduleException(false);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
MaybeHandle<Object> result = ClearInternalStackTrace(isolate, holder);
|
|
|
|
if (result.is_null()) {
|
|
|
|
isolate->OptionalRescheduleException(false);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// If stack is still an accessor (this could have changed in the meantime
|
|
|
|
// since FormatStackTrace can execute arbitrary JS), replace it with a data
|
|
|
|
// property.
|
2017-02-20 11:48:10 +00:00
|
|
|
Handle<Object> receiver =
|
|
|
|
Utils::OpenHandle(*v8::Local<v8::Value>(info.This()));
|
2016-07-20 13:02:36 +00:00
|
|
|
Handle<Name> name = Utils::OpenHandle(*key);
|
|
|
|
if (IsAccessor(receiver, name, holder)) {
|
2018-07-18 16:05:36 +00:00
|
|
|
result = Accessors::ReplaceAccessorWithDataProperty(receiver, holder, name,
|
|
|
|
formatted_stack_trace);
|
2016-07-20 13:02:36 +00:00
|
|
|
if (result.is_null()) {
|
|
|
|
isolate->OptionalRescheduleException(false);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
// The stack property has been modified in the meantime.
|
2018-06-19 09:00:37 +00:00
|
|
|
if (!JSObject::GetProperty(isolate, holder, name)
|
|
|
|
.ToHandle(&formatted_stack_trace)) {
|
2016-07-20 13:02:36 +00:00
|
|
|
isolate->OptionalRescheduleException(false);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
v8::Local<v8::Value> value = Utils::ToLocal(formatted_stack_trace);
|
|
|
|
info.GetReturnValue().Set(value);
|
|
|
|
}
|
|
|
|
|
2016-10-26 08:59:24 +00:00
|
|
|
void Accessors::ErrorStackSetter(
|
|
|
|
v8::Local<v8::Name> name, v8::Local<v8::Value> val,
|
|
|
|
const v8::PropertyCallbackInfo<v8::Boolean>& info) {
|
2016-07-20 13:02:36 +00:00
|
|
|
i::Isolate* isolate = reinterpret_cast<i::Isolate*>(info.GetIsolate());
|
|
|
|
HandleScope scope(isolate);
|
2017-02-20 11:48:10 +00:00
|
|
|
Handle<JSObject> obj = Handle<JSObject>::cast(
|
|
|
|
Utils::OpenHandle(*v8::Local<v8::Value>(info.This())));
|
2016-07-20 13:02:36 +00:00
|
|
|
|
|
|
|
// Clear internal properties to avoid memory leaks.
|
|
|
|
Handle<Symbol> stack_trace_symbol = isolate->factory()->stack_trace_symbol();
|
|
|
|
if (JSReceiver::HasOwnProperty(obj, stack_trace_symbol).FromMaybe(false)) {
|
|
|
|
ClearInternalStackTrace(isolate, obj);
|
|
|
|
}
|
|
|
|
|
|
|
|
Accessors::ReconfigureToDataProperty(name, val, info);
|
|
|
|
}
|
|
|
|
|
2017-10-27 09:06:44 +00:00
|
|
|
Handle<AccessorInfo> Accessors::MakeErrorStackInfo(Isolate* isolate) {
|
2017-10-26 15:18:09 +00:00
|
|
|
return MakeAccessor(isolate, isolate->factory()->stack_string(),
|
|
|
|
&ErrorStackGetter, &ErrorStackSetter);
|
2016-07-20 13:02:36 +00:00
|
|
|
}
|
2012-07-09 08:59:03 +00:00
|
|
|
|
2015-06-01 22:46:54 +00:00
|
|
|
} // namespace internal
|
|
|
|
} // namespace v8
|