// Copyright 2011 the V8 project authors. All rights reserved. // Use of this source code is governed by a BSD-style license that can be // found in the LICENSE file. #include "src/codegen/compilation-cache.h" #include "src/codegen/script-details.h" #include "src/common/globals.h" #include "src/heap/factory.h" #include "src/logging/counters.h" #include "src/logging/log.h" #include "src/objects/compilation-cache-table-inl.h" #include "src/objects/objects-inl.h" #include "src/objects/slots.h" #include "src/objects/visitors.h" #include "src/utils/ostreams.h" namespace v8 { namespace internal { // The number of generations for each sub cache. static const int kRegExpGenerations = 2; // Initial size of each compilation cache table allocated. static const int kInitialCacheSize = 64; CompilationCache::CompilationCache(Isolate* isolate) : isolate_(isolate), script_(isolate), eval_global_(isolate), eval_contextual_(isolate), reg_exp_(isolate, kRegExpGenerations), enabled_script_and_eval_(true) { CompilationSubCache* subcaches[kSubCacheCount] = { &script_, &eval_global_, &eval_contextual_, ®_exp_}; for (int i = 0; i < kSubCacheCount; ++i) { subcaches_[i] = subcaches[i]; } } Handle<CompilationCacheTable> CompilationSubCache::GetTable(int generation) { DCHECK_LT(generation, generations()); Handle<CompilationCacheTable> result; if (tables_[generation].IsUndefined(isolate())) { result = CompilationCacheTable::New(isolate(), kInitialCacheSize); tables_[generation] = *result; } else { CompilationCacheTable table = CompilationCacheTable::cast(tables_[generation]); result = Handle<CompilationCacheTable>(table, isolate()); } return result; } // static void CompilationSubCache::AgeByGeneration(CompilationSubCache* c) { DCHECK_GT(c->generations(), 1); // Age the generations implicitly killing off the oldest. for (int i = c->generations() - 1; i > 0; i--) { c->tables_[i] = c->tables_[i - 1]; } // Set the first generation as unborn. c->tables_[0] = ReadOnlyRoots(c->isolate()).undefined_value(); } // static void CompilationSubCache::AgeCustom(CompilationSubCache* c) { DCHECK_EQ(c->generations(), 1); if (c->tables_[0].IsUndefined(c->isolate())) return; CompilationCacheTable::cast(c->tables_[0]).Age(c->isolate()); } void CompilationCacheScript::Age() { if (FLAG_isolate_script_cache_ageing) AgeCustom(this); } void CompilationCacheEval::Age() { AgeCustom(this); } void CompilationCacheRegExp::Age() { AgeByGeneration(this); } void CompilationSubCache::Iterate(RootVisitor* v) { v->VisitRootPointers(Root::kCompilationCache, nullptr, FullObjectSlot(&tables_[0]), FullObjectSlot(&tables_[generations()])); } void CompilationSubCache::Clear() { MemsetPointer(reinterpret_cast<Address*>(tables_), ReadOnlyRoots(isolate()).undefined_value().ptr(), generations()); } void CompilationSubCache::Remove(Handle<SharedFunctionInfo> function_info) { // Probe the script generation tables. Make sure not to leak handles // into the caller's handle scope. { HandleScope scope(isolate()); for (int generation = 0; generation < generations(); generation++) { Handle<CompilationCacheTable> table = GetTable(generation); table->Remove(*function_info); } } } CompilationCacheScript::CompilationCacheScript(Isolate* isolate) : CompilationSubCache(isolate, 1) {} namespace { // We only re-use a cached function for some script source code if the // script originates from the same place. This is to avoid issues // when reporting errors, etc. bool HasOrigin(Isolate* isolate, Handle<SharedFunctionInfo> function_info, const ScriptDetails& script_details) { Handle<Script> script = Handle<Script>(Script::cast(function_info->script()), isolate); // If the script name isn't set, the boilerplate script should have // an undefined name to have the same origin. Handle<Object> name; if (!script_details.name_obj.ToHandle(&name)) { return script->name().IsUndefined(isolate); } // Do the fast bailout checks first. if (script_details.line_offset != script->line_offset()) return false; if (script_details.column_offset != script->column_offset()) return false; // Check that both names are strings. If not, no match. if (!name->IsString() || !script->name().IsString()) return false; // Are the origin_options same? if (script_details.origin_options.Flags() != script->origin_options().Flags()) { return false; } // Compare the two name strings for equality. if (!String::Equals(isolate, Handle<String>::cast(name), Handle<String>(String::cast(script->name()), isolate))) { return false; } // TODO(cbruni, chromium:1244145): Remove once migrated to the context Handle<Object> maybe_host_defined_options; if (!script_details.host_defined_options.ToHandle( &maybe_host_defined_options)) { maybe_host_defined_options = isolate->factory()->empty_fixed_array(); } Handle<FixedArray> host_defined_options = Handle<FixedArray>::cast(maybe_host_defined_options); Handle<FixedArray> script_options( FixedArray::cast(script->host_defined_options()), isolate); int length = host_defined_options->length(); if (length != script_options->length()) return false; for (int i = 0; i < length; i++) { // host-defined options is a v8::PrimitiveArray. DCHECK(host_defined_options->get(i).IsPrimitive()); DCHECK(script_options->get(i).IsPrimitive()); if (!host_defined_options->get(i).StrictEquals(script_options->get(i))) { return false; } } return true; } } // namespace // TODO(245): Need to allow identical code from different contexts to // be cached in the same script generation. Currently the first use // will be cached, but subsequent code from different source / line // won't. MaybeHandle<SharedFunctionInfo> CompilationCacheScript::Lookup( Handle<String> source, const ScriptDetails& script_details, LanguageMode language_mode) { MaybeHandle<SharedFunctionInfo> result; // Probe the script generation tables. Make sure not to leak handles // into the caller's handle scope. { HandleScope scope(isolate()); const int generation = 0; DCHECK_EQ(generations(), 1); Handle<CompilationCacheTable> table = GetTable(generation); MaybeHandle<SharedFunctionInfo> probe = CompilationCacheTable::LookupScript( table, source, language_mode, isolate()); Handle<SharedFunctionInfo> function_info; if (probe.ToHandle(&function_info)) { // Break when we've found a suitable shared function info that // matches the origin. if (HasOrigin(isolate(), function_info, script_details)) { result = scope.CloseAndEscape(function_info); } } } // Once outside the manacles of the handle scope, we need to recheck // to see if we actually found a cached script. If so, we return a // handle created in the caller's handle scope. Handle<SharedFunctionInfo> function_info; if (result.ToHandle(&function_info)) { // Since HasOrigin can allocate, we need to protect the SharedFunctionInfo // with handles during the call. DCHECK(HasOrigin(isolate(), function_info, script_details)); isolate()->counters()->compilation_cache_hits()->Increment(); LOG(isolate(), CompilationCacheEvent("hit", "script", *function_info)); } else { isolate()->counters()->compilation_cache_misses()->Increment(); } return result; } void CompilationCacheScript::Put(Handle<String> source, LanguageMode language_mode, Handle<SharedFunctionInfo> function_info) { HandleScope scope(isolate()); Handle<CompilationCacheTable> table = GetFirstTable(); SetFirstTable(CompilationCacheTable::PutScript(table, source, language_mode, function_info, isolate())); } InfoCellPair CompilationCacheEval::Lookup(Handle<String> source, Handle<SharedFunctionInfo> outer_info, Handle<Context> native_context, LanguageMode language_mode, int position) { HandleScope scope(isolate()); // Make sure not to leak the table into the surrounding handle // scope. Otherwise, we risk keeping old tables around even after // having cleared the cache. InfoCellPair result; const int generation = 0; DCHECK_EQ(generations(), 1); Handle<CompilationCacheTable> table = GetTable(generation); result = CompilationCacheTable::LookupEval( table, source, outer_info, native_context, language_mode, position); if (result.has_shared()) { isolate()->counters()->compilation_cache_hits()->Increment(); } else { isolate()->counters()->compilation_cache_misses()->Increment(); } return result; } void CompilationCacheEval::Put(Handle<String> source, Handle<SharedFunctionInfo> outer_info, Handle<SharedFunctionInfo> function_info, Handle<Context> native_context, Handle<FeedbackCell> feedback_cell, int position) { HandleScope scope(isolate()); Handle<CompilationCacheTable> table = GetFirstTable(); table = CompilationCacheTable::PutEval(table, source, outer_info, function_info, native_context, feedback_cell, position); SetFirstTable(table); } MaybeHandle<FixedArray> CompilationCacheRegExp::Lookup(Handle<String> source, JSRegExp::Flags flags) { HandleScope scope(isolate()); // Make sure not to leak the table into the surrounding handle // scope. Otherwise, we risk keeping old tables around even after // having cleared the cache. Handle<Object> result = isolate()->factory()->undefined_value(); int generation; for (generation = 0; generation < generations(); generation++) { Handle<CompilationCacheTable> table = GetTable(generation); result = table->LookupRegExp(source, flags); if (result->IsFixedArray()) break; } if (result->IsFixedArray()) { Handle<FixedArray> data = Handle<FixedArray>::cast(result); if (generation != 0) { Put(source, flags, data); } isolate()->counters()->compilation_cache_hits()->Increment(); return scope.CloseAndEscape(data); } else { isolate()->counters()->compilation_cache_misses()->Increment(); return MaybeHandle<FixedArray>(); } } void CompilationCacheRegExp::Put(Handle<String> source, JSRegExp::Flags flags, Handle<FixedArray> data) { HandleScope scope(isolate()); Handle<CompilationCacheTable> table = GetFirstTable(); SetFirstTable( CompilationCacheTable::PutRegExp(isolate(), table, source, flags, data)); } void CompilationCache::Remove(Handle<SharedFunctionInfo> function_info) { if (!IsEnabledScriptAndEval()) return; eval_global_.Remove(function_info); eval_contextual_.Remove(function_info); script_.Remove(function_info); } MaybeHandle<SharedFunctionInfo> CompilationCache::LookupScript( Handle<String> source, const ScriptDetails& script_details, LanguageMode language_mode) { if (!IsEnabledScriptAndEval()) return MaybeHandle<SharedFunctionInfo>(); return script_.Lookup(source, script_details, language_mode); } InfoCellPair CompilationCache::LookupEval(Handle<String> source, Handle<SharedFunctionInfo> outer_info, Handle<Context> context, LanguageMode language_mode, int position) { InfoCellPair result; if (!IsEnabledScriptAndEval()) return result; const char* cache_type; if (context->IsNativeContext()) { result = eval_global_.Lookup(source, outer_info, context, language_mode, position); cache_type = "eval-global"; } else { DCHECK_NE(position, kNoSourcePosition); Handle<Context> native_context(context->native_context(), isolate()); result = eval_contextual_.Lookup(source, outer_info, native_context, language_mode, position); cache_type = "eval-contextual"; } if (result.has_shared()) { LOG(isolate(), CompilationCacheEvent("hit", cache_type, result.shared())); } return result; } MaybeHandle<FixedArray> CompilationCache::LookupRegExp(Handle<String> source, JSRegExp::Flags flags) { return reg_exp_.Lookup(source, flags); } void CompilationCache::PutScript(Handle<String> source, LanguageMode language_mode, Handle<SharedFunctionInfo> function_info) { if (!IsEnabledScriptAndEval()) return; LOG(isolate(), CompilationCacheEvent("put", "script", *function_info)); script_.Put(source, language_mode, function_info); } void CompilationCache::PutEval(Handle<String> source, Handle<SharedFunctionInfo> outer_info, Handle<Context> context, Handle<SharedFunctionInfo> function_info, Handle<FeedbackCell> feedback_cell, int position) { if (!IsEnabledScriptAndEval()) return; const char* cache_type; HandleScope scope(isolate()); if (context->IsNativeContext()) { eval_global_.Put(source, outer_info, function_info, context, feedback_cell, position); cache_type = "eval-global"; } else { DCHECK_NE(position, kNoSourcePosition); Handle<Context> native_context(context->native_context(), isolate()); eval_contextual_.Put(source, outer_info, function_info, native_context, feedback_cell, position); cache_type = "eval-contextual"; } LOG(isolate(), CompilationCacheEvent("put", cache_type, *function_info)); } void CompilationCache::PutRegExp(Handle<String> source, JSRegExp::Flags flags, Handle<FixedArray> data) { reg_exp_.Put(source, flags, data); } void CompilationCache::Clear() { for (int i = 0; i < kSubCacheCount; i++) { subcaches_[i]->Clear(); } } void CompilationCache::Iterate(RootVisitor* v) { for (int i = 0; i < kSubCacheCount; i++) { subcaches_[i]->Iterate(v); } } void CompilationCache::MarkCompactPrologue() { for (int i = 0; i < kSubCacheCount; i++) { subcaches_[i]->Age(); } } void CompilationCache::EnableScriptAndEval() { enabled_script_and_eval_ = true; } void CompilationCache::DisableScriptAndEval() { enabled_script_and_eval_ = false; Clear(); } } // namespace internal } // namespace v8