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-03 08:12:43 +00:00
|
|
|
#include "src/v8.h"
|
2012-01-25 00:16:07 +00:00
|
|
|
|
2014-06-20 08:40:11 +00:00
|
|
|
#include "src/accessors.h"
|
|
|
|
#include "src/api.h"
|
2014-06-03 08:12:43 +00:00
|
|
|
#include "src/compiler.h"
|
|
|
|
#include "src/contexts.h"
|
|
|
|
#include "src/deoptimizer.h"
|
|
|
|
#include "src/execution.h"
|
|
|
|
#include "src/factory.h"
|
|
|
|
#include "src/frames-inl.h"
|
|
|
|
#include "src/isolate.h"
|
|
|
|
#include "src/list-inl.h"
|
|
|
|
#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(
|
|
|
|
Isolate* isolate,
|
|
|
|
Handle<String> name,
|
|
|
|
AccessorGetterCallback getter,
|
|
|
|
AccessorSetterCallback setter,
|
|
|
|
PropertyAttributes attributes) {
|
2014-04-15 13:25:17 +00:00
|
|
|
Factory* factory = isolate->factory();
|
|
|
|
Handle<ExecutableAccessorInfo> info = factory->NewExecutableAccessorInfo();
|
|
|
|
info->set_property_attributes(attributes);
|
2014-04-28 13:41:12 +00:00
|
|
|
info->set_all_can_read(false);
|
|
|
|
info->set_all_can_write(false);
|
2014-04-16 11:57:23 +00:00
|
|
|
info->set_name(*name);
|
2014-04-15 13:25:17 +00:00
|
|
|
Handle<Object> get = v8::FromCData(isolate, getter);
|
|
|
|
Handle<Object> set = v8::FromCData(isolate, setter);
|
|
|
|
info->set_getter(*get);
|
|
|
|
info->set_setter(*set);
|
|
|
|
return info;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-05-28 09:58:27 +00:00
|
|
|
Handle<ExecutableAccessorInfo> Accessors::CloneAccessor(
|
|
|
|
Isolate* isolate,
|
|
|
|
Handle<ExecutableAccessorInfo> accessor) {
|
|
|
|
Factory* factory = isolate->factory();
|
|
|
|
Handle<ExecutableAccessorInfo> info = factory->NewExecutableAccessorInfo();
|
|
|
|
info->set_name(accessor->name());
|
|
|
|
info->set_flag(accessor->flag());
|
|
|
|
info->set_expected_receiver_type(accessor->expected_receiver_type());
|
|
|
|
info->set_getter(accessor->getter());
|
|
|
|
info->set_setter(accessor->setter());
|
|
|
|
info->set_data(accessor->data());
|
|
|
|
return info;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
template <class C>
|
2013-02-27 13:22:29 +00:00
|
|
|
static C* FindInstanceOf(Isolate* isolate, Object* obj) {
|
2014-07-14 07:19:49 +00:00
|
|
|
for (PrototypeIterator iter(isolate, obj,
|
|
|
|
PrototypeIterator::START_AT_RECEIVER);
|
|
|
|
!iter.IsAtEnd(); iter.Advance()) {
|
|
|
|
if (Is<C>(iter.GetCurrent())) return C::cast(iter.GetCurrent());
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
2012-10-08 12:58:46 +00:00
|
|
|
return NULL;
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-09-25 08:19:35 +00:00
|
|
|
static V8_INLINE bool CheckForName(Handle<String> name,
|
2014-04-11 07:27:25 +00:00
|
|
|
Handle<String> property_name,
|
2013-09-25 08:19:35 +00:00
|
|
|
int offset,
|
|
|
|
int* object_offset) {
|
2014-04-11 07:27:25 +00:00
|
|
|
if (String::Equals(name, property_name)) {
|
2013-09-25 08:19:35 +00:00
|
|
|
*object_offset = offset;
|
|
|
|
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.
|
|
|
|
template <class T>
|
|
|
|
bool Accessors::IsJSObjectFieldAccessor(typename T::TypeHandle type,
|
2014-01-30 11:30:38 +00:00
|
|
|
Handle<String> name,
|
|
|
|
int* object_offset) {
|
|
|
|
Isolate* isolate = name->GetIsolate();
|
|
|
|
|
2014-02-04 12:44:15 +00:00
|
|
|
if (type->Is(T::String())) {
|
2014-04-11 07:27:25 +00:00
|
|
|
return CheckForName(name, isolate->factory()->length_string(),
|
2014-01-30 11:30:38 +00:00
|
|
|
String::kLengthOffset, object_offset);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!type->IsClass()) return false;
|
2014-04-16 16:16:37 +00:00
|
|
|
Handle<Map> map = type->AsClass()->Map();
|
2014-01-30 11:30:38 +00:00
|
|
|
|
2013-09-25 08:19:35 +00:00
|
|
|
switch (map->instance_type()) {
|
|
|
|
case JS_ARRAY_TYPE:
|
|
|
|
return
|
2014-04-11 07:27:25 +00:00
|
|
|
CheckForName(name, isolate->factory()->length_string(),
|
2013-09-25 08:19:35 +00:00
|
|
|
JSArray::kLengthOffset, object_offset);
|
|
|
|
case JS_TYPED_ARRAY_TYPE:
|
|
|
|
return
|
2014-04-11 07:27:25 +00:00
|
|
|
CheckForName(name, isolate->factory()->length_string(),
|
2013-09-25 08:19:35 +00:00
|
|
|
JSTypedArray::kLengthOffset, object_offset) ||
|
2014-04-11 07:27:25 +00:00
|
|
|
CheckForName(name, isolate->factory()->byte_length_string(),
|
2013-09-25 08:19:35 +00:00
|
|
|
JSTypedArray::kByteLengthOffset, object_offset) ||
|
2014-04-11 07:27:25 +00:00
|
|
|
CheckForName(name, isolate->factory()->byte_offset_string(),
|
2014-03-26 12:50:13 +00:00
|
|
|
JSTypedArray::kByteOffsetOffset, object_offset);
|
2013-09-25 08:19:35 +00:00
|
|
|
case JS_ARRAY_BUFFER_TYPE:
|
|
|
|
return
|
2014-04-11 07:27:25 +00:00
|
|
|
CheckForName(name, isolate->factory()->byte_length_string(),
|
2013-09-25 08:19:35 +00:00
|
|
|
JSArrayBuffer::kByteLengthOffset, object_offset);
|
|
|
|
case JS_DATA_VIEW_TYPE:
|
|
|
|
return
|
2014-04-11 07:27:25 +00:00
|
|
|
CheckForName(name, isolate->factory()->byte_length_string(),
|
2013-09-25 08:19:35 +00:00
|
|
|
JSDataView::kByteLengthOffset, object_offset) ||
|
2014-04-11 07:27:25 +00:00
|
|
|
CheckForName(name, isolate->factory()->byte_offset_string(),
|
2014-03-26 12:50:13 +00:00
|
|
|
JSDataView::kByteOffsetOffset, object_offset);
|
2014-01-30 11:30:38 +00:00
|
|
|
default:
|
2013-09-25 08:19:35 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-02-04 12:44:15 +00:00
|
|
|
template
|
|
|
|
bool Accessors::IsJSObjectFieldAccessor<Type>(Type* type,
|
|
|
|
Handle<String> name,
|
|
|
|
int* object_offset);
|
|
|
|
|
|
|
|
|
|
|
|
template
|
|
|
|
bool Accessors::IsJSObjectFieldAccessor<HeapType>(Handle<HeapType> type,
|
|
|
|
Handle<String> name,
|
|
|
|
int* object_offset);
|
|
|
|
|
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
//
|
|
|
|
// Accessors::ArrayLength
|
|
|
|
//
|
|
|
|
|
|
|
|
|
|
|
|
// The helper function will 'flatten' Number objects.
|
2013-11-05 13:47:51 +00:00
|
|
|
Handle<Object> Accessors::FlattenNumber(Isolate* isolate,
|
|
|
|
Handle<Object> value) {
|
2008-07-03 15:10:15 +00:00
|
|
|
if (value->IsNumber() || !value->IsJSValue()) return value;
|
2013-11-05 13:47:51 +00:00
|
|
|
Handle<JSValue> wrapper = Handle<JSValue>::cast(value);
|
2014-07-01 12:12:34 +00:00
|
|
|
ASSERT(wrapper->GetIsolate()->native_context()->number_function()->
|
2011-03-18 20:35:07 +00:00
|
|
|
has_initial_map());
|
2014-07-01 12:12:34 +00:00
|
|
|
if (wrapper->map() == isolate->number_function()->initial_map()) {
|
2013-11-05 13:47:51 +00:00
|
|
|
return handle(wrapper->value(), isolate);
|
|
|
|
}
|
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
return value;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-04-28 14:59:29 +00:00
|
|
|
void Accessors::ArrayLengthGetter(
|
|
|
|
v8::Local<v8::String> name,
|
|
|
|
const v8::PropertyCallbackInfo<v8::Value>& info) {
|
|
|
|
i::Isolate* isolate = reinterpret_cast<i::Isolate*>(info.GetIsolate());
|
|
|
|
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(
|
|
|
|
v8::Local<v8::String> name,
|
|
|
|
v8::Local<v8::Value> val,
|
|
|
|
const v8::PropertyCallbackInfo<void>& info) {
|
|
|
|
i::Isolate* isolate = reinterpret_cast<i::Isolate*>(info.GetIsolate());
|
|
|
|
HandleScope scope(isolate);
|
2014-07-23 20:27:32 +00:00
|
|
|
Handle<JSObject> object = Utils::OpenHandle(*info.This());
|
2014-04-28 14:59:29 +00:00
|
|
|
Handle<Object> value = Utils::OpenHandle(*val);
|
2014-07-07 16:18:15 +00:00
|
|
|
// This means one of the object's prototypes is a JSArray and the
|
|
|
|
// object does not have a 'length' property. Calling SetProperty
|
|
|
|
// causes an infinite loop.
|
|
|
|
if (!object->IsJSArray()) {
|
2014-07-23 20:27:32 +00:00
|
|
|
// This behaves sloppy since we lost the actual strict-mode.
|
|
|
|
// TODO(verwaest): Fix by making ExecutableAccessorInfo behave like data
|
|
|
|
// properties.
|
|
|
|
if (!object->map()->is_extensible()) return;
|
2014-07-07 16:18:15 +00:00
|
|
|
MaybeHandle<Object> maybe_result = JSObject::SetOwnPropertyIgnoreAttributes(
|
|
|
|
object, isolate->factory()->length_string(), value, NONE);
|
|
|
|
maybe_result.Check();
|
|
|
|
return;
|
|
|
|
}
|
2011-06-21 08:07:45 +00:00
|
|
|
|
2013-09-02 09:25:20 +00:00
|
|
|
value = FlattenNumber(isolate, value);
|
2008-07-03 15:10:15 +00:00
|
|
|
|
2013-11-05 13:47:51 +00:00
|
|
|
Handle<JSArray> array_handle = Handle<JSArray>::cast(object);
|
2014-04-28 14:59:29 +00:00
|
|
|
MaybeHandle<Object> maybe;
|
2014-04-11 10:41:09 +00:00
|
|
|
Handle<Object> uint32_v;
|
2014-04-28 14:59:29 +00:00
|
|
|
maybe = Execution::ToUint32(isolate, value);
|
|
|
|
if (!maybe.ToHandle(&uint32_v)) {
|
|
|
|
isolate->OptionalRescheduleException(false);
|
|
|
|
return;
|
|
|
|
}
|
2014-04-11 10:41:09 +00:00
|
|
|
Handle<Object> number_v;
|
2014-04-28 14:59:29 +00:00
|
|
|
maybe = Execution::ToNumber(isolate, value);
|
|
|
|
if (!maybe.ToHandle(&number_v)) {
|
|
|
|
isolate->OptionalRescheduleException(false);
|
|
|
|
return;
|
|
|
|
}
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
if (uint32_v->Number() == number_v->Number()) {
|
2014-04-29 12:52:42 +00:00
|
|
|
maybe = JSArray::SetElementsLength(array_handle, uint32_v);
|
|
|
|
maybe.Check();
|
2014-04-28 14:59:29 +00:00
|
|
|
return;
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
2014-04-28 14:59:29 +00:00
|
|
|
|
|
|
|
isolate->ScheduleThrow(
|
2011-03-29 07:34:23 +00:00
|
|
|
*isolate->factory()->NewRangeError("invalid_array_length",
|
|
|
|
HandleVector<Object>(NULL, 0)));
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-04-28 14:59:29 +00:00
|
|
|
Handle<AccessorInfo> Accessors::ArrayLengthInfo(
|
|
|
|
Isolate* isolate, PropertyAttributes attributes) {
|
|
|
|
return MakeAccessor(isolate,
|
|
|
|
isolate->factory()->length_string(),
|
|
|
|
&ArrayLengthGetter,
|
|
|
|
&ArrayLengthSetter,
|
|
|
|
attributes);
|
|
|
|
}
|
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
|
|
|
|
//
|
|
|
|
// Accessors::StringLength
|
|
|
|
//
|
|
|
|
|
2014-04-15 13:25:17 +00:00
|
|
|
void Accessors::StringLengthGetter(
|
|
|
|
v8::Local<v8::String> name,
|
|
|
|
const v8::PropertyCallbackInfo<v8::Value>& info) {
|
|
|
|
i::Isolate* isolate = reinterpret_cast<i::Isolate*>(info.GetIsolate());
|
|
|
|
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
|
|
|
|
2014-04-15 13:25:17 +00:00
|
|
|
|
|
|
|
void Accessors::StringLengthSetter(
|
|
|
|
v8::Local<v8::String> name,
|
|
|
|
v8::Local<v8::Value> value,
|
|
|
|
const v8::PropertyCallbackInfo<void>& info) {
|
|
|
|
UNREACHABLE();
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-04-15 13:25:17 +00:00
|
|
|
Handle<AccessorInfo> Accessors::StringLengthInfo(
|
|
|
|
Isolate* isolate, PropertyAttributes attributes) {
|
|
|
|
return MakeAccessor(isolate,
|
|
|
|
isolate->factory()->length_string(),
|
|
|
|
&StringLengthGetter,
|
|
|
|
&StringLengthSetter,
|
|
|
|
attributes);
|
|
|
|
}
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
|
|
|
|
//
|
2014-04-16 11:57:23 +00:00
|
|
|
// Accessors::ScriptColumnOffset
|
2008-07-03 15:10:15 +00:00
|
|
|
//
|
|
|
|
|
|
|
|
|
2014-04-16 11:57:23 +00:00
|
|
|
void Accessors::ScriptColumnOffsetGetter(
|
|
|
|
v8::Local<v8::String> name,
|
|
|
|
const v8::PropertyCallbackInfo<v8::Value>& info) {
|
|
|
|
i::Isolate* isolate = reinterpret_cast<i::Isolate*>(info.GetIsolate());
|
|
|
|
DisallowHeapAllocation no_allocation;
|
|
|
|
HandleScope scope(isolate);
|
|
|
|
Object* object = *Utils::OpenHandle(*info.This());
|
|
|
|
Object* res = Script::cast(JSValue::cast(object)->value())->column_offset();
|
|
|
|
info.GetReturnValue().Set(Utils::ToLocal(Handle<Object>(res, isolate)));
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-04-16 11:57:23 +00:00
|
|
|
void Accessors::ScriptColumnOffsetSetter(
|
|
|
|
v8::Local<v8::String> name,
|
|
|
|
v8::Local<v8::Value> value,
|
|
|
|
const v8::PropertyCallbackInfo<void>& info) {
|
|
|
|
UNREACHABLE();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Handle<AccessorInfo> Accessors::ScriptColumnOffsetInfo(
|
|
|
|
Isolate* isolate, PropertyAttributes attributes) {
|
|
|
|
Handle<String> name(isolate->factory()->InternalizeOneByteString(
|
|
|
|
STATIC_ASCII_VECTOR("column_offset")));
|
|
|
|
return MakeAccessor(isolate,
|
|
|
|
name,
|
|
|
|
&ScriptColumnOffsetGetter,
|
|
|
|
&ScriptColumnOffsetSetter,
|
|
|
|
attributes);
|
|
|
|
}
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
|
|
|
|
//
|
2014-04-16 11:57:23 +00:00
|
|
|
// Accessors::ScriptId
|
2008-07-03 15:10:15 +00:00
|
|
|
//
|
|
|
|
|
|
|
|
|
2014-04-16 11:57:23 +00:00
|
|
|
void Accessors::ScriptIdGetter(
|
|
|
|
v8::Local<v8::String> name,
|
|
|
|
const v8::PropertyCallbackInfo<v8::Value>& info) {
|
|
|
|
i::Isolate* isolate = reinterpret_cast<i::Isolate*>(info.GetIsolate());
|
|
|
|
DisallowHeapAllocation no_allocation;
|
|
|
|
HandleScope scope(isolate);
|
|
|
|
Object* object = *Utils::OpenHandle(*info.This());
|
|
|
|
Object* id = Script::cast(JSValue::cast(object)->value())->id();
|
|
|
|
info.GetReturnValue().Set(Utils::ToLocal(Handle<Object>(id, isolate)));
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-04-16 11:57:23 +00:00
|
|
|
void Accessors::ScriptIdSetter(
|
|
|
|
v8::Local<v8::String> name,
|
|
|
|
v8::Local<v8::Value> value,
|
|
|
|
const v8::PropertyCallbackInfo<void>& info) {
|
|
|
|
UNREACHABLE();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Handle<AccessorInfo> Accessors::ScriptIdInfo(
|
|
|
|
Isolate* isolate, PropertyAttributes attributes) {
|
|
|
|
Handle<String> name(isolate->factory()->InternalizeOneByteString(
|
|
|
|
STATIC_ASCII_VECTOR("id")));
|
|
|
|
return MakeAccessor(isolate,
|
|
|
|
name,
|
|
|
|
&ScriptIdGetter,
|
|
|
|
&ScriptIdSetter,
|
|
|
|
attributes);
|
|
|
|
}
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
|
2009-03-10 08:10:50 +00:00
|
|
|
//
|
2014-04-16 11:57:23 +00:00
|
|
|
// Accessors::ScriptName
|
2009-03-10 08:10:50 +00:00
|
|
|
//
|
|
|
|
|
|
|
|
|
2014-04-16 11:57:23 +00:00
|
|
|
void Accessors::ScriptNameGetter(
|
|
|
|
v8::Local<v8::String> name,
|
|
|
|
const v8::PropertyCallbackInfo<v8::Value>& info) {
|
|
|
|
i::Isolate* isolate = reinterpret_cast<i::Isolate*>(info.GetIsolate());
|
|
|
|
DisallowHeapAllocation no_allocation;
|
|
|
|
HandleScope scope(isolate);
|
|
|
|
Object* object = *Utils::OpenHandle(*info.This());
|
|
|
|
Object* source = Script::cast(JSValue::cast(object)->value())->name();
|
|
|
|
info.GetReturnValue().Set(Utils::ToLocal(Handle<Object>(source, isolate)));
|
2009-03-10 08:10:50 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-04-16 11:57:23 +00:00
|
|
|
void Accessors::ScriptNameSetter(
|
|
|
|
v8::Local<v8::String> name,
|
|
|
|
v8::Local<v8::Value> value,
|
|
|
|
const v8::PropertyCallbackInfo<void>& info) {
|
|
|
|
UNREACHABLE();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Handle<AccessorInfo> Accessors::ScriptNameInfo(
|
|
|
|
Isolate* isolate, PropertyAttributes attributes) {
|
|
|
|
return MakeAccessor(isolate,
|
|
|
|
isolate->factory()->name_string(),
|
|
|
|
&ScriptNameGetter,
|
|
|
|
&ScriptNameSetter,
|
|
|
|
attributes);
|
|
|
|
}
|
2009-03-10 08:10:50 +00:00
|
|
|
|
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
//
|
2014-04-16 11:57:23 +00:00
|
|
|
// Accessors::ScriptSource
|
2008-07-03 15:10:15 +00:00
|
|
|
//
|
|
|
|
|
|
|
|
|
2014-04-16 11:57:23 +00:00
|
|
|
void Accessors::ScriptSourceGetter(
|
|
|
|
v8::Local<v8::String> name,
|
|
|
|
const v8::PropertyCallbackInfo<v8::Value>& info) {
|
|
|
|
i::Isolate* isolate = reinterpret_cast<i::Isolate*>(info.GetIsolate());
|
|
|
|
DisallowHeapAllocation no_allocation;
|
|
|
|
HandleScope scope(isolate);
|
|
|
|
Object* object = *Utils::OpenHandle(*info.This());
|
|
|
|
Object* source = Script::cast(JSValue::cast(object)->value())->source();
|
|
|
|
info.GetReturnValue().Set(Utils::ToLocal(Handle<Object>(source, isolate)));
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-04-16 11:57:23 +00:00
|
|
|
void Accessors::ScriptSourceSetter(
|
|
|
|
v8::Local<v8::String> name,
|
|
|
|
v8::Local<v8::Value> value,
|
|
|
|
const v8::PropertyCallbackInfo<void>& info) {
|
|
|
|
UNREACHABLE();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Handle<AccessorInfo> Accessors::ScriptSourceInfo(
|
|
|
|
Isolate* isolate, PropertyAttributes attributes) {
|
|
|
|
return MakeAccessor(isolate,
|
|
|
|
isolate->factory()->source_string(),
|
|
|
|
&ScriptSourceGetter,
|
|
|
|
&ScriptSourceSetter,
|
|
|
|
attributes);
|
|
|
|
}
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
|
|
|
|
//
|
2014-04-16 11:57:23 +00:00
|
|
|
// Accessors::ScriptLineOffset
|
2008-07-03 15:10:15 +00:00
|
|
|
//
|
|
|
|
|
|
|
|
|
2014-04-16 11:57:23 +00:00
|
|
|
void Accessors::ScriptLineOffsetGetter(
|
|
|
|
v8::Local<v8::String> name,
|
|
|
|
const v8::PropertyCallbackInfo<v8::Value>& info) {
|
|
|
|
i::Isolate* isolate = reinterpret_cast<i::Isolate*>(info.GetIsolate());
|
|
|
|
DisallowHeapAllocation no_allocation;
|
|
|
|
HandleScope scope(isolate);
|
|
|
|
Object* object = *Utils::OpenHandle(*info.This());
|
|
|
|
Object* res = Script::cast(JSValue::cast(object)->value())->line_offset();
|
|
|
|
info.GetReturnValue().Set(Utils::ToLocal(Handle<Object>(res, isolate)));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void Accessors::ScriptLineOffsetSetter(
|
|
|
|
v8::Local<v8::String> name,
|
|
|
|
v8::Local<v8::Value> value,
|
|
|
|
const v8::PropertyCallbackInfo<void>& info) {
|
|
|
|
UNREACHABLE();
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-04-16 11:57:23 +00:00
|
|
|
Handle<AccessorInfo> Accessors::ScriptLineOffsetInfo(
|
|
|
|
Isolate* isolate, PropertyAttributes attributes) {
|
|
|
|
Handle<String> name(isolate->factory()->InternalizeOneByteString(
|
|
|
|
STATIC_ASCII_VECTOR("line_offset")));
|
|
|
|
return MakeAccessor(isolate,
|
|
|
|
name,
|
|
|
|
&ScriptLineOffsetGetter,
|
|
|
|
&ScriptLineOffsetSetter,
|
|
|
|
attributes);
|
|
|
|
}
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
|
|
|
|
//
|
|
|
|
// Accessors::ScriptType
|
|
|
|
//
|
|
|
|
|
|
|
|
|
2014-04-16 14:30:58 +00:00
|
|
|
void Accessors::ScriptTypeGetter(
|
|
|
|
v8::Local<v8::String> name,
|
|
|
|
const v8::PropertyCallbackInfo<v8::Value>& info) {
|
|
|
|
i::Isolate* isolate = reinterpret_cast<i::Isolate*>(info.GetIsolate());
|
|
|
|
DisallowHeapAllocation no_allocation;
|
|
|
|
HandleScope scope(isolate);
|
|
|
|
Object* object = *Utils::OpenHandle(*info.This());
|
|
|
|
Object* res = Script::cast(JSValue::cast(object)->value())->type();
|
|
|
|
info.GetReturnValue().Set(Utils::ToLocal(Handle<Object>(res, isolate)));
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-04-16 14:30:58 +00:00
|
|
|
void Accessors::ScriptTypeSetter(
|
|
|
|
v8::Local<v8::String> name,
|
|
|
|
v8::Local<v8::Value> value,
|
|
|
|
const v8::PropertyCallbackInfo<void>& info) {
|
|
|
|
UNREACHABLE();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Handle<AccessorInfo> Accessors::ScriptTypeInfo(
|
|
|
|
Isolate* isolate, PropertyAttributes attributes) {
|
|
|
|
Handle<String> name(isolate->factory()->InternalizeOneByteString(
|
|
|
|
STATIC_ASCII_VECTOR("type")));
|
|
|
|
return MakeAccessor(isolate,
|
|
|
|
name,
|
|
|
|
&ScriptTypeGetter,
|
|
|
|
&ScriptTypeSetter,
|
|
|
|
attributes);
|
|
|
|
}
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
|
2009-06-08 10:47:49 +00:00
|
|
|
//
|
|
|
|
// Accessors::ScriptCompilationType
|
|
|
|
//
|
|
|
|
|
|
|
|
|
2014-04-16 14:30:58 +00:00
|
|
|
void Accessors::ScriptCompilationTypeGetter(
|
|
|
|
v8::Local<v8::String> name,
|
|
|
|
const v8::PropertyCallbackInfo<v8::Value>& info) {
|
|
|
|
i::Isolate* isolate = reinterpret_cast<i::Isolate*>(info.GetIsolate());
|
|
|
|
DisallowHeapAllocation no_allocation;
|
|
|
|
HandleScope scope(isolate);
|
|
|
|
Object* object = *Utils::OpenHandle(*info.This());
|
|
|
|
Object* res = Smi::FromInt(
|
|
|
|
Script::cast(JSValue::cast(object)->value())->compilation_type());
|
|
|
|
info.GetReturnValue().Set(Utils::ToLocal(Handle<Object>(res, isolate)));
|
2009-06-08 10:47:49 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-04-16 14:30:58 +00:00
|
|
|
void Accessors::ScriptCompilationTypeSetter(
|
|
|
|
v8::Local<v8::String> name,
|
|
|
|
v8::Local<v8::Value> value,
|
|
|
|
const v8::PropertyCallbackInfo<void>& info) {
|
|
|
|
UNREACHABLE();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Handle<AccessorInfo> Accessors::ScriptCompilationTypeInfo(
|
|
|
|
Isolate* isolate, PropertyAttributes attributes) {
|
|
|
|
Handle<String> name(isolate->factory()->InternalizeOneByteString(
|
|
|
|
STATIC_ASCII_VECTOR("compilation_type")));
|
|
|
|
return MakeAccessor(isolate,
|
|
|
|
name,
|
|
|
|
&ScriptCompilationTypeGetter,
|
|
|
|
&ScriptCompilationTypeSetter,
|
|
|
|
attributes);
|
|
|
|
}
|
2009-06-08 10:47:49 +00:00
|
|
|
|
|
|
|
|
2009-02-04 12:07:45 +00:00
|
|
|
//
|
|
|
|
// Accessors::ScriptGetLineEnds
|
|
|
|
//
|
|
|
|
|
|
|
|
|
2014-04-16 14:30:58 +00:00
|
|
|
void Accessors::ScriptLineEndsGetter(
|
|
|
|
v8::Local<v8::String> name,
|
|
|
|
const v8::PropertyCallbackInfo<v8::Value>& info) {
|
|
|
|
i::Isolate* isolate = reinterpret_cast<i::Isolate*>(info.GetIsolate());
|
2011-03-29 07:34:23 +00:00
|
|
|
HandleScope scope(isolate);
|
2014-04-16 14:30:58 +00:00
|
|
|
Handle<Object> object = Utils::OpenHandle(*info.This());
|
|
|
|
Handle<Script> script(
|
|
|
|
Script::cast(Handle<JSValue>::cast(object)->value()), isolate);
|
2014-04-16 13:28:11 +00:00
|
|
|
Script::InitLineEnds(script);
|
2009-11-27 14:10:48 +00:00
|
|
|
ASSERT(script->line_ends()->IsFixedArray());
|
|
|
|
Handle<FixedArray> line_ends(FixedArray::cast(script->line_ends()));
|
2010-11-17 12:49:27 +00:00
|
|
|
// We do not want anyone to modify this array from JS.
|
2011-03-29 07:34:23 +00:00
|
|
|
ASSERT(*line_ends == isolate->heap()->empty_fixed_array() ||
|
|
|
|
line_ends->map() == isolate->heap()->fixed_cow_array_map());
|
|
|
|
Handle<JSArray> js_array =
|
|
|
|
isolate->factory()->NewJSArrayWithElements(line_ends);
|
2014-04-16 14:30:58 +00:00
|
|
|
info.GetReturnValue().Set(Utils::ToLocal(js_array));
|
2009-02-04 12:07:45 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-04-16 14:30:58 +00:00
|
|
|
void Accessors::ScriptLineEndsSetter(
|
|
|
|
v8::Local<v8::String> name,
|
|
|
|
v8::Local<v8::Value> value,
|
|
|
|
const v8::PropertyCallbackInfo<void>& info) {
|
|
|
|
UNREACHABLE();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Handle<AccessorInfo> Accessors::ScriptLineEndsInfo(
|
|
|
|
Isolate* isolate, PropertyAttributes attributes) {
|
|
|
|
Handle<String> name(isolate->factory()->InternalizeOneByteString(
|
|
|
|
STATIC_ASCII_VECTOR("line_ends")));
|
|
|
|
return MakeAccessor(isolate,
|
|
|
|
name,
|
|
|
|
&ScriptLineEndsGetter,
|
|
|
|
&ScriptLineEndsSetter,
|
|
|
|
attributes);
|
|
|
|
}
|
2009-02-04 12:07:45 +00:00
|
|
|
|
|
|
|
|
2014-07-02 07:01:31 +00:00
|
|
|
//
|
|
|
|
// Accessors::ScriptSourceUrl
|
|
|
|
//
|
|
|
|
|
|
|
|
|
|
|
|
void Accessors::ScriptSourceUrlGetter(
|
|
|
|
v8::Local<v8::String> name,
|
|
|
|
const v8::PropertyCallbackInfo<v8::Value>& info) {
|
|
|
|
i::Isolate* isolate = reinterpret_cast<i::Isolate*>(info.GetIsolate());
|
|
|
|
DisallowHeapAllocation no_allocation;
|
|
|
|
HandleScope scope(isolate);
|
|
|
|
Object* object = *Utils::OpenHandle(*info.This());
|
|
|
|
Object* url = Script::cast(JSValue::cast(object)->value())->source_url();
|
|
|
|
info.GetReturnValue().Set(Utils::ToLocal(Handle<Object>(url, isolate)));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void Accessors::ScriptSourceUrlSetter(
|
|
|
|
v8::Local<v8::String> name,
|
|
|
|
v8::Local<v8::Value> value,
|
|
|
|
const v8::PropertyCallbackInfo<void>& info) {
|
|
|
|
UNREACHABLE();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Handle<AccessorInfo> Accessors::ScriptSourceUrlInfo(
|
|
|
|
Isolate* isolate, PropertyAttributes attributes) {
|
|
|
|
return MakeAccessor(isolate,
|
|
|
|
isolate->factory()->source_url_string(),
|
|
|
|
&ScriptSourceUrlGetter,
|
|
|
|
&ScriptSourceUrlSetter,
|
|
|
|
attributes);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//
|
|
|
|
// Accessors::ScriptSourceMappingUrl
|
|
|
|
//
|
|
|
|
|
|
|
|
|
|
|
|
void Accessors::ScriptSourceMappingUrlGetter(
|
|
|
|
v8::Local<v8::String> name,
|
|
|
|
const v8::PropertyCallbackInfo<v8::Value>& info) {
|
|
|
|
i::Isolate* isolate = reinterpret_cast<i::Isolate*>(info.GetIsolate());
|
|
|
|
DisallowHeapAllocation no_allocation;
|
|
|
|
HandleScope scope(isolate);
|
|
|
|
Object* object = *Utils::OpenHandle(*info.This());
|
|
|
|
Object* url =
|
|
|
|
Script::cast(JSValue::cast(object)->value())->source_mapping_url();
|
|
|
|
info.GetReturnValue().Set(Utils::ToLocal(Handle<Object>(url, isolate)));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void Accessors::ScriptSourceMappingUrlSetter(
|
|
|
|
v8::Local<v8::String> name,
|
|
|
|
v8::Local<v8::Value> value,
|
|
|
|
const v8::PropertyCallbackInfo<void>& info) {
|
|
|
|
UNREACHABLE();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Handle<AccessorInfo> Accessors::ScriptSourceMappingUrlInfo(
|
|
|
|
Isolate* isolate, PropertyAttributes attributes) {
|
|
|
|
return MakeAccessor(isolate,
|
|
|
|
isolate->factory()->source_mapping_url_string(),
|
|
|
|
&ScriptSourceMappingUrlGetter,
|
|
|
|
&ScriptSourceMappingUrlSetter,
|
|
|
|
attributes);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2009-05-06 08:52:48 +00:00
|
|
|
//
|
|
|
|
// Accessors::ScriptGetContextData
|
|
|
|
//
|
|
|
|
|
|
|
|
|
2014-04-16 14:30:58 +00:00
|
|
|
void Accessors::ScriptContextDataGetter(
|
|
|
|
v8::Local<v8::String> name,
|
|
|
|
const v8::PropertyCallbackInfo<v8::Value>& info) {
|
|
|
|
i::Isolate* isolate = reinterpret_cast<i::Isolate*>(info.GetIsolate());
|
|
|
|
DisallowHeapAllocation no_allocation;
|
|
|
|
HandleScope scope(isolate);
|
|
|
|
Object* object = *Utils::OpenHandle(*info.This());
|
|
|
|
Object* res = Script::cast(JSValue::cast(object)->value())->context_data();
|
|
|
|
info.GetReturnValue().Set(Utils::ToLocal(Handle<Object>(res, isolate)));
|
2009-05-06 08:52:48 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-04-16 14:30:58 +00:00
|
|
|
void Accessors::ScriptContextDataSetter(
|
|
|
|
v8::Local<v8::String> name,
|
|
|
|
v8::Local<v8::Value> value,
|
|
|
|
const v8::PropertyCallbackInfo<void>& info) {
|
|
|
|
UNREACHABLE();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Handle<AccessorInfo> Accessors::ScriptContextDataInfo(
|
|
|
|
Isolate* isolate, PropertyAttributes attributes) {
|
|
|
|
Handle<String> name(isolate->factory()->InternalizeOneByteString(
|
|
|
|
STATIC_ASCII_VECTOR("context_data")));
|
|
|
|
return MakeAccessor(isolate,
|
|
|
|
name,
|
|
|
|
&ScriptContextDataGetter,
|
|
|
|
&ScriptContextDataSetter,
|
|
|
|
attributes);
|
|
|
|
}
|
2009-05-06 08:52:48 +00:00
|
|
|
|
|
|
|
|
2009-06-08 10:47:49 +00:00
|
|
|
//
|
2009-12-01 14:36:45 +00:00
|
|
|
// Accessors::ScriptGetEvalFromScript
|
2009-06-08 10:47:49 +00:00
|
|
|
//
|
|
|
|
|
|
|
|
|
2014-04-16 14:30:58 +00:00
|
|
|
void Accessors::ScriptEvalFromScriptGetter(
|
|
|
|
v8::Local<v8::String> name,
|
|
|
|
const v8::PropertyCallbackInfo<v8::Value>& info) {
|
|
|
|
i::Isolate* isolate = reinterpret_cast<i::Isolate*>(info.GetIsolate());
|
|
|
|
HandleScope scope(isolate);
|
|
|
|
Handle<Object> object = Utils::OpenHandle(*info.This());
|
|
|
|
Handle<Script> script(
|
|
|
|
Script::cast(Handle<JSValue>::cast(object)->value()), isolate);
|
|
|
|
Handle<Object> result = isolate->factory()->undefined_value();
|
|
|
|
if (!script->eval_from_shared()->IsUndefined()) {
|
2009-12-01 14:36:45 +00:00
|
|
|
Handle<SharedFunctionInfo> eval_from_shared(
|
2014-04-16 14:30:58 +00:00
|
|
|
SharedFunctionInfo::cast(script->eval_from_shared()));
|
2009-12-01 14:36:45 +00:00
|
|
|
if (eval_from_shared->script()->IsScript()) {
|
|
|
|
Handle<Script> eval_from_script(Script::cast(eval_from_shared->script()));
|
2014-04-16 14:30:58 +00:00
|
|
|
result = Script::GetWrapper(eval_from_script);
|
2009-12-01 14:36:45 +00:00
|
|
|
}
|
|
|
|
}
|
2014-04-16 14:30:58 +00:00
|
|
|
|
|
|
|
info.GetReturnValue().Set(Utils::ToLocal(result));
|
2009-06-08 10:47:49 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-04-16 14:30:58 +00:00
|
|
|
void Accessors::ScriptEvalFromScriptSetter(
|
|
|
|
v8::Local<v8::String> name,
|
|
|
|
v8::Local<v8::Value> value,
|
|
|
|
const v8::PropertyCallbackInfo<void>& info) {
|
|
|
|
UNREACHABLE();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Handle<AccessorInfo> Accessors::ScriptEvalFromScriptInfo(
|
|
|
|
Isolate* isolate, PropertyAttributes attributes) {
|
|
|
|
Handle<String> name(isolate->factory()->InternalizeOneByteString(
|
|
|
|
STATIC_ASCII_VECTOR("eval_from_script")));
|
|
|
|
return MakeAccessor(isolate,
|
|
|
|
name,
|
|
|
|
&ScriptEvalFromScriptGetter,
|
|
|
|
&ScriptEvalFromScriptSetter,
|
|
|
|
attributes);
|
|
|
|
}
|
2009-06-08 10:47:49 +00:00
|
|
|
|
|
|
|
|
|
|
|
//
|
2009-12-01 14:36:45 +00:00
|
|
|
// Accessors::ScriptGetEvalFromScriptPosition
|
2009-06-08 10:47:49 +00:00
|
|
|
//
|
|
|
|
|
|
|
|
|
2014-04-16 14:30:58 +00:00
|
|
|
void Accessors::ScriptEvalFromScriptPositionGetter(
|
|
|
|
v8::Local<v8::String> name,
|
|
|
|
const v8::PropertyCallbackInfo<v8::Value>& info) {
|
|
|
|
i::Isolate* isolate = reinterpret_cast<i::Isolate*>(info.GetIsolate());
|
2013-09-02 09:25:20 +00:00
|
|
|
HandleScope scope(isolate);
|
2014-04-16 14:30:58 +00:00
|
|
|
Handle<Object> object = Utils::OpenHandle(*info.This());
|
|
|
|
Handle<Script> script(
|
|
|
|
Script::cast(Handle<JSValue>::cast(object)->value()), isolate);
|
|
|
|
Handle<Object> result = isolate->factory()->undefined_value();
|
|
|
|
if (script->compilation_type() == Script::COMPILATION_TYPE_EVAL) {
|
|
|
|
Handle<Code> code(SharedFunctionInfo::cast(
|
|
|
|
script->eval_from_shared())->code());
|
|
|
|
result = Handle<Object>(
|
|
|
|
Smi::FromInt(code->SourcePosition(code->instruction_start() +
|
|
|
|
script->eval_from_instructions_offset()->value())),
|
|
|
|
isolate);
|
2009-06-08 10:47:49 +00:00
|
|
|
}
|
2014-04-16 14:30:58 +00:00
|
|
|
info.GetReturnValue().Set(Utils::ToLocal(result));
|
|
|
|
}
|
|
|
|
|
2009-06-08 10:47:49 +00:00
|
|
|
|
2014-04-16 14:30:58 +00:00
|
|
|
void Accessors::ScriptEvalFromScriptPositionSetter(
|
|
|
|
v8::Local<v8::String> name,
|
|
|
|
v8::Local<v8::Value> value,
|
|
|
|
const v8::PropertyCallbackInfo<void>& info) {
|
|
|
|
UNREACHABLE();
|
2009-06-08 10:47:49 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-04-16 14:30:58 +00:00
|
|
|
Handle<AccessorInfo> Accessors::ScriptEvalFromScriptPositionInfo(
|
|
|
|
Isolate* isolate, PropertyAttributes attributes) {
|
|
|
|
Handle<String> name(isolate->factory()->InternalizeOneByteString(
|
|
|
|
STATIC_ASCII_VECTOR("eval_from_script_position")));
|
|
|
|
return MakeAccessor(isolate,
|
|
|
|
name,
|
|
|
|
&ScriptEvalFromScriptPositionGetter,
|
|
|
|
&ScriptEvalFromScriptPositionSetter,
|
|
|
|
attributes);
|
|
|
|
}
|
2009-12-01 14:36:45 +00:00
|
|
|
|
|
|
|
|
|
|
|
//
|
|
|
|
// Accessors::ScriptGetEvalFromFunctionName
|
|
|
|
//
|
|
|
|
|
|
|
|
|
2014-04-16 14:30:58 +00:00
|
|
|
void Accessors::ScriptEvalFromFunctionNameGetter(
|
|
|
|
v8::Local<v8::String> name,
|
|
|
|
const v8::PropertyCallbackInfo<v8::Value>& info) {
|
|
|
|
i::Isolate* isolate = reinterpret_cast<i::Isolate*>(info.GetIsolate());
|
|
|
|
HandleScope scope(isolate);
|
|
|
|
Handle<Object> object = Utils::OpenHandle(*info.This());
|
|
|
|
Handle<Script> script(
|
|
|
|
Script::cast(Handle<JSValue>::cast(object)->value()), isolate);
|
|
|
|
Handle<Object> result;
|
|
|
|
Handle<SharedFunctionInfo> shared(
|
|
|
|
SharedFunctionInfo::cast(script->eval_from_shared()));
|
2009-12-01 14:36:45 +00:00
|
|
|
// Find the name of the function calling eval.
|
|
|
|
if (!shared->name()->IsUndefined()) {
|
2014-04-16 14:30:58 +00:00
|
|
|
result = Handle<Object>(shared->name(), isolate);
|
2009-12-01 14:36:45 +00:00
|
|
|
} else {
|
2014-04-16 14:30:58 +00:00
|
|
|
result = Handle<Object>(shared->inferred_name(), isolate);
|
2009-12-01 14:36:45 +00:00
|
|
|
}
|
2014-04-16 14:30:58 +00:00
|
|
|
info.GetReturnValue().Set(Utils::ToLocal(result));
|
2009-12-01 14:36:45 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-04-16 14:30:58 +00:00
|
|
|
void Accessors::ScriptEvalFromFunctionNameSetter(
|
|
|
|
v8::Local<v8::String> name,
|
|
|
|
v8::Local<v8::Value> value,
|
|
|
|
const v8::PropertyCallbackInfo<void>& info) {
|
|
|
|
UNREACHABLE();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Handle<AccessorInfo> Accessors::ScriptEvalFromFunctionNameInfo(
|
|
|
|
Isolate* isolate, PropertyAttributes attributes) {
|
|
|
|
Handle<String> name(isolate->factory()->InternalizeOneByteString(
|
|
|
|
STATIC_ASCII_VECTOR("eval_from_function_name")));
|
|
|
|
return MakeAccessor(isolate,
|
|
|
|
name,
|
|
|
|
&ScriptEvalFromFunctionNameGetter,
|
|
|
|
&ScriptEvalFromFunctionNameSetter,
|
|
|
|
attributes);
|
|
|
|
}
|
2009-06-08 10:47:49 +00:00
|
|
|
|
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
//
|
|
|
|
// Accessors::FunctionPrototype
|
|
|
|
//
|
|
|
|
|
2014-04-17 09:12:19 +00:00
|
|
|
static Handle<Object> GetFunctionPrototype(Isolate* isolate,
|
2014-07-23 20:11:33 +00:00
|
|
|
Handle<JSFunction> function) {
|
2014-04-17 09:12:19 +00:00
|
|
|
if (!function->has_prototype()) {
|
|
|
|
Handle<Object> proto = isolate->factory()->NewFunctionPrototype(function);
|
|
|
|
JSFunction::SetPrototype(function, proto);
|
|
|
|
}
|
|
|
|
return Handle<Object>(function->prototype(), isolate);
|
2013-08-16 21:27:11 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-04-24 08:35:53 +00:00
|
|
|
static Handle<Object> SetFunctionPrototype(Isolate* isolate,
|
|
|
|
Handle<JSObject> receiver,
|
|
|
|
Handle<Object> value) {
|
|
|
|
Handle<JSFunction> function;
|
2014-04-17 09:12:19 +00:00
|
|
|
{
|
|
|
|
DisallowHeapAllocation no_allocation;
|
|
|
|
JSFunction* function_raw = FindInstanceOf<JSFunction>(isolate, *receiver);
|
|
|
|
if (function_raw == NULL) return isolate->factory()->undefined_value();
|
|
|
|
function = Handle<JSFunction>(function_raw, isolate);
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
2011-02-22 12:27:36 +00:00
|
|
|
|
2014-07-07 16:18:15 +00:00
|
|
|
if (!function->should_have_prototype()) {
|
|
|
|
// Since we hit this accessor, object will have no prototype property.
|
|
|
|
MaybeHandle<Object> maybe_result = JSObject::SetOwnPropertyIgnoreAttributes(
|
|
|
|
receiver, isolate->factory()->prototype_string(), value, NONE);
|
|
|
|
return maybe_result.ToHandleChecked();
|
|
|
|
}
|
2012-12-10 10:53:57 +00:00
|
|
|
|
|
|
|
Handle<Object> old_value;
|
2014-04-17 09:12:19 +00:00
|
|
|
bool is_observed = *function == *receiver && function->map()->is_observed();
|
2012-12-10 10:53:57 +00:00
|
|
|
if (is_observed) {
|
|
|
|
if (function->has_prototype())
|
|
|
|
old_value = handle(function->prototype(), isolate);
|
|
|
|
else
|
|
|
|
old_value = isolate->factory()->NewFunctionPrototype(function);
|
|
|
|
}
|
|
|
|
|
2013-07-18 07:59:48 +00:00
|
|
|
JSFunction::SetPrototype(function, value);
|
2012-12-10 10:53:57 +00:00
|
|
|
ASSERT(function->prototype() == *value);
|
|
|
|
|
|
|
|
if (is_observed && !old_value->SameValue(*value)) {
|
|
|
|
JSObject::EnqueueChangeRecord(
|
2013-11-06 12:14:24 +00:00
|
|
|
function, "update", isolate->factory()->prototype_string(), old_value);
|
2010-10-25 15:22:03 +00:00
|
|
|
}
|
2012-12-10 10:53:57 +00:00
|
|
|
|
2014-04-17 09:12:19 +00:00
|
|
|
return function;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-04-24 08:35:53 +00:00
|
|
|
Handle<Object> Accessors::FunctionGetPrototype(Handle<JSFunction> function) {
|
|
|
|
return GetFunctionPrototype(function->GetIsolate(), function);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-04-17 09:12:19 +00:00
|
|
|
Handle<Object> Accessors::FunctionSetPrototype(Handle<JSFunction> function,
|
|
|
|
Handle<Object> prototype) {
|
|
|
|
ASSERT(function->should_have_prototype());
|
|
|
|
Isolate* isolate = function->GetIsolate();
|
2014-04-24 08:35:53 +00:00
|
|
|
return SetFunctionPrototype(isolate, function, prototype);
|
2014-04-17 09:12:19 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-04-24 08:35:53 +00:00
|
|
|
void Accessors::FunctionPrototypeGetter(
|
|
|
|
v8::Local<v8::String> name,
|
|
|
|
const v8::PropertyCallbackInfo<v8::Value>& info) {
|
|
|
|
i::Isolate* isolate = reinterpret_cast<i::Isolate*>(info.GetIsolate());
|
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()));
|
|
|
|
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(
|
|
|
|
v8::Local<v8::String> name,
|
|
|
|
v8::Local<v8::Value> val,
|
|
|
|
const v8::PropertyCallbackInfo<void>& info) {
|
|
|
|
i::Isolate* isolate = reinterpret_cast<i::Isolate*>(info.GetIsolate());
|
|
|
|
HandleScope scope(isolate);
|
2014-04-28 13:42:03 +00:00
|
|
|
Handle<JSObject> object =
|
|
|
|
Handle<JSObject>::cast(Utils::OpenHandle(*info.This()));
|
2014-04-24 08:35:53 +00:00
|
|
|
Handle<Object> value = Utils::OpenHandle(*val);
|
|
|
|
|
|
|
|
SetFunctionPrototype(isolate, object, value);
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-04-24 08:35:53 +00:00
|
|
|
Handle<AccessorInfo> Accessors::FunctionPrototypeInfo(
|
|
|
|
Isolate* isolate, PropertyAttributes attributes) {
|
|
|
|
return MakeAccessor(isolate,
|
|
|
|
isolate->factory()->prototype_string(),
|
|
|
|
&FunctionPrototypeGetter,
|
|
|
|
&FunctionPrototypeSetter,
|
|
|
|
attributes);
|
|
|
|
}
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
|
|
|
|
//
|
|
|
|
// Accessors::FunctionLength
|
|
|
|
//
|
|
|
|
|
|
|
|
|
2014-04-24 11:24:13 +00:00
|
|
|
void Accessors::FunctionLengthGetter(
|
|
|
|
v8::Local<v8::String> name,
|
|
|
|
const v8::PropertyCallbackInfo<v8::Value>& info) {
|
|
|
|
i::Isolate* isolate = reinterpret_cast<i::Isolate*>(info.GetIsolate());
|
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()));
|
2014-04-24 11:24:13 +00:00
|
|
|
|
|
|
|
int length = 0;
|
2014-07-23 20:11:33 +00:00
|
|
|
if (function->shared()->is_compiled()) {
|
|
|
|
length = function->shared()->length();
|
|
|
|
} else {
|
|
|
|
// If the function isn't compiled yet, the length is not computed
|
|
|
|
// correctly yet. Compile it now and return the right length.
|
|
|
|
if (Compiler::EnsureCompiled(function, KEEP_EXCEPTION)) {
|
2014-04-24 11:24:13 +00:00
|
|
|
length = function->shared()->length();
|
2014-07-23 20:11:33 +00:00
|
|
|
}
|
|
|
|
if (isolate->has_pending_exception()) {
|
|
|
|
isolate->OptionalRescheduleException(false);
|
2014-04-24 11:24:13 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
Handle<Object> result(Smi::FromInt(length), isolate);
|
|
|
|
info.GetReturnValue().Set(Utils::ToLocal(result));
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-04-24 11:24:13 +00:00
|
|
|
void Accessors::FunctionLengthSetter(
|
|
|
|
v8::Local<v8::String> name,
|
|
|
|
v8::Local<v8::Value> val,
|
|
|
|
const v8::PropertyCallbackInfo<void>& info) {
|
|
|
|
// Do nothing.
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Handle<AccessorInfo> Accessors::FunctionLengthInfo(
|
|
|
|
Isolate* isolate, PropertyAttributes attributes) {
|
|
|
|
return MakeAccessor(isolate,
|
|
|
|
isolate->factory()->length_string(),
|
|
|
|
&FunctionLengthGetter,
|
|
|
|
&FunctionLengthSetter,
|
|
|
|
attributes);
|
|
|
|
}
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
|
|
|
|
//
|
|
|
|
// Accessors::FunctionName
|
|
|
|
//
|
|
|
|
|
|
|
|
|
2014-04-28 08:26:35 +00:00
|
|
|
void Accessors::FunctionNameGetter(
|
|
|
|
v8::Local<v8::String> name,
|
|
|
|
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()));
|
|
|
|
Handle<Object> result(function->shared()->name(), isolate);
|
2014-04-28 08:26:35 +00:00
|
|
|
info.GetReturnValue().Set(Utils::ToLocal(result));
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-04-28 08:26:35 +00:00
|
|
|
void Accessors::FunctionNameSetter(
|
|
|
|
v8::Local<v8::String> name,
|
|
|
|
v8::Local<v8::Value> val,
|
|
|
|
const v8::PropertyCallbackInfo<void>& info) {
|
|
|
|
// Do nothing.
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Handle<AccessorInfo> Accessors::FunctionNameInfo(
|
|
|
|
Isolate* isolate, PropertyAttributes attributes) {
|
|
|
|
return MakeAccessor(isolate,
|
|
|
|
isolate->factory()->name_string(),
|
|
|
|
&FunctionNameGetter,
|
|
|
|
&FunctionNameSetter,
|
|
|
|
attributes);
|
|
|
|
}
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
|
|
|
|
//
|
|
|
|
// Accessors::FunctionArguments
|
|
|
|
//
|
|
|
|
|
2010-12-07 11:31:57 +00:00
|
|
|
|
2014-04-24 14:23:15 +00:00
|
|
|
static Handle<Object> ArgumentsForInlinedFunction(
|
2010-12-07 11:31:57 +00:00
|
|
|
JavaScriptFrame* frame,
|
|
|
|
Handle<JSFunction> inlined_function,
|
|
|
|
int inlined_frame_index) {
|
2013-02-25 14:46:09 +00:00
|
|
|
Isolate* isolate = inlined_function->GetIsolate();
|
|
|
|
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
|
|
|
SlotRefValueBuilder slot_refs(
|
|
|
|
frame,
|
|
|
|
inlined_frame_index,
|
|
|
|
inlined_function->shared()->formal_parameter_count());
|
|
|
|
|
|
|
|
int args_count = slot_refs.args_length();
|
2010-12-07 11:31:57 +00:00
|
|
|
Handle<JSObject> arguments =
|
2011-03-18 20:35:07 +00:00
|
|
|
factory->NewArgumentsObject(inlined_function, args_count);
|
|
|
|
Handle<FixedArray> array = factory->NewFixedArray(args_count);
|
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
|
|
|
slot_refs.Prepare(isolate);
|
2010-12-07 11:31:57 +00:00
|
|
|
for (int i = 0; i < args_count; ++i) {
|
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
|
|
|
Handle<Object> value = slot_refs.GetNext(isolate, 0);
|
2010-12-07 11:31:57 +00:00
|
|
|
array->set(i, *value);
|
|
|
|
}
|
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
|
|
|
slot_refs.Finish(isolate);
|
2010-12-07 11:31:57 +00:00
|
|
|
arguments->set_elements(*array);
|
|
|
|
|
|
|
|
// Return the freshly allocated arguments object.
|
2014-04-24 14:23:15 +00:00
|
|
|
return arguments;
|
2010-12-07 11:31:57 +00:00
|
|
|
}
|
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
|
2014-04-24 14:23:15 +00:00
|
|
|
static int FindFunctionInFrame(JavaScriptFrame* frame,
|
|
|
|
Handle<JSFunction> function) {
|
|
|
|
DisallowHeapAllocation no_allocation;
|
|
|
|
List<JSFunction*> functions(2);
|
|
|
|
frame->GetFunctions(&functions);
|
|
|
|
for (int i = functions.length() - 1; i >= 0; i--) {
|
|
|
|
if (functions[i] == *function) return i;
|
|
|
|
}
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Handle<Object> GetFunctionArguments(Isolate* isolate,
|
|
|
|
Handle<JSFunction> function) {
|
|
|
|
if (function->shared()->native()) return isolate->factory()->null_value();
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
// Find the top invocation of the function by traversing frames.
|
Simplify isolates access during stack iteration (WAS: Move SafeStackFrameIterator::active_count_...)
While trying to fix Mac and Windows versions for this change:
http://codereview.chromium.org/6771047/, I figured out, that we
already store an isolate in StackFrameIterator, so we can use it in
frame objects, instead of requiring it from caller.
I've changed iterators usage to the following scheme: whenever a
caller maintains an isolate pointer, it just passes it to stack
iterator, and no more worries about passing it to frame content
accessors. If a caller uses current isolate, it can omit passing it
to iterator, in this case, an iterator will use the current isolate,
too.
There was a special case with LiveEdit, which creates
detached copies of frame objects.
R=vitalyr@chromium.org
BUG=none
TEST=none
Review URL: http://codereview.chromium.org/6794019
git-svn-id: http://v8.googlecode.com/svn/branches/bleeding_edge@7499 ce2b1a6d-e550-0410-aec6-3dcde31c8c00
2011-04-05 09:01:47 +00:00
|
|
|
for (JavaScriptFrameIterator it(isolate); !it.done(); it.Advance()) {
|
2008-07-03 15:10:15 +00:00
|
|
|
JavaScriptFrame* frame = it.frame();
|
2014-04-24 14:23:15 +00:00
|
|
|
int function_index = FindFunctionInFrame(frame, function);
|
|
|
|
if (function_index < 0) continue;
|
|
|
|
|
|
|
|
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, function_index);
|
|
|
|
}
|
2011-02-02 15:08:29 +00:00
|
|
|
|
2014-04-24 14:23:15 +00:00
|
|
|
if (!frame->is_optimized()) {
|
|
|
|
// If there is an arguments variable in the stack, we return that.
|
|
|
|
Handle<ScopeInfo> scope_info(function->shared()->scope_info());
|
|
|
|
int index = scope_info->StackSlotIndex(
|
|
|
|
isolate->heap()->arguments_string());
|
|
|
|
if (index >= 0) {
|
|
|
|
Handle<Object> arguments(frame->GetExpression(index), isolate);
|
|
|
|
if (!arguments->IsArgumentsMarker()) return arguments;
|
2010-12-07 11:31:57 +00:00
|
|
|
}
|
2009-06-24 08:01:38 +00:00
|
|
|
}
|
2014-04-24 14:23:15 +00:00
|
|
|
|
|
|
|
// If there is no arguments variable in the stack or we have an
|
|
|
|
// optimized frame, we find the frame that holds the actual arguments
|
|
|
|
// passed to the function.
|
|
|
|
it.AdvanceToArgumentsFrame();
|
|
|
|
frame = it.frame();
|
|
|
|
|
|
|
|
// Get the number of arguments and construct an arguments object
|
|
|
|
// mirror for the right frame.
|
|
|
|
const int length = frame->ComputeParametersCount();
|
|
|
|
Handle<JSObject> arguments = isolate->factory()->NewArgumentsObject(
|
|
|
|
function, length);
|
|
|
|
Handle<FixedArray> array = isolate->factory()->NewFixedArray(length);
|
|
|
|
|
|
|
|
// Copy the parameters to the arguments object.
|
|
|
|
ASSERT(array->length() == length);
|
|
|
|
for (int i = 0; i < length; i++) array->set(i, frame->GetParameter(i));
|
|
|
|
arguments->set_elements(*array);
|
|
|
|
|
|
|
|
// Return the freshly allocated arguments object.
|
|
|
|
return arguments;
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// No frame corresponding to the given function found. Return null.
|
2014-04-24 14:23:15 +00:00
|
|
|
return isolate->factory()->null_value();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Handle<Object> Accessors::FunctionGetArguments(Handle<JSFunction> function) {
|
|
|
|
return GetFunctionArguments(function->GetIsolate(), function);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-04-28 12:02:11 +00:00
|
|
|
void Accessors::FunctionArgumentsGetter(
|
|
|
|
v8::Local<v8::String> name,
|
|
|
|
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()));
|
|
|
|
Handle<Object> result = GetFunctionArguments(isolate, function);
|
2014-04-28 12:02:11 +00:00
|
|
|
info.GetReturnValue().Set(Utils::ToLocal(result));
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-04-28 12:02:11 +00:00
|
|
|
void Accessors::FunctionArgumentsSetter(
|
|
|
|
v8::Local<v8::String> name,
|
|
|
|
v8::Local<v8::Value> val,
|
|
|
|
const v8::PropertyCallbackInfo<void>& info) {
|
|
|
|
// Do nothing.
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Handle<AccessorInfo> Accessors::FunctionArgumentsInfo(
|
|
|
|
Isolate* isolate, PropertyAttributes attributes) {
|
|
|
|
return MakeAccessor(isolate,
|
|
|
|
isolate->factory()->arguments_string(),
|
|
|
|
&FunctionArgumentsGetter,
|
|
|
|
&FunctionArgumentsSetter,
|
|
|
|
attributes);
|
|
|
|
}
|
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:
|
2013-06-03 15:32:22 +00:00
|
|
|
FrameFunctionIterator(Isolate* isolate, const DisallowHeapAllocation& promise)
|
2014-05-19 13:45:45 +00:00
|
|
|
: isolate_(isolate),
|
|
|
|
frame_iterator_(isolate),
|
2011-08-25 13:38:58 +00:00
|
|
|
functions_(2),
|
|
|
|
index_(0) {
|
|
|
|
GetFunctions();
|
|
|
|
}
|
|
|
|
JSFunction* next() {
|
2014-05-19 13:45:45 +00:00
|
|
|
while (true) {
|
|
|
|
if (functions_.length() == 0) return NULL;
|
|
|
|
JSFunction* next_function = functions_[index_];
|
|
|
|
index_--;
|
|
|
|
if (index_ < 0) {
|
|
|
|
GetFunctions();
|
|
|
|
}
|
|
|
|
// Skip functions from other origins.
|
|
|
|
if (!AllowAccessToFunction(isolate_->context(), next_function)) continue;
|
|
|
|
return next_function;
|
2011-08-25 13:38:58 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Iterate through functions until the first occurence of 'function'.
|
|
|
|
// Returns true if 'function' is found, and false if the iterator ends
|
|
|
|
// without finding it.
|
|
|
|
bool Find(JSFunction* function) {
|
|
|
|
JSFunction* next_function;
|
|
|
|
do {
|
|
|
|
next_function = next();
|
|
|
|
if (next_function == function) return true;
|
|
|
|
} while (next_function != NULL);
|
|
|
|
return false;
|
|
|
|
}
|
2011-09-08 19:57:14 +00:00
|
|
|
|
2011-08-25 13:38:58 +00:00
|
|
|
private:
|
|
|
|
void GetFunctions() {
|
|
|
|
functions_.Rewind(0);
|
|
|
|
if (frame_iterator_.done()) return;
|
|
|
|
JavaScriptFrame* frame = frame_iterator_.frame();
|
|
|
|
frame->GetFunctions(&functions_);
|
|
|
|
ASSERT(functions_.length() > 0);
|
|
|
|
frame_iterator_.Advance();
|
|
|
|
index_ = functions_.length() - 1;
|
|
|
|
}
|
2014-05-19 13:45:45 +00:00
|
|
|
Isolate* isolate_;
|
2011-08-25 13:38:58 +00:00
|
|
|
JavaScriptFrameIterator frame_iterator_;
|
|
|
|
List<JSFunction*> functions_;
|
|
|
|
int index_;
|
|
|
|
};
|
|
|
|
|
|
|
|
|
2014-04-28 13:41:12 +00:00
|
|
|
MaybeHandle<JSFunction> FindCaller(Isolate* isolate,
|
|
|
|
Handle<JSFunction> function) {
|
2013-06-03 15:32:22 +00:00
|
|
|
DisallowHeapAllocation no_allocation;
|
|
|
|
FrameFunctionIterator it(isolate, no_allocation);
|
2014-04-28 13:41:12 +00:00
|
|
|
if (function->shared()->native()) {
|
|
|
|
return MaybeHandle<JSFunction>();
|
|
|
|
}
|
2011-08-25 13:38:58 +00:00
|
|
|
// Find the function from the frames.
|
|
|
|
if (!it.Find(*function)) {
|
|
|
|
// No frame corresponding to the given function found. Return null.
|
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.
|
|
|
|
JSFunction* caller;
|
|
|
|
do {
|
|
|
|
caller = it.next();
|
2014-04-28 13:41:12 +00:00
|
|
|
if (caller == NULL) return MaybeHandle<JSFunction>();
|
2011-08-25 13:38:58 +00:00
|
|
|
} while (caller->shared()->is_toplevel());
|
|
|
|
|
|
|
|
// If caller is a built-in function and caller's caller is also built-in,
|
|
|
|
// use that instead.
|
|
|
|
JSFunction* potential_caller = caller;
|
|
|
|
while (potential_caller != NULL && potential_caller->IsBuiltin()) {
|
|
|
|
caller = potential_caller;
|
|
|
|
potential_caller = it.next();
|
|
|
|
}
|
2012-09-05 08:19:49 +00:00
|
|
|
if (!caller->shared()->native() && potential_caller != NULL) {
|
|
|
|
caller = potential_caller;
|
|
|
|
}
|
2011-10-18 12:26:53 +00:00
|
|
|
// If caller is bound, return null. This is compatible with JSC, and
|
|
|
|
// allows us to make bound functions use the strict function map
|
|
|
|
// and its associated throwing caller and arguments.
|
|
|
|
if (caller->shared()->bound()) {
|
2014-04-28 13:41:12 +00:00
|
|
|
return MaybeHandle<JSFunction>();
|
2011-10-18 12:26:53 +00:00
|
|
|
}
|
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
|
2014-03-11 14:41:22 +00:00
|
|
|
if (caller->shared()->strict_mode() == STRICT) {
|
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.
|
|
|
|
if (!AllowAccessToFunction(isolate->context(), caller)) {
|
|
|
|
return MaybeHandle<JSFunction>();
|
|
|
|
}
|
2014-04-28 13:41:12 +00:00
|
|
|
return Handle<JSFunction>(caller);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void Accessors::FunctionCallerGetter(
|
|
|
|
v8::Local<v8::String> name,
|
|
|
|
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));
|
|
|
|
}
|
|
|
|
|
2012-11-23 15:47:58 +00:00
|
|
|
|
2014-04-28 13:41:12 +00:00
|
|
|
void Accessors::FunctionCallerSetter(
|
|
|
|
v8::Local<v8::String> name,
|
|
|
|
v8::Local<v8::Value> val,
|
|
|
|
const v8::PropertyCallbackInfo<void>& info) {
|
|
|
|
// Do nothing.
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-04-28 13:41:12 +00:00
|
|
|
Handle<AccessorInfo> Accessors::FunctionCallerInfo(
|
|
|
|
Isolate* isolate, PropertyAttributes attributes) {
|
|
|
|
return MakeAccessor(isolate,
|
|
|
|
isolate->factory()->caller_string(),
|
|
|
|
&FunctionCallerGetter,
|
|
|
|
&FunctionCallerSetter,
|
|
|
|
attributes);
|
|
|
|
}
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
|
2012-07-09 08:59:03 +00:00
|
|
|
//
|
|
|
|
// Accessors::MakeModuleExport
|
|
|
|
//
|
|
|
|
|
2013-06-05 12:36:33 +00:00
|
|
|
static void ModuleGetExport(
|
2012-07-09 08:59:03 +00:00
|
|
|
v8::Local<v8::String> property,
|
2013-06-05 12:36:33 +00:00
|
|
|
const v8::PropertyCallbackInfo<v8::Value>& info) {
|
2012-07-09 08:59:03 +00:00
|
|
|
JSModule* instance = JSModule::cast(*v8::Utils::OpenHandle(*info.Holder()));
|
|
|
|
Context* context = Context::cast(instance->context());
|
|
|
|
ASSERT(context->IsModuleContext());
|
|
|
|
int slot = info.Data()->Int32Value();
|
|
|
|
Object* value = context->get(slot);
|
2013-02-25 14:46:09 +00:00
|
|
|
Isolate* isolate = instance->GetIsolate();
|
2012-07-09 08:59:03 +00:00
|
|
|
if (value->IsTheHole()) {
|
|
|
|
Handle<String> name = v8::Utils::OpenHandle(*property);
|
|
|
|
isolate->ScheduleThrow(
|
|
|
|
*isolate->factory()->NewReferenceError("not_defined",
|
|
|
|
HandleVector(&name, 1)));
|
2013-06-05 12:36:33 +00:00
|
|
|
return;
|
2012-07-09 08:59:03 +00:00
|
|
|
}
|
2013-06-05 12:36:33 +00:00
|
|
|
info.GetReturnValue().Set(v8::Utils::ToLocal(Handle<Object>(value, isolate)));
|
2012-07-09 08:59:03 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static void ModuleSetExport(
|
|
|
|
v8::Local<v8::String> property,
|
|
|
|
v8::Local<v8::Value> value,
|
2013-06-05 12:36:33 +00:00
|
|
|
const v8::PropertyCallbackInfo<v8::Value>& info) {
|
2012-07-09 08:59:03 +00:00
|
|
|
JSModule* instance = JSModule::cast(*v8::Utils::OpenHandle(*info.Holder()));
|
|
|
|
Context* context = Context::cast(instance->context());
|
|
|
|
ASSERT(context->IsModuleContext());
|
|
|
|
int slot = info.Data()->Int32Value();
|
|
|
|
Object* old_value = context->get(slot);
|
|
|
|
if (old_value->IsTheHole()) {
|
|
|
|
Handle<String> name = v8::Utils::OpenHandle(*property);
|
|
|
|
Isolate* isolate = instance->GetIsolate();
|
|
|
|
isolate->ScheduleThrow(
|
|
|
|
*isolate->factory()->NewReferenceError("not_defined",
|
|
|
|
HandleVector(&name, 1)));
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
context->set(slot, *v8::Utils::OpenHandle(*value));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Handle<AccessorInfo> Accessors::MakeModuleExport(
|
|
|
|
Handle<String> name,
|
|
|
|
int index,
|
|
|
|
PropertyAttributes attributes) {
|
2013-09-02 09:27:27 +00:00
|
|
|
Isolate* isolate = name->GetIsolate();
|
|
|
|
Factory* factory = isolate->factory();
|
2013-02-12 14:33:08 +00:00
|
|
|
Handle<ExecutableAccessorInfo> info = factory->NewExecutableAccessorInfo();
|
2012-07-09 08:59:03 +00:00
|
|
|
info->set_property_attributes(attributes);
|
|
|
|
info->set_all_can_read(true);
|
|
|
|
info->set_all_can_write(true);
|
|
|
|
info->set_name(*name);
|
|
|
|
info->set_data(Smi::FromInt(index));
|
2013-09-02 09:27:27 +00:00
|
|
|
Handle<Object> getter = v8::FromCData(isolate, &ModuleGetExport);
|
|
|
|
Handle<Object> setter = v8::FromCData(isolate, &ModuleSetExport);
|
2012-07-09 16:22:53 +00:00
|
|
|
info->set_getter(*getter);
|
|
|
|
if (!(attributes & ReadOnly)) info->set_setter(*setter);
|
2012-07-09 08:59:03 +00:00
|
|
|
return info;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
} } // namespace v8::internal
|