2012-01-24 08:43:12 +00:00
|
|
|
// Copyright 2012 the V8 project authors. All rights reserved.
|
2008-07-03 15:10:15 +00:00
|
|
|
// Redistribution and use in source and binary forms, with or without
|
|
|
|
// modification, are permitted provided that the following conditions are
|
|
|
|
// met:
|
|
|
|
//
|
|
|
|
// * Redistributions of source code must retain the above copyright
|
|
|
|
// notice, this list of conditions and the following disclaimer.
|
|
|
|
// * Redistributions in binary form must reproduce the above
|
|
|
|
// copyright notice, this list of conditions and the following
|
|
|
|
// disclaimer in the documentation and/or other materials provided
|
|
|
|
// with the distribution.
|
|
|
|
// * Neither the name of Google Inc. nor the names of its
|
|
|
|
// contributors may be used to endorse or promote products derived
|
|
|
|
// from this software without specific prior written permission.
|
|
|
|
//
|
|
|
|
// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
|
|
|
|
// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
|
|
|
|
// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
|
|
|
|
// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
|
|
|
|
// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
|
|
|
|
// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
|
|
|
|
// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
|
|
|
|
// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
|
|
|
|
// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
|
|
|
// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
|
|
|
|
// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
|
|
|
|
|
|
#include "v8.h"
|
|
|
|
#include "accessors.h"
|
2012-01-25 00:16:07 +00:00
|
|
|
|
|
|
|
#include "contexts.h"
|
2010-12-07 11:31:57 +00:00
|
|
|
#include "deoptimizer.h"
|
2008-07-03 15:10:15 +00:00
|
|
|
#include "execution.h"
|
|
|
|
#include "factory.h"
|
2012-01-25 00:16:07 +00:00
|
|
|
#include "frames-inl.h"
|
|
|
|
#include "isolate.h"
|
2011-05-03 08:23:58 +00:00
|
|
|
#include "list-inl.h"
|
2012-01-25 00:16:07 +00:00
|
|
|
#include "property-details.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
|
|
|
|
|
|
|
|
|
|
|
template <class C>
|
2013-02-27 13:22:29 +00:00
|
|
|
static C* FindInstanceOf(Isolate* isolate, Object* obj) {
|
|
|
|
for (Object* cur = obj; !cur->IsNull(); cur = cur->GetPrototype(isolate)) {
|
2012-10-08 12:58:46 +00:00
|
|
|
if (Is<C>(cur)) return C::cast(cur);
|
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
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// Entry point that never should be called.
|
2013-09-02 09:25:20 +00:00
|
|
|
MaybeObject* Accessors::IllegalSetter(Isolate* isolate,
|
|
|
|
JSObject*,
|
|
|
|
Object*,
|
|
|
|
void*) {
|
2008-07-03 15:10:15 +00:00
|
|
|
UNREACHABLE();
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-09-02 09:25:20 +00:00
|
|
|
Object* Accessors::IllegalGetAccessor(Isolate* isolate,
|
|
|
|
Object* object,
|
|
|
|
void*) {
|
2008-07-03 15:10:15 +00:00
|
|
|
UNREACHABLE();
|
|
|
|
return object;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-09-02 09:25:20 +00:00
|
|
|
MaybeObject* Accessors::ReadOnlySetAccessor(Isolate* isolate,
|
|
|
|
JSObject*,
|
|
|
|
Object* value,
|
|
|
|
void*) {
|
2008-07-03 15:10:15 +00:00
|
|
|
// According to ECMA-262, section 8.6.2.2, page 28, setting
|
|
|
|
// read-only properties must be silently ignored.
|
|
|
|
return value;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-09-25 08:19:35 +00:00
|
|
|
static V8_INLINE bool CheckForName(Handle<String> name,
|
|
|
|
String* property_name,
|
|
|
|
int offset,
|
|
|
|
int* object_offset) {
|
|
|
|
if (name->Equals(property_name)) {
|
|
|
|
*object_offset = offset;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool Accessors::IsJSObjectFieldAccessor(
|
|
|
|
Handle<Map> map, Handle<String> name,
|
|
|
|
int* object_offset) {
|
|
|
|
Isolate* isolate = map->GetIsolate();
|
|
|
|
switch (map->instance_type()) {
|
|
|
|
case JS_ARRAY_TYPE:
|
|
|
|
return
|
|
|
|
CheckForName(name, isolate->heap()->length_string(),
|
|
|
|
JSArray::kLengthOffset, object_offset);
|
|
|
|
case JS_TYPED_ARRAY_TYPE:
|
|
|
|
return
|
|
|
|
CheckForName(name, isolate->heap()->length_string(),
|
|
|
|
JSTypedArray::kLengthOffset, object_offset) ||
|
|
|
|
CheckForName(name, isolate->heap()->byte_length_string(),
|
|
|
|
JSTypedArray::kByteLengthOffset, object_offset) ||
|
|
|
|
CheckForName(name, isolate->heap()->byte_offset_string(),
|
|
|
|
JSTypedArray::kByteOffsetOffset, object_offset) ||
|
|
|
|
CheckForName(name, isolate->heap()->buffer_string(),
|
|
|
|
JSTypedArray::kBufferOffset, object_offset);
|
|
|
|
case JS_ARRAY_BUFFER_TYPE:
|
|
|
|
return
|
|
|
|
CheckForName(name, isolate->heap()->byte_length_string(),
|
|
|
|
JSArrayBuffer::kByteLengthOffset, object_offset);
|
|
|
|
case JS_DATA_VIEW_TYPE:
|
|
|
|
return
|
|
|
|
CheckForName(name, isolate->heap()->byte_length_string(),
|
|
|
|
JSDataView::kByteLengthOffset, object_offset) ||
|
|
|
|
CheckForName(name, isolate->heap()->byte_offset_string(),
|
|
|
|
JSDataView::kByteOffsetOffset, object_offset) ||
|
|
|
|
CheckForName(name, isolate->heap()->buffer_string(),
|
|
|
|
JSDataView::kBufferOffset, object_offset);
|
|
|
|
default: {
|
|
|
|
if (map->instance_type() < FIRST_NONSTRING_TYPE) {
|
|
|
|
return
|
|
|
|
CheckForName(name, isolate->heap()->length_string(),
|
|
|
|
String::kLengthOffset, object_offset);
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
//
|
|
|
|
// Accessors::ArrayLength
|
|
|
|
//
|
|
|
|
|
|
|
|
|
2013-09-02 09:25:20 +00:00
|
|
|
MaybeObject* Accessors::ArrayGetLength(Isolate* isolate,
|
|
|
|
Object* object,
|
|
|
|
void*) {
|
2008-07-03 15:10:15 +00:00
|
|
|
// Traverse the prototype chain until we reach an array.
|
2013-09-02 09:25:20 +00:00
|
|
|
JSArray* holder = FindInstanceOf<JSArray>(isolate, object);
|
2012-10-08 12:58:46 +00:00
|
|
|
return holder == NULL ? Smi::FromInt(0) : holder->length();
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// The helper function will 'flatten' Number objects.
|
2013-09-02 09:25:20 +00:00
|
|
|
Object* Accessors::FlattenNumber(Isolate* isolate, Object* value) {
|
2008-07-03 15:10:15 +00:00
|
|
|
if (value->IsNumber() || !value->IsJSValue()) return value;
|
|
|
|
JSValue* wrapper = JSValue::cast(value);
|
2013-09-02 09:25:20 +00:00
|
|
|
ASSERT(wrapper->GetIsolate()->context()->native_context()->number_function()->
|
2011-03-18 20:35:07 +00:00
|
|
|
has_initial_map());
|
2013-09-02 09:25:20 +00:00
|
|
|
Map* number_map = isolate->context()->native_context()->
|
2011-03-18 20:35:07 +00:00
|
|
|
number_function()->initial_map();
|
2008-07-03 15:10:15 +00:00
|
|
|
if (wrapper->map() == number_map) return wrapper->value();
|
|
|
|
return value;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-09-02 09:25:20 +00:00
|
|
|
MaybeObject* Accessors::ArraySetLength(Isolate* isolate,
|
|
|
|
JSObject* object,
|
|
|
|
Object* value,
|
|
|
|
void*) {
|
2011-06-21 08:07:45 +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()) {
|
2013-09-12 13:27:42 +00:00
|
|
|
return object->SetLocalPropertyIgnoreAttributesTrampoline(
|
2013-02-28 17:03:34 +00:00
|
|
|
isolate->heap()->length_string(), value, NONE);
|
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
|
|
|
|
|
|
|
// Need to call methods that may trigger GC.
|
2011-03-29 07:34:23 +00:00
|
|
|
HandleScope scope(isolate);
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
// Protect raw pointers.
|
2012-11-08 16:12:12 +00:00
|
|
|
Handle<JSArray> array_handle(JSArray::cast(object), isolate);
|
2011-03-29 07:34:23 +00:00
|
|
|
Handle<Object> value_handle(value, isolate);
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
bool has_exception;
|
2013-09-03 06:59:01 +00:00
|
|
|
Handle<Object> uint32_v =
|
|
|
|
Execution::ToUint32(isolate, value_handle, &has_exception);
|
2008-07-03 15:10:15 +00:00
|
|
|
if (has_exception) return Failure::Exception();
|
2013-09-03 06:59:01 +00:00
|
|
|
Handle<Object> number_v =
|
|
|
|
Execution::ToNumber(isolate, value_handle, &has_exception);
|
2008-07-03 15:10:15 +00:00
|
|
|
if (has_exception) return Failure::Exception();
|
|
|
|
|
|
|
|
if (uint32_v->Number() == number_v->Number()) {
|
2012-11-14 16:51:21 +00:00
|
|
|
return array_handle->SetElementsLength(*uint32_v);
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
2011-03-29 07:34:23 +00:00
|
|
|
return isolate->Throw(
|
|
|
|
*isolate->factory()->NewRangeError("invalid_array_length",
|
|
|
|
HandleVector<Object>(NULL, 0)));
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
const AccessorDescriptor Accessors::ArrayLength = {
|
|
|
|
ArrayGetLength,
|
|
|
|
ArraySetLength,
|
|
|
|
0
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
//
|
|
|
|
// Accessors::StringLength
|
|
|
|
//
|
|
|
|
|
|
|
|
|
2013-09-02 09:25:20 +00:00
|
|
|
MaybeObject* Accessors::StringGetLength(Isolate* isolate,
|
|
|
|
Object* object,
|
|
|
|
void*) {
|
2008-07-03 15:10:15 +00:00
|
|
|
Object* value = object;
|
|
|
|
if (object->IsJSValue()) value = JSValue::cast(object)->value();
|
|
|
|
if (value->IsString()) return Smi::FromInt(String::cast(value)->length());
|
|
|
|
// If object is not a string we return 0 to be compatible with WebKit.
|
|
|
|
// Note: Firefox returns the length of ToString(object).
|
|
|
|
return Smi::FromInt(0);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
const AccessorDescriptor Accessors::StringLength = {
|
|
|
|
StringGetLength,
|
|
|
|
IllegalSetter,
|
|
|
|
0
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
//
|
|
|
|
// Accessors::ScriptSource
|
|
|
|
//
|
|
|
|
|
|
|
|
|
2013-09-02 09:25:20 +00:00
|
|
|
MaybeObject* Accessors::ScriptGetSource(Isolate* isolate,
|
|
|
|
Object* object,
|
|
|
|
void*) {
|
2008-07-03 15:10:15 +00:00
|
|
|
Object* script = JSValue::cast(object)->value();
|
|
|
|
return Script::cast(script)->source();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
const AccessorDescriptor Accessors::ScriptSource = {
|
|
|
|
ScriptGetSource,
|
|
|
|
IllegalSetter,
|
|
|
|
0
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
//
|
|
|
|
// Accessors::ScriptName
|
|
|
|
//
|
|
|
|
|
|
|
|
|
2013-09-02 09:25:20 +00:00
|
|
|
MaybeObject* Accessors::ScriptGetName(Isolate* isolate,
|
|
|
|
Object* object,
|
|
|
|
void*) {
|
2008-07-03 15:10:15 +00:00
|
|
|
Object* script = JSValue::cast(object)->value();
|
|
|
|
return Script::cast(script)->name();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
const AccessorDescriptor Accessors::ScriptName = {
|
|
|
|
ScriptGetName,
|
|
|
|
IllegalSetter,
|
|
|
|
0
|
|
|
|
};
|
|
|
|
|
|
|
|
|
2009-03-10 08:10:50 +00:00
|
|
|
//
|
|
|
|
// Accessors::ScriptId
|
|
|
|
//
|
|
|
|
|
|
|
|
|
2013-09-02 09:25:20 +00:00
|
|
|
MaybeObject* Accessors::ScriptGetId(Isolate* isolate, Object* object, void*) {
|
2009-03-10 08:10:50 +00:00
|
|
|
Object* script = JSValue::cast(object)->value();
|
|
|
|
return Script::cast(script)->id();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
const AccessorDescriptor Accessors::ScriptId = {
|
|
|
|
ScriptGetId,
|
|
|
|
IllegalSetter,
|
|
|
|
0
|
|
|
|
};
|
|
|
|
|
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
//
|
|
|
|
// Accessors::ScriptLineOffset
|
|
|
|
//
|
|
|
|
|
|
|
|
|
2013-09-02 09:25:20 +00:00
|
|
|
MaybeObject* Accessors::ScriptGetLineOffset(Isolate* isolate,
|
|
|
|
Object* object,
|
|
|
|
void*) {
|
2008-07-03 15:10:15 +00:00
|
|
|
Object* script = JSValue::cast(object)->value();
|
|
|
|
return Script::cast(script)->line_offset();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
const AccessorDescriptor Accessors::ScriptLineOffset = {
|
|
|
|
ScriptGetLineOffset,
|
|
|
|
IllegalSetter,
|
|
|
|
0
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
//
|
|
|
|
// Accessors::ScriptColumnOffset
|
|
|
|
//
|
|
|
|
|
|
|
|
|
2013-09-02 09:25:20 +00:00
|
|
|
MaybeObject* Accessors::ScriptGetColumnOffset(Isolate* isolate,
|
|
|
|
Object* object,
|
|
|
|
void*) {
|
2008-07-03 15:10:15 +00:00
|
|
|
Object* script = JSValue::cast(object)->value();
|
|
|
|
return Script::cast(script)->column_offset();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
const AccessorDescriptor Accessors::ScriptColumnOffset = {
|
|
|
|
ScriptGetColumnOffset,
|
|
|
|
IllegalSetter,
|
|
|
|
0
|
|
|
|
};
|
|
|
|
|
|
|
|
|
2009-04-21 07:22:06 +00:00
|
|
|
//
|
|
|
|
// Accessors::ScriptData
|
|
|
|
//
|
|
|
|
|
|
|
|
|
2013-09-02 09:25:20 +00:00
|
|
|
MaybeObject* Accessors::ScriptGetData(Isolate* isolate,
|
|
|
|
Object* object,
|
|
|
|
void*) {
|
2009-04-21 07:22:06 +00:00
|
|
|
Object* script = JSValue::cast(object)->value();
|
|
|
|
return Script::cast(script)->data();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
const AccessorDescriptor Accessors::ScriptData = {
|
|
|
|
ScriptGetData,
|
|
|
|
IllegalSetter,
|
|
|
|
0
|
|
|
|
};
|
|
|
|
|
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
//
|
|
|
|
// Accessors::ScriptType
|
|
|
|
//
|
|
|
|
|
|
|
|
|
2013-09-02 09:25:20 +00:00
|
|
|
MaybeObject* Accessors::ScriptGetType(Isolate* isolate,
|
|
|
|
Object* object,
|
|
|
|
void*) {
|
2008-07-03 15:10:15 +00:00
|
|
|
Object* script = JSValue::cast(object)->value();
|
|
|
|
return Script::cast(script)->type();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
const AccessorDescriptor Accessors::ScriptType = {
|
|
|
|
ScriptGetType,
|
|
|
|
IllegalSetter,
|
|
|
|
0
|
|
|
|
};
|
|
|
|
|
|
|
|
|
2009-06-08 10:47:49 +00:00
|
|
|
//
|
|
|
|
// Accessors::ScriptCompilationType
|
|
|
|
//
|
|
|
|
|
|
|
|
|
2013-09-02 09:25:20 +00:00
|
|
|
MaybeObject* Accessors::ScriptGetCompilationType(Isolate* isolate,
|
|
|
|
Object* object,
|
|
|
|
void*) {
|
2009-06-08 10:47:49 +00:00
|
|
|
Object* script = JSValue::cast(object)->value();
|
2013-07-30 17:00:05 +00:00
|
|
|
return Smi::FromInt(Script::cast(script)->compilation_type());
|
2009-06-08 10:47:49 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
const AccessorDescriptor Accessors::ScriptCompilationType = {
|
|
|
|
ScriptGetCompilationType,
|
|
|
|
IllegalSetter,
|
|
|
|
0
|
|
|
|
};
|
|
|
|
|
|
|
|
|
2009-02-04 12:07:45 +00:00
|
|
|
//
|
|
|
|
// Accessors::ScriptGetLineEnds
|
|
|
|
//
|
|
|
|
|
|
|
|
|
2013-09-02 09:25:20 +00:00
|
|
|
MaybeObject* Accessors::ScriptGetLineEnds(Isolate* isolate,
|
|
|
|
Object* object,
|
|
|
|
void*) {
|
2011-03-29 07:34:23 +00:00
|
|
|
JSValue* wrapper = JSValue::cast(object);
|
|
|
|
HandleScope scope(isolate);
|
|
|
|
Handle<Script> script(Script::cast(wrapper->value()), isolate);
|
2009-03-10 15:08:45 +00:00
|
|
|
InitScriptLineEnds(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);
|
2009-11-27 14:10:48 +00:00
|
|
|
return *js_array;
|
2009-02-04 12:07:45 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
const AccessorDescriptor Accessors::ScriptLineEnds = {
|
|
|
|
ScriptGetLineEnds,
|
|
|
|
IllegalSetter,
|
|
|
|
0
|
|
|
|
};
|
|
|
|
|
|
|
|
|
2009-05-06 08:52:48 +00:00
|
|
|
//
|
|
|
|
// Accessors::ScriptGetContextData
|
|
|
|
//
|
|
|
|
|
|
|
|
|
2013-09-02 09:25:20 +00:00
|
|
|
MaybeObject* Accessors::ScriptGetContextData(Isolate* isolate,
|
|
|
|
Object* object,
|
|
|
|
void*) {
|
2009-06-08 10:47:49 +00:00
|
|
|
Object* script = JSValue::cast(object)->value();
|
|
|
|
return Script::cast(script)->context_data();
|
2009-05-06 08:52:48 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
const AccessorDescriptor Accessors::ScriptContextData = {
|
|
|
|
ScriptGetContextData,
|
|
|
|
IllegalSetter,
|
|
|
|
0
|
|
|
|
};
|
|
|
|
|
|
|
|
|
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
|
|
|
//
|
|
|
|
|
|
|
|
|
2013-09-02 09:25:20 +00:00
|
|
|
MaybeObject* Accessors::ScriptGetEvalFromScript(Isolate* isolate,
|
|
|
|
Object* object,
|
|
|
|
void*) {
|
2009-06-08 10:47:49 +00:00
|
|
|
Object* script = JSValue::cast(object)->value();
|
2009-12-01 14:36:45 +00:00
|
|
|
if (!Script::cast(script)->eval_from_shared()->IsUndefined()) {
|
|
|
|
Handle<SharedFunctionInfo> eval_from_shared(
|
|
|
|
SharedFunctionInfo::cast(Script::cast(script)->eval_from_shared()));
|
|
|
|
|
|
|
|
if (eval_from_shared->script()->IsScript()) {
|
|
|
|
Handle<Script> eval_from_script(Script::cast(eval_from_shared->script()));
|
|
|
|
return *GetScriptWrapper(eval_from_script);
|
|
|
|
}
|
|
|
|
}
|
2013-09-10 14:30:36 +00:00
|
|
|
return isolate->heap()->undefined_value();
|
2009-06-08 10:47:49 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2009-12-01 14:36:45 +00:00
|
|
|
const AccessorDescriptor Accessors::ScriptEvalFromScript = {
|
|
|
|
ScriptGetEvalFromScript,
|
2009-06-08 10:47:49 +00:00
|
|
|
IllegalSetter,
|
|
|
|
0
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
//
|
2009-12-01 14:36:45 +00:00
|
|
|
// Accessors::ScriptGetEvalFromScriptPosition
|
2009-06-08 10:47:49 +00:00
|
|
|
//
|
|
|
|
|
|
|
|
|
2013-09-02 09:25:20 +00:00
|
|
|
MaybeObject* Accessors::ScriptGetEvalFromScriptPosition(Isolate* isolate,
|
|
|
|
Object* object,
|
|
|
|
void*) {
|
2013-02-15 09:27:10 +00:00
|
|
|
Script* raw_script = Script::cast(JSValue::cast(object)->value());
|
2013-09-02 09:25:20 +00:00
|
|
|
HandleScope scope(isolate);
|
2013-02-15 09:27:10 +00:00
|
|
|
Handle<Script> script(raw_script);
|
2009-06-08 10:47:49 +00:00
|
|
|
|
|
|
|
// If this is not a script compiled through eval there is no eval position.
|
2013-07-30 17:00:05 +00:00
|
|
|
if (script->compilation_type() != Script::COMPILATION_TYPE_EVAL) {
|
2013-02-15 09:27:10 +00:00
|
|
|
return script->GetHeap()->undefined_value();
|
2009-06-08 10:47:49 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Get the function from where eval was called and find the source position
|
|
|
|
// from the instruction offset.
|
2009-12-01 14:36:45 +00:00
|
|
|
Handle<Code> code(SharedFunctionInfo::cast(
|
|
|
|
script->eval_from_shared())->code());
|
2009-06-08 10:47:49 +00:00
|
|
|
return Smi::FromInt(code->SourcePosition(code->instruction_start() +
|
|
|
|
script->eval_from_instructions_offset()->value()));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2009-12-01 14:36:45 +00:00
|
|
|
const AccessorDescriptor Accessors::ScriptEvalFromScriptPosition = {
|
|
|
|
ScriptGetEvalFromScriptPosition,
|
|
|
|
IllegalSetter,
|
|
|
|
0
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
//
|
|
|
|
// Accessors::ScriptGetEvalFromFunctionName
|
|
|
|
//
|
|
|
|
|
|
|
|
|
2013-09-02 09:25:20 +00:00
|
|
|
MaybeObject* Accessors::ScriptGetEvalFromFunctionName(Isolate* isolate,
|
|
|
|
Object* object,
|
|
|
|
void*) {
|
2009-12-01 14:36:45 +00:00
|
|
|
Object* script = JSValue::cast(object)->value();
|
|
|
|
Handle<SharedFunctionInfo> shared(SharedFunctionInfo::cast(
|
|
|
|
Script::cast(script)->eval_from_shared()));
|
|
|
|
|
|
|
|
|
|
|
|
// Find the name of the function calling eval.
|
|
|
|
if (!shared->name()->IsUndefined()) {
|
|
|
|
return shared->name();
|
|
|
|
} else {
|
|
|
|
return shared->inferred_name();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
const AccessorDescriptor Accessors::ScriptEvalFromFunctionName = {
|
|
|
|
ScriptGetEvalFromFunctionName,
|
2009-06-08 10:47:49 +00:00
|
|
|
IllegalSetter,
|
|
|
|
0
|
|
|
|
};
|
|
|
|
|
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
//
|
|
|
|
// Accessors::FunctionPrototype
|
|
|
|
//
|
|
|
|
|
|
|
|
|
2013-08-16 21:27:11 +00:00
|
|
|
Handle<Object> Accessors::FunctionGetPrototype(Handle<JSFunction> function) {
|
|
|
|
CALL_HEAP_FUNCTION(function->GetIsolate(),
|
2013-09-02 09:25:20 +00:00
|
|
|
Accessors::FunctionGetPrototype(function->GetIsolate(),
|
|
|
|
*function,
|
|
|
|
NULL),
|
2013-08-16 21:27:11 +00:00
|
|
|
Object);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Handle<Object> Accessors::FunctionSetPrototype(Handle<JSFunction> function,
|
|
|
|
Handle<Object> prototype) {
|
|
|
|
ASSERT(function->should_have_prototype());
|
|
|
|
CALL_HEAP_FUNCTION(function->GetIsolate(),
|
2013-09-02 09:25:20 +00:00
|
|
|
Accessors::FunctionSetPrototype(function->GetIsolate(),
|
|
|
|
*function,
|
2013-08-16 21:27:11 +00:00
|
|
|
*prototype,
|
|
|
|
NULL),
|
|
|
|
Object);
|
2013-04-19 08:30:49 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-09-02 09:25:20 +00:00
|
|
|
MaybeObject* Accessors::FunctionGetPrototype(Isolate* isolate,
|
|
|
|
Object* object,
|
|
|
|
void*) {
|
2013-07-18 07:59:48 +00:00
|
|
|
JSFunction* function_raw = FindInstanceOf<JSFunction>(isolate, object);
|
|
|
|
if (function_raw == NULL) return isolate->heap()->undefined_value();
|
|
|
|
while (!function_raw->should_have_prototype()) {
|
|
|
|
function_raw = FindInstanceOf<JSFunction>(isolate,
|
|
|
|
function_raw->GetPrototype());
|
2011-02-22 12:27:36 +00:00
|
|
|
// There has to be one because we hit the getter.
|
2013-07-18 07:59:48 +00:00
|
|
|
ASSERT(function_raw != NULL);
|
2011-02-22 12:27:36 +00:00
|
|
|
}
|
|
|
|
|
2013-07-18 07:59:48 +00:00
|
|
|
if (!function_raw->has_prototype()) {
|
|
|
|
HandleScope scope(isolate);
|
|
|
|
Handle<JSFunction> function(function_raw);
|
|
|
|
Handle<Object> proto = isolate->factory()->NewFunctionPrototype(function);
|
|
|
|
JSFunction::SetPrototype(function, proto);
|
|
|
|
function_raw = *function;
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
2013-07-18 07:59:48 +00:00
|
|
|
return function_raw->prototype();
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-09-02 09:25:20 +00:00
|
|
|
MaybeObject* Accessors::FunctionSetPrototype(Isolate* isolate,
|
|
|
|
JSObject* object,
|
2012-12-10 10:53:57 +00:00
|
|
|
Object* value_raw,
|
2010-10-25 15:22:03 +00:00
|
|
|
void*) {
|
2012-12-10 10:53:57 +00:00
|
|
|
Heap* heap = isolate->heap();
|
2013-02-27 13:22:29 +00:00
|
|
|
JSFunction* function_raw = FindInstanceOf<JSFunction>(isolate, object);
|
2012-12-10 10:53:57 +00:00
|
|
|
if (function_raw == NULL) return heap->undefined_value();
|
|
|
|
if (!function_raw->should_have_prototype()) {
|
2011-02-22 12:27:36 +00:00
|
|
|
// Since we hit this accessor, object will have no prototype property.
|
2013-09-12 13:27:42 +00:00
|
|
|
return object->SetLocalPropertyIgnoreAttributesTrampoline(
|
|
|
|
heap->prototype_string(), value_raw, NONE);
|
2011-02-22 12:27:36 +00:00
|
|
|
}
|
|
|
|
|
2012-12-10 10:53:57 +00:00
|
|
|
HandleScope scope(isolate);
|
|
|
|
Handle<JSFunction> function(function_raw, isolate);
|
|
|
|
Handle<Object> value(value_raw, isolate);
|
|
|
|
|
|
|
|
Handle<Object> old_value;
|
|
|
|
bool is_observed =
|
|
|
|
FLAG_harmony_observation &&
|
|
|
|
*function == object &&
|
|
|
|
function->map()->is_observed();
|
|
|
|
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-02-28 17:03:34 +00:00
|
|
|
function, "updated", isolate->factory()->prototype_string(), old_value);
|
2010-10-25 15:22:03 +00:00
|
|
|
}
|
2012-12-10 10:53:57 +00:00
|
|
|
|
|
|
|
return *function;
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
const AccessorDescriptor Accessors::FunctionPrototype = {
|
|
|
|
FunctionGetPrototype,
|
|
|
|
FunctionSetPrototype,
|
|
|
|
0
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
//
|
|
|
|
// Accessors::FunctionLength
|
|
|
|
//
|
|
|
|
|
|
|
|
|
2013-09-02 09:25:20 +00:00
|
|
|
MaybeObject* Accessors::FunctionGetLength(Isolate* isolate,
|
|
|
|
Object* object,
|
|
|
|
void*) {
|
2013-02-27 13:22:29 +00:00
|
|
|
JSFunction* function = FindInstanceOf<JSFunction>(isolate, object);
|
2012-10-08 12:58:46 +00:00
|
|
|
if (function == NULL) return Smi::FromInt(0);
|
2008-07-03 15:10:15 +00:00
|
|
|
// Check if already compiled.
|
2012-10-08 12:58:46 +00:00
|
|
|
if (function->shared()->is_compiled()) {
|
2008-07-03 15:10:15 +00:00
|
|
|
return Smi::FromInt(function->shared()->length());
|
|
|
|
}
|
2012-10-08 12:58:46 +00:00
|
|
|
// If the function isn't compiled yet, the length is not computed correctly
|
|
|
|
// yet. Compile it now and return the right length.
|
2013-02-27 13:22:29 +00:00
|
|
|
HandleScope scope(isolate);
|
2012-10-08 12:58:46 +00:00
|
|
|
Handle<JSFunction> handle(function);
|
|
|
|
if (JSFunction::CompileLazy(handle, KEEP_EXCEPTION)) {
|
|
|
|
return Smi::FromInt(handle->shared()->length());
|
|
|
|
}
|
|
|
|
return Failure::Exception();
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
const AccessorDescriptor Accessors::FunctionLength = {
|
|
|
|
FunctionGetLength,
|
|
|
|
ReadOnlySetAccessor,
|
|
|
|
0
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
//
|
|
|
|
// Accessors::FunctionName
|
|
|
|
//
|
|
|
|
|
|
|
|
|
2013-09-02 09:25:20 +00:00
|
|
|
MaybeObject* Accessors::FunctionGetName(Isolate* isolate,
|
|
|
|
Object* object,
|
|
|
|
void*) {
|
2013-02-27 13:22:29 +00:00
|
|
|
JSFunction* holder = FindInstanceOf<JSFunction>(isolate, object);
|
|
|
|
return holder == NULL
|
|
|
|
? isolate->heap()->undefined_value()
|
|
|
|
: holder->shared()->name();
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
const AccessorDescriptor Accessors::FunctionName = {
|
|
|
|
FunctionGetName,
|
|
|
|
ReadOnlySetAccessor,
|
|
|
|
0
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
//
|
|
|
|
// Accessors::FunctionArguments
|
|
|
|
//
|
|
|
|
|
2010-12-07 11:31:57 +00:00
|
|
|
|
2013-08-16 21:27:11 +00:00
|
|
|
Handle<Object> Accessors::FunctionGetArguments(Handle<JSFunction> function) {
|
|
|
|
CALL_HEAP_FUNCTION(function->GetIsolate(),
|
2013-09-02 09:25:20 +00:00
|
|
|
Accessors::FunctionGetArguments(function->GetIsolate(),
|
|
|
|
*function,
|
|
|
|
NULL),
|
2013-08-16 21:27:11 +00:00
|
|
|
Object);
|
2013-07-18 14:00:53 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-12-07 11:31:57 +00:00
|
|
|
static MaybeObject* ConstructArgumentsObjectForInlinedFunction(
|
|
|
|
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();
|
2012-01-24 08:43:12 +00:00
|
|
|
Vector<SlotRef> args_slots =
|
|
|
|
SlotRef::ComputeSlotMappingForArguments(
|
|
|
|
frame,
|
|
|
|
inlined_frame_index,
|
|
|
|
inlined_function->shared()->formal_parameter_count());
|
|
|
|
int args_count = args_slots.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);
|
2010-12-07 11:31:57 +00:00
|
|
|
for (int i = 0; i < args_count; ++i) {
|
2013-02-25 14:46:09 +00:00
|
|
|
Handle<Object> value = args_slots[i].GetValue(isolate);
|
2010-12-07 11:31:57 +00:00
|
|
|
array->set(i, *value);
|
|
|
|
}
|
|
|
|
arguments->set_elements(*array);
|
2012-01-24 08:43:12 +00:00
|
|
|
args_slots.Dispose();
|
2010-12-07 11:31:57 +00:00
|
|
|
|
|
|
|
// Return the freshly allocated arguments object.
|
|
|
|
return *arguments;
|
|
|
|
}
|
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
|
2013-09-02 09:25:20 +00:00
|
|
|
MaybeObject* Accessors::FunctionGetArguments(Isolate* isolate,
|
|
|
|
Object* object,
|
|
|
|
void*) {
|
2011-03-18 20:35:07 +00:00
|
|
|
HandleScope scope(isolate);
|
2013-02-27 13:22:29 +00:00
|
|
|
JSFunction* holder = FindInstanceOf<JSFunction>(isolate, object);
|
2012-10-08 12:58:46 +00:00
|
|
|
if (holder == NULL) return isolate->heap()->undefined_value();
|
2011-03-18 20:35:07 +00:00
|
|
|
Handle<JSFunction> function(holder, isolate);
|
2008-07-03 15:10:15 +00:00
|
|
|
|
2011-09-06 14:03:32 +00:00
|
|
|
if (function->shared()->native()) return isolate->heap()->null_value();
|
2008-07-03 15:10:15 +00:00
|
|
|
// Find the top invocation of the function by traversing frames.
|
2010-12-07 11:31:57 +00:00
|
|
|
List<JSFunction*> functions(2);
|
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();
|
2010-12-07 11:31:57 +00:00
|
|
|
frame->GetFunctions(&functions);
|
|
|
|
for (int i = functions.length() - 1; i >= 0; i--) {
|
|
|
|
// Skip all frames that aren't invocations of the given function.
|
|
|
|
if (functions[i] != *function) continue;
|
|
|
|
|
|
|
|
if (i > 0) {
|
2011-02-02 15:08:29 +00:00
|
|
|
// 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.
|
2010-12-07 11:31:57 +00:00
|
|
|
return ConstructArgumentsObjectForInlinedFunction(frame, function, i);
|
2011-02-02 15:08:29 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (!frame->is_optimized()) {
|
2010-12-07 11:31:57 +00:00
|
|
|
// If there is an arguments variable in the stack, we return that.
|
2011-11-03 10:36:55 +00:00
|
|
|
Handle<ScopeInfo> scope_info(function->shared()->scope_info());
|
|
|
|
int index = scope_info->StackSlotIndex(
|
2013-02-28 17:03:34 +00:00
|
|
|
isolate->heap()->arguments_string());
|
2010-12-07 11:31:57 +00:00
|
|
|
if (index >= 0) {
|
2011-03-18 20:35:07 +00:00
|
|
|
Handle<Object> arguments(frame->GetExpression(index), isolate);
|
2011-01-06 15:53:56 +00:00
|
|
|
if (!arguments->IsArgumentsMarker()) return *arguments;
|
2010-12-07 11:31:57 +00:00
|
|
|
}
|
|
|
|
}
|
2011-02-02 15:08:29 +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.
|
2011-03-08 14:18:28 +00:00
|
|
|
const int length = frame->ComputeParametersCount();
|
2011-03-18 20:35:07 +00:00
|
|
|
Handle<JSObject> arguments = isolate->factory()->NewArgumentsObject(
|
|
|
|
function, length);
|
|
|
|
Handle<FixedArray> array = isolate->factory()->NewFixedArray(length);
|
2011-02-02 15:08:29 +00:00
|
|
|
|
|
|
|
// 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;
|
2009-06-24 08:01:38 +00:00
|
|
|
}
|
2010-12-07 11:31:57 +00:00
|
|
|
functions.Rewind(0);
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// No frame corresponding to the given function found. Return null.
|
2011-03-18 20:35:07 +00:00
|
|
|
return isolate->heap()->null_value();
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
const AccessorDescriptor Accessors::FunctionArguments = {
|
|
|
|
FunctionGetArguments,
|
|
|
|
ReadOnlySetAccessor,
|
|
|
|
0
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
//
|
|
|
|
// Accessors::FunctionCaller
|
|
|
|
//
|
|
|
|
|
|
|
|
|
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)
|
2011-08-25 13:38:58 +00:00
|
|
|
: frame_iterator_(isolate),
|
|
|
|
functions_(2),
|
|
|
|
index_(0) {
|
|
|
|
GetFunctions();
|
|
|
|
}
|
|
|
|
JSFunction* next() {
|
|
|
|
if (functions_.length() == 0) return NULL;
|
|
|
|
JSFunction* next_function = functions_[index_];
|
|
|
|
index_--;
|
|
|
|
if (index_ < 0) {
|
|
|
|
GetFunctions();
|
|
|
|
}
|
|
|
|
return next_function;
|
|
|
|
}
|
|
|
|
|
|
|
|
// 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;
|
|
|
|
}
|
|
|
|
JavaScriptFrameIterator frame_iterator_;
|
|
|
|
List<JSFunction*> functions_;
|
|
|
|
int index_;
|
|
|
|
};
|
|
|
|
|
|
|
|
|
2013-09-02 09:25:20 +00:00
|
|
|
MaybeObject* Accessors::FunctionGetCaller(Isolate* isolate,
|
|
|
|
Object* object,
|
|
|
|
void*) {
|
2011-03-18 20:35:07 +00:00
|
|
|
HandleScope scope(isolate);
|
2013-06-03 15:32:22 +00:00
|
|
|
DisallowHeapAllocation no_allocation;
|
2013-02-27 13:22:29 +00:00
|
|
|
JSFunction* holder = FindInstanceOf<JSFunction>(isolate, object);
|
2012-10-08 12:58:46 +00:00
|
|
|
if (holder == NULL) return isolate->heap()->undefined_value();
|
2011-09-06 14:03:32 +00:00
|
|
|
if (holder->shared()->native()) return isolate->heap()->null_value();
|
2011-03-18 20:35:07 +00:00
|
|
|
Handle<JSFunction> function(holder, isolate);
|
2008-07-03 15:10:15 +00:00
|
|
|
|
2013-06-03 15:32:22 +00:00
|
|
|
FrameFunctionIterator it(isolate, no_allocation);
|
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.
|
|
|
|
return isolate->heap()->null_value();
|
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();
|
|
|
|
if (caller == NULL) return isolate->heap()->null_value();
|
|
|
|
} 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()) {
|
|
|
|
return isolate->heap()->null_value();
|
|
|
|
}
|
2012-11-23 15:47:58 +00:00
|
|
|
// Censor if the caller is not a classic mode function.
|
|
|
|
// Change from ES5, which used to throw, see:
|
|
|
|
// https://bugs.ecmascript.org/show_bug.cgi?id=310
|
|
|
|
if (!caller->shared()->is_classic_mode()) {
|
|
|
|
return isolate->heap()->null_value();
|
|
|
|
}
|
|
|
|
|
|
|
|
return caller;
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
const AccessorDescriptor Accessors::FunctionCaller = {
|
|
|
|
FunctionGetCaller,
|
|
|
|
ReadOnlySetAccessor,
|
|
|
|
0
|
|
|
|
};
|
|
|
|
|
|
|
|
|
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
|