Chromium Code Reviews| Index: src/objects.cc |
| diff --git a/src/objects.cc b/src/objects.cc |
| index 7d00a561c549eb06eed1ba95f3e78ba180cbff47..8c769d1f1e7e071fcab9021beb4838efee9afc2f 100644 |
| --- a/src/objects.cc |
| +++ b/src/objects.cc |
| @@ -163,6 +163,152 @@ 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, |
| + Isolate* isolate) { |
|
Sven Panne
2013/03/07 08:18:48
Passing just the Heap, not the Isolate in its full
|
| + 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; |
| + } |
| + bool result = (bitmask & value) == (bitmask & compare_value); |
| + return result ? isolate->heap()->true_value() |
|
Sven Panne
2013/03/07 08:18:48
return isolate->heap()->ToBoolean(result);
|
| + : isolate->heap()->false_value(); |
| +} |
| + |
| + |
| +static MaybeObject* PerformCompare(const PointerCompareDescriptor& descriptor, |
| + char* ptr, |
| + Isolate* isolate) { |
| + uintptr_t compare_value = |
| + reinterpret_cast<uintptr_t>(descriptor.compare_value); |
| + uintptr_t value = *CheckedCast<uintptr_t>(ptr); |
| + bool result = compare_value == value; |
| + return result ? isolate->heap()->true_value() |
|
Sven Panne
2013/03/07 08:18:48
See above
|
| + : isolate->heap()->false_value(); |
| +} |
| + |
| + |
| +static MaybeObject* GetPrimitiveValue( |
| + const PrimitiveValueDescriptor& descriptor, |
| + char* ptr, |
| + Isolate* isolate) { |
| + 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 isolate->heap()->NumberFromUint32(value); |
| + } |
| + case kDescriptorBoolType: { |
| + uint8_t byte = *CheckedCast<uint8_t>(ptr); |
| + bool result = byte & (0x1 << descriptor.bool_offset); |
| + return result ? isolate->heap()->true_value() |
|
Sven Panne
2013/03/07 08:18:48
See above
|
| + : isolate->heap()->false_value(); |
| + } |
| + case kDescriptorFloatType: { |
| + float value = *CheckedCast<float>(ptr); |
| + return isolate->heap()->NumberFromDouble(value); |
| + } |
| + case kDescriptorDoubleType: { |
| + double value = *CheckedCast<double>(ptr); |
| + return isolate->heap()->NumberFromDouble(value); |
| + } |
| + } |
| + return isolate->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); |
| + case kDescriptorPointerCompare: |
| + ASSERT(iterator.Complete()); |
| + return PerformCompare(data->pointer_compare_descriptor, |
| + current, |
| + isolate); |
| + case kDescriptorPrimitiveValue: |
| + ASSERT(iterator.Complete()); |
| + return GetPrimitiveValue(data->primitive_value_descriptor, |
| + current, |
| + isolate); |
| + } |
| + } |
| + UNREACHABLE(); |
| + return NULL; |
| +} |
| + |
| + |
| MaybeObject* JSObject::GetPropertyWithCallback(Object* receiver, |
| Object* structure, |
| String* name) { |
| @@ -180,9 +326,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); |
| Handle<Object> receiver_handle(receiver); |
| Handle<Object> args[2] = { name_handle, receiver_handle }; |
| @@ -192,6 +337,12 @@ MaybeObject* JSObject::GetPropertyWithCallback(Object* receiver, |
| ARRAY_SIZE(args))); |
| return isolate->Throw(*error); |
| } |
| + 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(); |
| @@ -227,11 +378,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; |
| } |
| @@ -9783,8 +9929,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(); |
| @@ -13547,6 +13694,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) { |