Chromium Code Reviews| Index: src/api.cc |
| diff --git a/src/api.cc b/src/api.cc |
| index 110468e2313ec75196b7c7604d6c2612ff09e922..8d89909273491b42c8cb00b6346191945e7553ec 100644 |
| --- a/src/api.cc |
| +++ b/src/api.cc |
| @@ -3266,18 +3266,34 @@ void v8::Object::SetInternalField(int index, v8::Handle<Value> value) { |
| } |
| +static bool CanBeEncodedAsSmi(void* ptr) { |
| + const intptr_t address = reinterpret_cast<intptr_t>(ptr); |
| + const intptr_t mask = (intptr_t(1) << i::kSmiTagSize) - 1; |
|
Lasse Reichstein
2011/01/13 08:35:18
Just use kSmiTagMask.
|
| + return ((address & mask) == 0) && i::Smi::IsValid(address >> i::kSmiTagSize); |
|
Lasse Reichstein
2011/01/13 08:35:18
Almost no addresses will be valid in 64-bit mode.
|
| +} |
| + |
| + |
| +static i::Smi* EncodeAsSmi(void* ptr) { |
| + ASSERT(CanBeEncodedAsSmi(ptr)); |
| + i::Smi* result = i::Smi::FromIntptr( |
| + reinterpret_cast<intptr_t>(ptr) >> i::kSmiTagSize); |
| + 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)); |
| } |
| @@ -3560,11 +3576,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; |
| } |
| @@ -3572,7 +3590,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 { |
| @@ -3586,8 +3604,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 { |