Added Crankshaft support for JavaScript getters.

Instead of calling the getter via a stub, we now call it more directly via a
(guarded) CallConstantFunction instruction.

Review URL: https://chromiumcodereview.appspot.com/10735023

git-svn-id: http://v8.googlecode.com/svn/branches/bleeding_edge@12013 ce2b1a6d-e550-0410-aec6-3dcde31c8c00
parent b6ebc22d
...@@ -5623,6 +5623,33 @@ HInstruction* HGraphBuilder::BuildLoadNamedGeneric(HValue* obj, ...@@ -5623,6 +5623,33 @@ HInstruction* HGraphBuilder::BuildLoadNamedGeneric(HValue* obj,
} }
static void LookupInPrototypes(Handle<Map> map,
Handle<String> name,
LookupResult* lookup) {
while (map->prototype()->IsJSObject()) {
Handle<JSObject> holder(JSObject::cast(map->prototype()));
map = Handle<Map>(holder->map());
map->LookupDescriptor(*holder, *name, lookup);
if (lookup->IsFound()) return;
}
lookup->NotFound();
}
HInstruction* HGraphBuilder::BuildCallGetter(HValue* obj,
Property* expr,
Handle<Map> map,
Handle<Object> callback,
Handle<JSObject> holder) {
if (!callback->IsAccessorPair()) return BuildLoadNamedGeneric(obj, expr);
Handle<Object> getter(Handle<AccessorPair>::cast(callback)->getter());
Handle<JSFunction> function(Handle<JSFunction>::cast(getter));
AddCheckConstantFunction(holder, obj, map, true);
AddInstruction(new(zone()) HPushArgument(obj));
return new(zone()) HCallConstantFunction(function, 1);
}
HInstruction* HGraphBuilder::BuildLoadNamed(HValue* obj, HInstruction* HGraphBuilder::BuildLoadNamed(HValue* obj,
Property* expr, Property* expr,
Handle<Map> map, Handle<Map> map,
...@@ -5640,7 +5667,17 @@ HInstruction* HGraphBuilder::BuildLoadNamed(HValue* obj, ...@@ -5640,7 +5667,17 @@ HInstruction* HGraphBuilder::BuildLoadNamed(HValue* obj,
AddInstruction(HCheckMaps::NewWithTransitions(obj, map, zone())); AddInstruction(HCheckMaps::NewWithTransitions(obj, map, zone()));
Handle<JSFunction> function(lookup.GetConstantFunctionFromMap(*map)); Handle<JSFunction> function(lookup.GetConstantFunctionFromMap(*map));
return new(zone()) HConstant(function, Representation::Tagged()); return new(zone()) HConstant(function, Representation::Tagged());
} else if (lookup.IsPropertyCallbacks()) {
Handle<Object> callback(lookup.GetValueFromMap(*map));
Handle<JSObject> holder;
return BuildCallGetter(obj, expr, map, callback, holder);
} else { } else {
LookupInPrototypes(map, name, &lookup);
if (lookup.IsPropertyCallbacks()) {
Handle<Object> callback(lookup.GetValue());
Handle<JSObject> holder(lookup.holder());
return BuildCallGetter(obj, expr, map, callback, holder);
}
return BuildLoadNamedGeneric(obj, expr); return BuildLoadNamedGeneric(obj, expr);
} }
} }
......
...@@ -1138,6 +1138,11 @@ class HGraphBuilder: public AstVisitor { ...@@ -1138,6 +1138,11 @@ class HGraphBuilder: public AstVisitor {
bool is_store, bool is_store,
bool* has_side_effects); bool* has_side_effects);
HInstruction* BuildCallGetter(HValue* obj,
Property* expr,
Handle<Map> map,
Handle<Object> callback,
Handle<JSObject> holder);
HInstruction* BuildLoadNamed(HValue* object, HInstruction* BuildLoadNamed(HValue* object,
Property* prop, Property* prop,
Handle<Map> map, Handle<Map> map,
......
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