Index: src/objects.cc |
diff --git a/src/objects.cc b/src/objects.cc |
index 381961be9a0dbc57a47e96a592e89aeb3ef18c68..00a81756f05781702b409d5468017f6454b95374 100644 |
--- a/src/objects.cc |
+++ b/src/objects.cc |
@@ -165,6 +165,146 @@ MaybeObject* Object::GetPropertyWithReceiver(Object* receiver, |
} |
+template<typename To> |
+static inline To* CheckedCast(void *from) { |
+ uintptr_t temp = reinterpret_cast<uintptr_t>(from); |
+ ASSERT(temp % sizeof(To) == 0); |
+ return reinterpret_cast<To*>(temp); |
+} |
+ |
+ |
+static MaybeObject* PerformCompare(const BitmaskCompareDescriptor& descriptor, |
+ char* ptr, |
+ Heap* heap) { |
+ uint32_t bitmask = descriptor.bitmask; |
+ uint32_t compare_value = descriptor.compare_value; |
+ uint32_t value; |
+ switch (descriptor.size) { |
+ case 1: |
+ value = static_cast<uint32_t>(*CheckedCast<uint8_t>(ptr)); |
+ compare_value &= 0xff; |
+ bitmask &= 0xff; |
+ break; |
+ case 2: |
+ value = static_cast<uint32_t>(*CheckedCast<uint16_t>(ptr)); |
+ compare_value &= 0xffff; |
+ bitmask &= 0xffff; |
+ break; |
+ case 4: |
+ value = *CheckedCast<uint32_t>(ptr); |
+ break; |
+ default: |
+ UNREACHABLE(); |
+ return NULL; |
+ } |
+ return heap->ToBoolean((bitmask & value) == (bitmask & compare_value)); |
+} |
+ |
+ |
+static MaybeObject* PerformCompare(const PointerCompareDescriptor& descriptor, |
+ char* ptr, |
+ Heap* heap) { |
+ uintptr_t compare_value = |
+ reinterpret_cast<uintptr_t>(descriptor.compare_value); |
+ uintptr_t value = *CheckedCast<uintptr_t>(ptr); |
+ return heap->ToBoolean(compare_value == value); |
+} |
+ |
+ |
+static MaybeObject* GetPrimitiveValue( |
+ const PrimitiveValueDescriptor& descriptor, |
+ char* ptr, |
+ Heap* heap) { |
+ int32_t int32_value; |
+ switch (descriptor.data_type) { |
+ case kDescriptorInt8Type: |
+ int32_value = *CheckedCast<int8_t>(ptr); |
+ break; |
+ case kDescriptorUint8Type: |
+ int32_value = *CheckedCast<uint8_t>(ptr); |
+ break; |
+ case kDescriptorInt16Type: |
+ int32_value = *CheckedCast<int16_t>(ptr); |
+ break; |
+ case kDescriptorUint16Type: |
+ int32_value = *CheckedCast<uint16_t>(ptr); |
+ break; |
+ case kDescriptorInt32Type: |
+ int32_value = *CheckedCast<int32_t>(ptr); |
+ break; |
+ case kDescriptorUint32Type: { |
+ uint32_t value = *CheckedCast<uint32_t>(ptr); |
+ return heap->NumberFromUint32(value); |
+ } |
+ case kDescriptorBoolType: { |
+ uint8_t byte = *CheckedCast<uint8_t>(ptr); |
+ return heap->ToBoolean(byte & (0x1 << descriptor.bool_offset)); |
+ } |
+ case kDescriptorFloatType: { |
+ float value = *CheckedCast<float>(ptr); |
+ return heap->NumberFromDouble(value); |
+ } |
+ case kDescriptorDoubleType: { |
+ double value = *CheckedCast<double>(ptr); |
+ return heap->NumberFromDouble(value); |
+ } |
+ } |
+ return heap->NumberFromInt32(int32_value); |
+} |
+ |
+ |
+static MaybeObject* GetDeclaredAccessorProperty(Object* receiver, |
+ DeclaredAccessorInfo* info, |
+ Isolate* isolate) { |
+ char* current = reinterpret_cast<char*>(receiver); |
+ DeclaredAccessorDescriptorIterator iterator(info->descriptor()); |
+ while (true) { |
+ const DeclaredAccessorDescriptorData* data = iterator.Next(); |
+ switch (data->type) { |
+ case kDescriptorReturnObject: { |
+ ASSERT(iterator.Complete()); |
+ current = *CheckedCast<char*>(current); |
+ return *CheckedCast<Object*>(current); |
+ } |
+ case kDescriptorPointerDereference: |
+ ASSERT(!iterator.Complete()); |
+ current = *reinterpret_cast<char**>(current); |
+ break; |
+ case kDescriptorPointerShift: |
+ ASSERT(!iterator.Complete()); |
+ current += data->pointer_shift_descriptor.byte_offset; |
+ break; |
+ case kDescriptorObjectDereference: { |
+ ASSERT(!iterator.Complete()); |
+ Object* object = CheckedCast<Object>(current); |
+ int field = data->object_dereference_descriptor.internal_field; |
+ Object* smi = JSObject::cast(object)->GetInternalField(field); |
+ ASSERT(smi->IsSmi()); |
+ current = reinterpret_cast<char*>(smi); |
+ break; |
+ } |
+ case kDescriptorBitmaskCompare: |
+ ASSERT(iterator.Complete()); |
+ return PerformCompare(data->bitmask_compare_descriptor, |
+ current, |
+ isolate->heap()); |
+ case kDescriptorPointerCompare: |
+ ASSERT(iterator.Complete()); |
+ return PerformCompare(data->pointer_compare_descriptor, |
+ current, |
+ isolate->heap()); |
+ case kDescriptorPrimitiveValue: |
+ ASSERT(iterator.Complete()); |
+ return GetPrimitiveValue(data->primitive_value_descriptor, |
+ current, |
+ isolate->heap()); |
+ } |
+ } |
+ UNREACHABLE(); |
+ return NULL; |
+} |
+ |
+ |
MaybeObject* JSObject::GetPropertyWithCallback(Object* receiver, |
Object* structure, |
Name* name) { |
@@ -182,9 +322,8 @@ MaybeObject* JSObject::GetPropertyWithCallback(Object* receiver, |
} |
// api style callbacks. |
- if (structure->IsExecutableAccessorInfo()) { |
- ExecutableAccessorInfo* data = ExecutableAccessorInfo::cast(structure); |
- if (!data->IsCompatibleReceiver(receiver)) { |
+ if (structure->IsAccessorInfo()) { |
+ if (!AccessorInfo::cast(structure)->IsCompatibleReceiver(receiver)) { |
Handle<Object> name_handle(name, isolate); |
Handle<Object> receiver_handle(receiver, isolate); |
Handle<Object> args[2] = { name_handle, receiver_handle }; |
@@ -197,6 +336,12 @@ MaybeObject* JSObject::GetPropertyWithCallback(Object* receiver, |
// TODO(rossberg): Handling symbols in the API requires changing the API, |
// so we do not support it for now. |
if (name->IsSymbol()) return isolate->heap()->undefined_value(); |
+ if (structure->IsDeclaredAccessorInfo()) { |
+ return GetDeclaredAccessorProperty(receiver, |
+ DeclaredAccessorInfo::cast(structure), |
+ isolate); |
+ } |
+ ExecutableAccessorInfo* data = ExecutableAccessorInfo::cast(structure); |
Object* fun_obj = data->getter(); |
v8::AccessorGetter call_fun = v8::ToCData<v8::AccessorGetter>(fun_obj); |
if (call_fun == NULL) return isolate->heap()->undefined_value(); |
@@ -232,11 +377,6 @@ MaybeObject* JSObject::GetPropertyWithCallback(Object* receiver, |
return isolate->heap()->undefined_value(); |
} |
- // TODO(dcarney): Handle correctly. |
- if (structure->IsDeclaredAccessorInfo()) { |
- return isolate->heap()->undefined_value(); |
- } |
- |
UNREACHABLE(); |
return NULL; |
} |
@@ -9941,8 +10081,9 @@ MaybeObject* JSObject::GetElementWithCallback(Object* receiver, |
} |
if (structure->IsDeclaredAccessorInfo()) { |
- // TODO(dcarney): Handle correctly. |
- return isolate->heap()->undefined_value(); |
+ return GetDeclaredAccessorProperty(receiver, |
+ DeclaredAccessorInfo::cast(structure), |
+ isolate); |
} |
UNREACHABLE(); |
@@ -13724,6 +13865,58 @@ void ObjectHashTable::RemoveEntry(int entry) { |
} |
+DeclaredAccessorDescriptorIterator::DeclaredAccessorDescriptorIterator( |
+ DeclaredAccessorDescriptor* descriptor) |
+ : array_(descriptor->serialized_data()->GetDataStartAddress()), |
+ length_(descriptor->serialized_data()->length()), |
+ offset_(0) { |
+} |
+ |
+ |
+const DeclaredAccessorDescriptorData* |
+ DeclaredAccessorDescriptorIterator::Next() { |
+ ASSERT(offset_ < length_); |
+ uint8_t* ptr = &array_[offset_]; |
+ ASSERT(reinterpret_cast<uintptr_t>(ptr) % sizeof(uintptr_t) == 0); |
+ const DeclaredAccessorDescriptorData* data = |
+ reinterpret_cast<const DeclaredAccessorDescriptorData*>(ptr); |
+ offset_ += sizeof(*data); |
+ ASSERT(offset_ <= length_); |
+ return data; |
+} |
+ |
+ |
+Handle<DeclaredAccessorDescriptor> DeclaredAccessorDescriptor::Create( |
+ Isolate* isolate, |
+ const DeclaredAccessorDescriptorData& descriptor, |
+ Handle<DeclaredAccessorDescriptor> previous) { |
+ int previous_length = |
+ previous.is_null() ? 0 : previous->serialized_data()->length(); |
+ int length = sizeof(descriptor) + previous_length; |
+ Handle<ByteArray> serialized_descriptor = |
+ isolate->factory()->NewByteArray(length); |
+ Handle<DeclaredAccessorDescriptor> value = |
+ isolate->factory()->NewDeclaredAccessorDescriptor(); |
+ value->set_serialized_data(*serialized_descriptor); |
+ // Copy in the data. |
+ { |
+ AssertNoAllocation no_allocation; |
+ uint8_t* array = serialized_descriptor->GetDataStartAddress(); |
+ if (previous_length != 0) { |
+ uint8_t* previous_array = |
+ previous->serialized_data()->GetDataStartAddress(); |
+ memcpy(array, previous_array, previous_length); |
+ array += previous_length; |
+ } |
+ ASSERT(reinterpret_cast<uintptr_t>(array) % sizeof(uintptr_t) == 0); |
+ DeclaredAccessorDescriptorData* data = |
+ reinterpret_cast<DeclaredAccessorDescriptorData*>(array); |
+ *data = descriptor; |
+ } |
+ return value; |
+} |
+ |
+ |
#ifdef ENABLE_DEBUGGER_SUPPORT |
// Check if there is a break point at this code position. |
bool DebugInfo::HasBreakPoint(int code_position) { |