Index: src/api.cc |
diff --git a/src/api.cc b/src/api.cc |
index 36248437c789f569d53e4a1cd3b1cd5ea7ae5404..073306f071a2a260029750d82317014e0067f84f 100644 |
--- a/src/api.cc |
+++ b/src/api.cc |
@@ -3266,18 +3266,35 @@ void v8::Object::SetInternalField(int index, v8::Handle<Value> value) { |
} |
+static bool CanBeEncodedAsSmi(void* ptr) { |
+ const intptr_t address = reinterpret_cast<intptr_t>(ptr); |
+ return ((address & i::kEncodablePointerMask) == 0); |
+} |
+ |
+ |
+static i::Smi* EncodeAsSmi(void* ptr) { |
+ ASSERT(CanBeEncodedAsSmi(ptr)); |
+ const intptr_t address = reinterpret_cast<intptr_t>(ptr); |
+ i::Smi* result = reinterpret_cast<i::Smi*>(address << i::kPointerToSmiShift); |
+ ASSERT(i::Internals::HasSmiTag(result)); |
+ ASSERT_EQ(result, i::Smi::FromInt(result->value())); |
+ ASSERT_EQ(ptr, i::Internals::GetExternalPointerFromSmi(result)); |
+ return result; |
+} |
+ |
+ |
void v8::Object::SetPointerInInternalField(int index, void* value) { |
ENTER_V8; |
- i::Object* as_object = reinterpret_cast<i::Object*>(value); |
- if (as_object->IsSmi()) { |
- Utils::OpenHandle(this)->SetInternalField(index, as_object); |
- return; |
+ if (CanBeEncodedAsSmi(value)) { |
+ Utils::OpenHandle(this)->SetInternalField(index, EncodeAsSmi(value)); |
+ } else { |
+ HandleScope scope; |
+ i::Handle<i::Proxy> proxy = |
+ i::Factory::NewProxy(reinterpret_cast<i::Address>(value), i::TENURED); |
+ if (!proxy.is_null()) |
+ Utils::OpenHandle(this)->SetInternalField(index, *proxy); |
} |
- HandleScope scope; |
- i::Handle<i::Proxy> proxy = |
- i::Factory::NewProxy(reinterpret_cast<i::Address>(value), i::TENURED); |
- if (!proxy.is_null()) |
- Utils::OpenHandle(this)->SetInternalField(index, *proxy); |
+ ASSERT_EQ(value, GetPointerFromInternalField(index)); |
} |
@@ -3562,11 +3579,13 @@ Local<Value> v8::External::Wrap(void* data) { |
LOG_API("External::Wrap"); |
EnsureInitialized("v8::External::Wrap()"); |
ENTER_V8; |
- i::Object* as_object = reinterpret_cast<i::Object*>(data); |
- if (as_object->IsSmi()) { |
- return Utils::ToLocal(i::Handle<i::Object>(as_object)); |
- } |
- return ExternalNewImpl(data); |
+ |
+ v8::Local<v8::Value> result = CanBeEncodedAsSmi(data) |
+ ? Utils::ToLocal(i::Handle<i::Object>(EncodeAsSmi(data))) |
+ : v8::Local<v8::Value>(ExternalNewImpl(data)); |
+ |
+ ASSERT_EQ(data, Unwrap(result)); |
+ return result; |
} |
@@ -3574,7 +3593,7 @@ void* v8::Object::SlowGetPointerFromInternalField(int index) { |
i::Handle<i::JSObject> obj = Utils::OpenHandle(this); |
i::Object* value = obj->GetInternalField(index); |
if (value->IsSmi()) { |
- return value; |
+ return i::Internals::GetExternalPointerFromSmi(value); |
} else if (value->IsProxy()) { |
return reinterpret_cast<void*>(i::Proxy::cast(value)->proxy()); |
} else { |
@@ -3588,8 +3607,7 @@ void* v8::External::FullUnwrap(v8::Handle<v8::Value> wrapper) { |
i::Handle<i::Object> obj = Utils::OpenHandle(*wrapper); |
void* result; |
if (obj->IsSmi()) { |
- // The external value was an aligned pointer. |
- result = *obj; |
+ result = i::Internals::GetExternalPointerFromSmi(*obj); |
} else if (obj->IsProxy()) { |
result = ExternalValueImpl(obj); |
} else { |