Commit 4b928a49 authored by ager@chromium.org's avatar ager@chromium.org

Inline the fast path for handle creation.

Simplify the interceptor IC code by using raw pointers instead of handles.
Review URL: http://codereview.chromium.org/49001

git-svn-id: http://v8.googlecode.com/svn/branches/bleeding_edge@1539 ce2b1a6d-e550-0410-aec6-3dcde31c8c00
parent 2d4dd93b
...@@ -52,9 +52,10 @@ int HandleScope::NumberOfHandles() { ...@@ -52,9 +52,10 @@ int HandleScope::NumberOfHandles() {
} }
void** HandleScope::CreateHandle(void* value) { void** HandleScope::Extend() {
void** result = current_.next; void** result = current_.next;
if (result == current_.limit) {
ASSERT(result == current_.limit);
// Make sure there's at least one scope on the stack and that the // Make sure there's at least one scope on the stack and that the
// top of the scope stack isn't a barrier. // top of the scope stack isn't a barrier.
if (current_.extensions < 0) { if (current_.extensions < 0) {
...@@ -83,13 +84,7 @@ void** HandleScope::CreateHandle(void* value) { ...@@ -83,13 +84,7 @@ void** HandleScope::CreateHandle(void* value) {
current_.extensions++; current_.extensions++;
current_.limit = &result[kHandleBlockSize]; current_.limit = &result[kHandleBlockSize];
} }
}
// Update the current next field, set the value in the created
// handle, and return the result.
ASSERT(result < current_.limit);
current_.next = result + 1;
*result = value;
return result; return result;
} }
......
...@@ -118,7 +118,16 @@ class HandleScope { ...@@ -118,7 +118,16 @@ class HandleScope {
static int NumberOfHandles(); static int NumberOfHandles();
// Creates a new handle with the given value. // Creates a new handle with the given value.
static void** CreateHandle(void* value); static inline void** CreateHandle(void* value) {
void** result = current_.next;
if (result == current_.limit) result = Extend();
// Update the current next field, set the value in the created
// handle, and return the result.
ASSERT(result < current_.limit);
current_.next = result + 1;
*result = value;
return result;
}
private: private:
// Prevent heap allocation or illegal handle scopes. // Prevent heap allocation or illegal handle scopes.
...@@ -150,6 +159,9 @@ class HandleScope { ...@@ -150,6 +159,9 @@ class HandleScope {
#endif #endif
} }
// Extend the handle scope making room for more handles.
static void** Extend();
// Deallocates any extensions used by the current scope. // Deallocates any extensions used by the current scope.
static void DeleteExtensions(); static void DeleteExtensions();
......
...@@ -656,9 +656,8 @@ Handle<Code> ComputeCallMiss(int argc) { ...@@ -656,9 +656,8 @@ Handle<Code> ComputeCallMiss(int argc) {
Object* LoadCallbackProperty(Arguments args) { Object* LoadCallbackProperty(Arguments args) {
Handle<JSObject> recv = args.at<JSObject>(0); Handle<JSObject> recv = args.at<JSObject>(0);
AccessorInfo* callback = AccessorInfo::cast(args[1]); AccessorInfo* callback = AccessorInfo::cast(args[1]);
v8::AccessorGetter fun = Address getter_address = v8::ToCData<Address>(callback->getter());
FUNCTION_CAST<v8::AccessorGetter>( v8::AccessorGetter fun = FUNCTION_CAST<v8::AccessorGetter>(getter_address);
v8::ToCData<Address>(callback->getter()));
ASSERT(fun != NULL); ASSERT(fun != NULL);
Handle<String> name = args.at<String>(2); Handle<String> name = args.at<String>(2);
Handle<JSObject> holder = args.at<JSObject>(3); Handle<JSObject> holder = args.at<JSObject>(3);
...@@ -669,8 +668,7 @@ Object* LoadCallbackProperty(Arguments args) { ...@@ -669,8 +668,7 @@ Object* LoadCallbackProperty(Arguments args) {
// locations of the arguments to this function maybe we don't have // locations of the arguments to this function maybe we don't have
// to explicitly create the structure but can just pass a pointer // to explicitly create the structure but can just pass a pointer
// into the stack. // into the stack.
v8::AccessorInfo info( v8::AccessorInfo info(v8::Utils::ToLocal(recv),
v8::Utils::ToLocal(recv),
v8::Utils::ToLocal(data), v8::Utils::ToLocal(data),
v8::Utils::ToLocal(holder)); v8::Utils::ToLocal(holder));
v8::Handle<v8::Value> result; v8::Handle<v8::Value> result;
...@@ -680,28 +678,23 @@ Object* LoadCallbackProperty(Arguments args) { ...@@ -680,28 +678,23 @@ Object* LoadCallbackProperty(Arguments args) {
result = fun(v8::Utils::ToLocal(name), info); result = fun(v8::Utils::ToLocal(name), info);
} }
RETURN_IF_SCHEDULED_EXCEPTION(); RETURN_IF_SCHEDULED_EXCEPTION();
if (result.IsEmpty()) { if (result.IsEmpty()) return Heap::undefined_value();
return Heap::undefined_value();
} else {
return *v8::Utils::OpenHandle(*result); return *v8::Utils::OpenHandle(*result);
}
} }
Object* StoreCallbackProperty(Arguments args) { Object* StoreCallbackProperty(Arguments args) {
Handle<JSObject> recv = args.at<JSObject>(0); Handle<JSObject> recv = args.at<JSObject>(0);
AccessorInfo* callback = AccessorInfo::cast(args[1]); AccessorInfo* callback = AccessorInfo::cast(args[1]);
v8::AccessorSetter fun = Address setter_address = v8::ToCData<Address>(callback->setter());
FUNCTION_CAST<v8::AccessorSetter>( v8::AccessorSetter fun = FUNCTION_CAST<v8::AccessorSetter>(setter_address);
v8::ToCData<Address>(callback->setter()));
ASSERT(fun != NULL); ASSERT(fun != NULL);
Handle<String> name = args.at<String>(2); Handle<String> name = args.at<String>(2);
Handle<Object> value = args.at<Object>(3); Handle<Object> value = args.at<Object>(3);
HandleScope scope; HandleScope scope;
Handle<Object> data(callback->data()); Handle<Object> data(callback->data());
LOG(ApiNamedPropertyAccess("store", *recv, *name)); LOG(ApiNamedPropertyAccess("store", *recv, *name));
v8::AccessorInfo info( v8::AccessorInfo info(v8::Utils::ToLocal(recv),
v8::Utils::ToLocal(recv),
v8::Utils::ToLocal(data), v8::Utils::ToLocal(data),
v8::Utils::ToLocal(recv)); v8::Utils::ToLocal(recv));
{ {
...@@ -715,28 +708,17 @@ Object* StoreCallbackProperty(Arguments args) { ...@@ -715,28 +708,17 @@ Object* StoreCallbackProperty(Arguments args) {
Object* LoadInterceptorProperty(Arguments args) { Object* LoadInterceptorProperty(Arguments args) {
HandleScope scope; JSObject* recv = JSObject::cast(args[0]);
Handle<JSObject> recv = args.at<JSObject>(0); JSObject* holder = JSObject::cast(args[1]);
Handle<JSObject> holder = args.at<JSObject>(1); String* name = String::cast(args[2]);
Handle<String> name = args.at<String>(2);
ASSERT(holder->HasNamedInterceptor()); ASSERT(holder->HasNamedInterceptor());
PropertyAttributes attr = NONE; PropertyAttributes attr = NONE;
Handle<Object> result = GetPropertyWithInterceptor(recv, holder, name, &attr); Object* result = holder->GetPropertyWithInterceptor(recv, name, &attr);
// GetPropertyWithInterceptor already converts a scheduled exception if (result->IsFailure()) return result;
// to a pending one if any. Don't use RETURN_IF_SCHEDULED_EXCEPTION() here.
// Make sure to propagate exceptions.
if (result.is_null()) {
// Failure::Exception is converted to a null handle in the
// handle-based methods such as SetProperty. We therefore need
// to convert null handles back to exceptions.
ASSERT(Top::has_pending_exception());
return Failure::Exception();
}
// If the property is present, return it. // If the property is present, return it.
if (attr != ABSENT) return *result; if (attr != ABSENT) return result;
// If the top frame is an internal frame, this is really a call // If the top frame is an internal frame, this is really a call
// IC. In this case, we simply return the undefined result which // IC. In this case, we simply return the undefined result which
...@@ -744,43 +726,38 @@ Object* LoadInterceptorProperty(Arguments args) { ...@@ -744,43 +726,38 @@ Object* LoadInterceptorProperty(Arguments args) {
// function. // function.
StackFrameIterator it; StackFrameIterator it;
it.Advance(); // skip exit frame it.Advance(); // skip exit frame
if (it.frame()->is_internal()) return *result; if (it.frame()->is_internal()) return result;
// If the load is non-contextual, just return the undefined result. // If the load is non-contextual, just return the undefined result.
// Note that both keyed and non-keyed loads may end up here, so we // Note that both keyed and non-keyed loads may end up here, so we
// can't use either LoadIC or KeyedLoadIC constructors. // can't use either LoadIC or KeyedLoadIC constructors.
IC ic(IC::NO_EXTRA_FRAME); IC ic(IC::NO_EXTRA_FRAME);
ASSERT(ic.target()->is_load_stub() || ic.target()->is_keyed_load_stub()); ASSERT(ic.target()->is_load_stub() || ic.target()->is_keyed_load_stub());
if (!ic.is_contextual()) return *result; if (!ic.is_contextual()) return result;
// Throw a reference error. // Throw a reference error.
{
HandleScope scope;
// We cannot use the raw name pointer here since getting the
// property might cause a GC. However, we can get the name from
// the stack using the arguments object.
Handle<String> name_handle = args.at<String>(2);
Handle<Object> error = Handle<Object> error =
Factory::NewReferenceError("not_defined", HandleVector(&name, 1)); Factory::NewReferenceError("not_defined",
HandleVector(&name_handle, 1));
return Top::Throw(*error); return Top::Throw(*error);
}
} }
Object* StoreInterceptorProperty(Arguments args) { Object* StoreInterceptorProperty(Arguments args) {
HandleScope scope; JSObject* recv = JSObject::cast(args[0]);
Handle<JSObject> recv = args.at<JSObject>(0); String* name = String::cast(args[1]);
Handle<String> name = args.at<String>(1); Object* value = args[2];
Handle<Object> value = args.at<Object>(2);
ASSERT(recv->HasNamedInterceptor()); ASSERT(recv->HasNamedInterceptor());
PropertyAttributes attr = NONE; PropertyAttributes attr = NONE;
Handle<Object> result = SetPropertyWithInterceptor(recv, name, value, attr); Object* result = recv->SetPropertyWithInterceptor(name, value, attr);
return result;
// SetPropertyWithInterceptor already converts a scheduled exception
// to a pending one if any. Don't use RETURN_IF_SCHEDULED_EXCEPTION() here.
// Make sure to propagate exceptions.
if (result.is_null()) {
// Failure::Exception is converted to a null handle in the
// handle-based methods such as SetProperty. We therefore need
// to convert null handles back to exceptions.
ASSERT(Top::has_pending_exception());
return Failure::Exception();
}
return *result;
} }
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment