Commit 3ec70cab authored by antonm@chromium.org's avatar antonm@chromium.org

Port caching of lookup followups for interceptors to ARM

Review URL: http://codereview.chromium.org/551191

git-svn-id: http://v8.googlecode.com/svn/branches/bleeding_edge@3766 ce2b1a6d-e550-0410-aec6-3dcde31c8c00
parent 7556a16c
This diff is collapsed.
...@@ -161,6 +161,7 @@ static void PushInterceptorArguments(MacroAssembler* masm, ...@@ -161,6 +161,7 @@ static void PushInterceptorArguments(MacroAssembler* masm,
__ push(holder); __ push(holder);
__ push(name); __ push(name);
InterceptorInfo* interceptor = holder_obj->GetNamedInterceptor(); InterceptorInfo* interceptor = holder_obj->GetNamedInterceptor();
ASSERT(!Heap::InNewSpace(interceptor));
__ mov(receiver, Immediate(Handle<Object>(interceptor))); __ mov(receiver, Immediate(Handle<Object>(interceptor)));
__ push(receiver); __ push(receiver);
__ push(FieldOperand(receiver, InterceptorInfo::kDataOffset)); __ push(FieldOperand(receiver, InterceptorInfo::kDataOffset));
...@@ -347,19 +348,6 @@ static void CompileLoadInterceptor(Compiler* compiler, ...@@ -347,19 +348,6 @@ static void CompileLoadInterceptor(Compiler* compiler,
} }
static void LookupPostInterceptor(JSObject* holder,
String* name,
LookupResult* lookup) {
holder->LocalLookupRealNamedProperty(name, lookup);
if (lookup->IsNotFound()) {
Object* proto = holder->GetPrototype();
if (proto != Heap::null_value()) {
proto->Lookup(name, lookup);
}
}
}
class LoadInterceptorCompiler BASE_EMBEDDED { class LoadInterceptorCompiler BASE_EMBEDDED {
public: public:
explicit LoadInterceptorCompiler(Register name) : name_(name) {} explicit LoadInterceptorCompiler(Register name) : name_(name) {}
...@@ -559,7 +547,6 @@ class CallInterceptorCompiler BASE_EMBEDDED { ...@@ -559,7 +547,6 @@ class CallInterceptorCompiler BASE_EMBEDDED {
__ mov(esi, FieldOperand(edi, JSFunction::kContextOffset)); __ mov(esi, FieldOperand(edi, JSFunction::kContextOffset));
// Jump to the cached code (tail call). // Jump to the cached code (tail call).
ASSERT(function->is_compiled());
Handle<Code> code(function->code()); Handle<Code> code(function->code());
ParameterCount expected(function->shared()->formal_parameter_count()); ParameterCount expected(function->shared()->formal_parameter_count());
__ InvokeCode(code, expected, arguments_, __ InvokeCode(code, expected, arguments_,
......
...@@ -1058,6 +1058,19 @@ Object* StubCompiler::GetCodeWithFlags(Code::Flags flags, String* name) { ...@@ -1058,6 +1058,19 @@ Object* StubCompiler::GetCodeWithFlags(Code::Flags flags, String* name) {
return GetCodeWithFlags(flags, reinterpret_cast<char*>(NULL)); return GetCodeWithFlags(flags, reinterpret_cast<char*>(NULL));
} }
void StubCompiler::LookupPostInterceptor(JSObject* holder,
String* name,
LookupResult* lookup) {
holder->LocalLookupRealNamedProperty(name, lookup);
if (lookup->IsNotFound()) {
Object* proto = holder->GetPrototype();
if (proto != Heap::null_value()) {
proto->Lookup(name, lookup);
}
}
}
Object* LoadStubCompiler::GetCode(PropertyType type, String* name) { Object* LoadStubCompiler::GetCode(PropertyType type, String* name) {
Code::Flags flags = Code::ComputeMonomorphicFlags(Code::LOAD_IC, type); Code::Flags flags = Code::ComputeMonomorphicFlags(Code::LOAD_IC, type);
......
...@@ -435,6 +435,10 @@ class StubCompiler BASE_EMBEDDED { ...@@ -435,6 +435,10 @@ class StubCompiler BASE_EMBEDDED {
String* name, String* name,
Label* miss); Label* miss);
static void LookupPostInterceptor(JSObject* holder,
String* name,
LookupResult* lookup);
private: private:
HandleScope scope_; HandleScope scope_;
MacroAssembler masm_; MacroAssembler masm_;
......
...@@ -143,6 +143,7 @@ static void PushInterceptorArguments(MacroAssembler* masm, ...@@ -143,6 +143,7 @@ static void PushInterceptorArguments(MacroAssembler* masm,
__ push(holder); __ push(holder);
__ push(name); __ push(name);
InterceptorInfo* interceptor = holder_obj->GetNamedInterceptor(); InterceptorInfo* interceptor = holder_obj->GetNamedInterceptor();
ASSERT(!Heap::InNewSpace(interceptor));
__ movq(kScratchRegister, Handle<Object>(interceptor), __ movq(kScratchRegister, Handle<Object>(interceptor),
RelocInfo::EMBEDDED_OBJECT); RelocInfo::EMBEDDED_OBJECT);
__ push(kScratchRegister); __ push(kScratchRegister);
...@@ -370,15 +371,47 @@ void StubCompiler::GenerateLoadFunctionPrototype(MacroAssembler* masm, ...@@ -370,15 +371,47 @@ void StubCompiler::GenerateLoadFunctionPrototype(MacroAssembler* masm,
} }
static void LookupPostInterceptor(JSObject* holder, template <class Compiler>
String* name, static void CompileLoadInterceptor(Compiler* compiler,
LookupResult* lookup) { StubCompiler* stub_compiler,
holder->LocalLookupRealNamedProperty(name, lookup); MacroAssembler* masm,
if (lookup->IsNotFound()) { JSObject* object,
Object* proto = holder->GetPrototype(); JSObject* holder,
if (proto != Heap::null_value()) { String* name,
proto->Lookup(name, lookup); LookupResult* lookup,
} Register receiver,
Register scratch1,
Register scratch2,
Label* miss) {
ASSERT(holder->HasNamedInterceptor());
ASSERT(!holder->GetNamedInterceptor()->getter()->IsUndefined());
// Check that the receiver isn't a smi.
__ JumpIfSmi(receiver, miss);
// Check that the maps haven't changed.
Register reg =
stub_compiler->CheckPrototypes(object, receiver, holder,
scratch1, scratch2, name, miss);
if (lookup->IsValid() && lookup->IsCacheable()) {
compiler->CompileCacheable(masm,
stub_compiler,
receiver,
reg,
scratch1,
scratch2,
holder,
lookup,
name,
miss);
} else {
compiler->CompileRegular(masm,
receiver,
reg,
scratch2,
holder,
miss);
} }
} }
...@@ -518,51 +551,6 @@ class LoadInterceptorCompiler BASE_EMBEDDED { ...@@ -518,51 +551,6 @@ class LoadInterceptorCompiler BASE_EMBEDDED {
}; };
template <class Compiler>
static void CompileLoadInterceptor(Compiler* compiler,
StubCompiler* stub_compiler,
MacroAssembler* masm,
JSObject* object,
JSObject* holder,
String* name,
LookupResult* lookup,
Register receiver,
Register scratch1,
Register scratch2,
Label* miss) {
ASSERT(holder->HasNamedInterceptor());
ASSERT(!holder->GetNamedInterceptor()->getter()->IsUndefined());
// Check that the receiver isn't a smi.
__ JumpIfSmi(receiver, miss);
// Check that the maps haven't changed.
Register reg =
stub_compiler->CheckPrototypes(object, receiver, holder,
scratch1, scratch2, name, miss);
if (lookup->IsValid() && lookup->IsCacheable()) {
compiler->CompileCacheable(masm,
stub_compiler,
receiver,
reg,
scratch1,
scratch2,
holder,
lookup,
name,
miss);
} else {
compiler->CompileRegular(masm,
receiver,
reg,
scratch2,
holder,
miss);
}
}
class CallInterceptorCompiler BASE_EMBEDDED { class CallInterceptorCompiler BASE_EMBEDDED {
public: public:
explicit CallInterceptorCompiler(const ParameterCount& arguments) explicit CallInterceptorCompiler(const ParameterCount& arguments)
...@@ -631,7 +619,6 @@ class CallInterceptorCompiler BASE_EMBEDDED { ...@@ -631,7 +619,6 @@ class CallInterceptorCompiler BASE_EMBEDDED {
__ movq(rsi, FieldOperand(rdi, JSFunction::kContextOffset)); __ movq(rsi, FieldOperand(rdi, JSFunction::kContextOffset));
// Jump to the cached code (tail call). // Jump to the cached code (tail call).
ASSERT(function->is_compiled());
Handle<Code> code(function->code()); Handle<Code> code(function->code());
ParameterCount expected(function->shared()->formal_parameter_count()); ParameterCount expected(function->shared()->formal_parameter_count());
__ InvokeCode(code, expected, arguments_, __ InvokeCode(code, expected, arguments_,
......
...@@ -5706,6 +5706,35 @@ THREADED_TEST(InterceptorCallICInvalidatedConstantFunctionViaGlobal) { ...@@ -5706,6 +5706,35 @@ THREADED_TEST(InterceptorCallICInvalidatedConstantFunctionViaGlobal) {
} }
// Test the case when actual function to call sits on global object.
THREADED_TEST(InterceptorCallICCachedFromGlobal) {
v8::HandleScope scope;
v8::Handle<v8::ObjectTemplate> templ_o = ObjectTemplate::New();
templ_o->SetNamedPropertyHandler(NoBlockGetterX);
LocalContext context;
context->Global()->Set(v8_str("o"), templ_o->NewInstance());
v8::Handle<Value> value = CompileRun(
"try {"
" o.__proto__ = this;"
" for (var i = 0; i < 10; i++) {"
" var v = o.parseFloat('239');"
" if (v != 239) throw v;"
// Now it should be ICed and keep a reference to parseFloat.
" }"
" var result = 0;"
" for (var i = 0; i < 10; i++) {"
" result += o.parseFloat('239');"
" }"
" result"
"} catch(e) {"
" e"
"};");
CHECK_EQ(239 * 10, value->Int32Value());
}
static int interceptor_call_count = 0; static int interceptor_call_count = 0;
static v8::Handle<Value> InterceptorICRefErrorGetter(Local<String> name, static v8::Handle<Value> InterceptorICRefErrorGetter(Local<String> name,
......
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