// Copyright 2006-2009 the V8 project authors. All rights reserved. // Redistribution and use in source and binary forms, with or without // modification, are permitted provided that the following conditions are // met: // // * Redistributions of source code must retain the above copyright // notice, this list of conditions and the following disclaimer. // * Redistributions in binary form must reproduce the above // copyright notice, this list of conditions and the following // disclaimer in the documentation and/or other materials provided // with the distribution. // * Neither the name of Google Inc. nor the names of its // contributors may be used to endorse or promote products derived // from this software without specific prior written permission. // // THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS // "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT // LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR // A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT // OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, // SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT // LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, // DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY // THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT // (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE // OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. // // Tests of logging functions from log.h #include #include #include "src/api/api-inl.h" #include "src/base/strings.h" #include "src/builtins/builtins.h" #include "src/codegen/compilation-cache.h" #include "src/execution/vm-state-inl.h" #include "src/init/v8.h" #include "src/logging/log-utils.h" #include "src/logging/log.h" #include "src/objects/objects-inl.h" #include "src/profiler/cpu-profiler.h" #include "src/utils/ostreams.h" #include "src/utils/version.h" #include "test/cctest/cctest.h" using v8::base::EmbeddedVector; using v8::internal::Address; using v8::internal::Logger; namespace { #define SETUP_FLAGS() \ i::FLAG_log = true; \ i::FLAG_prof = true; \ i::FLAG_log_code = true; \ i::FLAG_logfile = i::Log::kLogToTemporaryFile; \ i::FLAG_logfile_per_isolate = false static std::vector Split(const std::string& s, char delimiter) { std::vector result; std::string line; std::istringstream stream(s); while (std::getline(stream, line, delimiter)) { result.push_back(line); } return result; } class V8_NODISCARD ScopedLoggerInitializer { public: explicit ScopedLoggerInitializer(v8::Isolate* isolate) : temp_file_(nullptr), isolate_(isolate), isolate_scope_(isolate), scope_(isolate), env_(v8::Context::New(isolate)), logger_(reinterpret_cast(isolate)->logger()) { env_->Enter(); } ~ScopedLoggerInitializer() { env_->Exit(); FILE* log_file = logger_->TearDownAndGetLogFile(); if (log_file != nullptr) fclose(log_file); } ScopedLoggerInitializer(const ScopedLoggerInitializer&) = delete; ScopedLoggerInitializer& operator=(const ScopedLoggerInitializer&) = delete; v8::Local& env() { return env_; } v8::Isolate* isolate() { return isolate_; } i::Isolate* i_isolate() { return reinterpret_cast(isolate()); } Logger* logger() { return logger_; } v8::Local GetLogString() { int length = static_cast(raw_log_.size()); return v8::String::NewFromUtf8(isolate_, raw_log_.c_str(), v8::NewStringType::kNormal, length) .ToLocalChecked(); } void PrintLog() { i::StdoutStream os; os << raw_log_ << std::flush; } void StopLogging() { bool exists = false; raw_log_ = i::ReadFile(StopLoggingGetTempFile(), &exists, true); log_ = Split(raw_log_, '\n'); CHECK(exists); } // Searches |log_| for a line which contains all the strings in |search_terms| // as substrings, starting from the index |start|, and returns the index of // the found line. Returns std::string::npos if no line is found. size_t IndexOfLine(const std::vector& search_terms, size_t start = 0) { for (size_t i = start; i < log_.size(); ++i) { const std::string& line = log_.at(i); bool all_terms_found = true; for (const std::string& term : search_terms) { all_terms_found &= line.find(term) != std::string::npos; } if (all_terms_found) return i; } return std::string::npos; } bool ContainsLine(const std::vector& search_terms, size_t start = 0) { return IndexOfLine(search_terms, start) != std::string::npos; } // Calls IndexOfLine for each set of substring terms in // |all_line_search_terms|, in order. Returns true if they're all found. bool ContainsLinesInOrder( const std::vector>& all_line_search_terms, size_t start = 0) { CHECK_GT(log_.size(), 0); for (auto& search_terms : all_line_search_terms) { start = IndexOfLine(search_terms, start); if (start == std::string::npos) return false; ++start; // Skip the found line. } return true; } std::unordered_set ExtractLogAddresses(std::string search_term, size_t address_column, bool allow_duplicates) { CHECK_GT(log_.size(), 0); // Map addresses of Maps to log_lines. std::unordered_map map; size_t current = 0; while (true) { current = IndexOfLine({search_term}, current); if (current == std::string::npos) break; std::string current_line = log_.at(current); std::vector columns = Split(current_line, ','); ++current; // Skip the found line. // TODO(crbug.com/v8/8084): These two continue lines should really be // errors. But on Windows the log is sometimes mysteriously cut off at the // end. If the cut-off point happens to fall in the address field, the // conditions will be triggered. if (address_column >= columns.size()) continue; uintptr_t address = strtoull(columns.at(address_column).c_str(), nullptr, 16); if (address == 0) continue; if (!allow_duplicates) { auto match = map.find(address); // Ignore same address but different log line. if (match != map.end() && match->second.compare(current_line) == 0) { for (size_t i = 0; i < current; i++) { printf("%s\n", log_.at(i).c_str()); } printf("%zu\n", current); FATAL("%s, ... %p apperead twice:\n %s", search_term.c_str(), reinterpret_cast(address), current_line.c_str()); } } map.insert({address, current_line}); } // Extract all keys. std::unordered_set result; for (auto key_value : map) { result.insert(key_value.first); } return result; } void LogCodeObjects() { logger_->LogCodeObjects(); } void LogCompiledFunctions() { logger_->LogCompiledFunctions(); } void StringEvent(const char* name, const char* value) { logger_->StringEvent(name, value); } private: FILE* StopLoggingGetTempFile() { temp_file_ = logger_->TearDownAndGetLogFile(); CHECK(temp_file_); rewind(temp_file_); return temp_file_; } FILE* temp_file_; v8::Isolate* isolate_; v8::Isolate::Scope isolate_scope_; v8::HandleScope scope_; v8::Local env_; Logger* logger_; std::string raw_log_; std::vector log_; }; class TestCodeEventHandler : public v8::CodeEventHandler { public: explicit TestCodeEventHandler(v8::Isolate* isolate) : v8::CodeEventHandler(isolate), isolate_(isolate) {} size_t CountLines(std::string prefix, std::string suffix = std::string()) { if (event_log_.empty()) return 0; size_t match = 0; for (const std::string& line : event_log_) { size_t prefix_pos = line.find(prefix); if (prefix_pos == std::string::npos) continue; size_t suffix_pos = line.rfind(suffix); if (suffix_pos == std::string::npos) continue; if (suffix_pos != line.length() - suffix.length()) continue; if (prefix_pos >= suffix_pos) continue; match++; } return match; } void Handle(v8::CodeEvent* code_event) override { std::string log_line = ""; log_line += v8::CodeEvent::GetCodeEventTypeName(code_event->GetCodeType()); log_line += " "; log_line += FormatName(code_event); event_log_.push_back(log_line); } private: std::string FormatName(v8::CodeEvent* code_event) { std::string name = std::string(code_event->GetComment()); if (name.empty()) { v8::Local functionName = code_event->GetFunctionName(); std::string buffer(functionName->Utf8Length(isolate_) + 1, 0); functionName->WriteUtf8(isolate_, &buffer[0], functionName->Utf8Length(isolate_) + 1); // Sanitize name, removing unwanted \0 resulted from WriteUtf8 name = std::string(buffer.c_str()); } return name; } std::vector event_log_; v8::Isolate* isolate_; }; } // namespace // Test for issue http://crbug.com/23768 in Chromium. // Heap can contain scripts with already disposed external sources. // We need to verify that LogCompiledFunctions doesn't crash on them. namespace { class SimpleExternalString : public v8::String::ExternalStringResource { public: explicit SimpleExternalString(const char* source) : utf_source_( v8::base::OwnedVector::Of(v8::base::CStrVector(source))) { } ~SimpleExternalString() override = default; size_t length() const override { return utf_source_.size(); } const uint16_t* data() const override { return utf_source_.begin(); } private: v8::base::OwnedVector utf_source_; }; } // namespace TEST(Issue23768) { v8::HandleScope scope(CcTest::isolate()); v8::Local env = v8::Context::New(CcTest::isolate()); env->Enter(); SimpleExternalString source_ext_str("(function ext() {})();"); v8::Local source = v8::String::NewExternalTwoByte(CcTest::isolate(), &source_ext_str) .ToLocalChecked(); // Script needs to have a name in order to trigger InitLineEnds execution. v8::Local origin = v8::String::NewFromUtf8Literal(CcTest::isolate(), "issue-23768-test"); v8::Local evil_script = CompileWithOrigin(source, origin, false); CHECK(!evil_script.IsEmpty()); CHECK(!evil_script->Run(env).IsEmpty()); i::Handle i_source( i::ExternalTwoByteString::cast(*v8::Utils::OpenHandle(*source)), CcTest::i_isolate()); // This situation can happen if source was an external string disposed // by its owner. i_source->SetResource(CcTest::i_isolate(), nullptr); // Must not crash. CcTest::i_isolate()->logger()->LogCompiledFunctions(); } static void ObjMethod1(const v8::FunctionCallbackInfo& args) { } UNINITIALIZED_TEST(LogCallbacks) { SETUP_FLAGS(); v8::Isolate::CreateParams create_params; create_params.array_buffer_allocator = CcTest::array_buffer_allocator(); v8::Isolate* isolate = v8::Isolate::New(create_params); { ScopedLoggerInitializer logger(isolate); v8::Local obj = v8::Local::New( isolate, v8::FunctionTemplate::New(isolate)); obj->SetClassName(v8_str("Obj")); v8::Local proto = obj->PrototypeTemplate(); v8::Local signature = v8::Signature::New(isolate, obj); proto->Set(v8_str("method1"), v8::FunctionTemplate::New(isolate, ObjMethod1, v8::Local(), signature), static_cast(v8::DontDelete)); logger.env() ->Global() ->Set(logger.env(), v8_str("Obj"), obj->GetFunction(logger.env()).ToLocalChecked()) .FromJust(); CompileRun("Obj.prototype.method1.toString();"); logger.LogCompiledFunctions(); logger.StopLogging(); Address ObjMethod1_entry = reinterpret_cast
(ObjMethod1); #if USES_FUNCTION_DESCRIPTORS ObjMethod1_entry = *FUNCTION_ENTRYPOINT_ADDRESS(ObjMethod1_entry); #endif v8::base::EmbeddedVector suffix_buffer; v8::base::SNPrintF(suffix_buffer, ",0x%" V8PRIxPTR ",1,method1", ObjMethod1_entry); CHECK(logger.ContainsLine( {"code-creation,Callback,-2,", std::string(suffix_buffer.begin())})); } isolate->Dispose(); } static void Prop1Getter(v8::Local property, const v8::PropertyCallbackInfo& info) { } static void Prop1Setter(v8::Local property, v8::Local value, const v8::PropertyCallbackInfo& info) { } static void Prop2Getter(v8::Local property, const v8::PropertyCallbackInfo& info) { } UNINITIALIZED_TEST(LogAccessorCallbacks) { SETUP_FLAGS(); v8::Isolate::CreateParams create_params; create_params.array_buffer_allocator = CcTest::array_buffer_allocator(); v8::Isolate* isolate = v8::Isolate::New(create_params); { ScopedLoggerInitializer logger(isolate); v8::Local obj = v8::Local::New( isolate, v8::FunctionTemplate::New(isolate)); obj->SetClassName(v8_str("Obj")); v8::Local inst = obj->InstanceTemplate(); inst->SetAccessor(v8_str("prop1"), Prop1Getter, Prop1Setter); inst->SetAccessor(v8_str("prop2"), Prop2Getter); logger.logger()->LogAccessorCallbacks(); logger.StopLogging(); Address Prop1Getter_entry = reinterpret_cast
(Prop1Getter); #if USES_FUNCTION_DESCRIPTORS Prop1Getter_entry = *FUNCTION_ENTRYPOINT_ADDRESS(Prop1Getter_entry); #endif v8::base::EmbeddedVector prop1_getter_record; v8::base::SNPrintF(prop1_getter_record, ",0x%" V8PRIxPTR ",1,get prop1", Prop1Getter_entry); CHECK(logger.ContainsLine({"code-creation,Callback,-2,", std::string(prop1_getter_record.begin())})); Address Prop1Setter_entry = reinterpret_cast
(Prop1Setter); #if USES_FUNCTION_DESCRIPTORS Prop1Setter_entry = *FUNCTION_ENTRYPOINT_ADDRESS(Prop1Setter_entry); #endif v8::base::EmbeddedVector prop1_setter_record; v8::base::SNPrintF(prop1_setter_record, ",0x%" V8PRIxPTR ",1,set prop1", Prop1Setter_entry); CHECK(logger.ContainsLine({"code-creation,Callback,-2,", std::string(prop1_setter_record.begin())})); Address Prop2Getter_entry = reinterpret_cast
(Prop2Getter); #if USES_FUNCTION_DESCRIPTORS Prop2Getter_entry = *FUNCTION_ENTRYPOINT_ADDRESS(Prop2Getter_entry); #endif v8::base::EmbeddedVector prop2_getter_record; v8::base::SNPrintF(prop2_getter_record, ",0x%" V8PRIxPTR ",1,get prop2", Prop2Getter_entry); CHECK(logger.ContainsLine({"code-creation,Callback,-2,", std::string(prop2_getter_record.begin())})); } isolate->Dispose(); } UNINITIALIZED_TEST(LogVersion) { SETUP_FLAGS(); v8::Isolate::CreateParams create_params; create_params.array_buffer_allocator = CcTest::array_buffer_allocator(); v8::Isolate* isolate = v8::Isolate::New(create_params); { ScopedLoggerInitializer logger(isolate); logger.StopLogging(); v8::base::EmbeddedVector line_buffer; v8::base::SNPrintF(line_buffer, "%d,%d,%d,%d,%d", i::Version::GetMajor(), i::Version::GetMinor(), i::Version::GetBuild(), i::Version::GetPatch(), i::Version::IsCandidate()); CHECK( logger.ContainsLine({"v8-version,", std::string(line_buffer.begin())})); } isolate->Dispose(); } // https://crbug.com/539892 // CodeCreateEvents with really large names should not crash. UNINITIALIZED_TEST(Issue539892) { class FakeCodeEventLogger : public i::CodeEventLogger { public: explicit FakeCodeEventLogger(i::Isolate* isolate) : CodeEventLogger(isolate) {} void CodeMoveEvent(i::AbstractCode from, i::AbstractCode to) override {} void CodeDisableOptEvent(i::Handle code, i::Handle shared) override { } private: void LogRecordedBuffer(i::Handle code, i::MaybeHandle maybe_shared, const char* name, int length) override {} #if V8_ENABLE_WEBASSEMBLY void LogRecordedBuffer(const i::wasm::WasmCode* code, const char* name, int length) override {} #endif // V8_ENABLE_WEBASSEMBLY }; SETUP_FLAGS(); v8::Isolate::CreateParams create_params; create_params.array_buffer_allocator = CcTest::array_buffer_allocator(); v8::Isolate* isolate = v8::Isolate::New(create_params); FakeCodeEventLogger code_event_logger(reinterpret_cast(isolate)); { ScopedLoggerInitializer logger(isolate); logger.logger()->AddCodeEventListener(&code_event_logger); // Function with a really large name. const char* source_text = "(function " "baaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa" "aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa" "aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa" "aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa" "aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa" "aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa" "aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa" "aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa" "aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa" "aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa" "aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa" "aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa" "aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa" "aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa" "aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa" "aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa" "aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaac" "(){})();"; CompileRun(source_text); // Must not crash. logger.LogCompiledFunctions(); } isolate->Dispose(); } UNINITIALIZED_TEST(LogAll) { SETUP_FLAGS(); i::FLAG_log_all = true; i::FLAG_log_deopt = true; i::FLAG_log_api = true; i::FLAG_turbo_inlining = false; i::FLAG_log_internal_timer_events = true; i::FLAG_allow_natives_syntax = true; v8::Isolate::CreateParams create_params; create_params.array_buffer_allocator = CcTest::array_buffer_allocator(); v8::Isolate* isolate = v8::Isolate::New(create_params); { ScopedLoggerInitializer logger(isolate); const char* source_text = R"( function testAddFn(a,b) { return a + b }; let result; // Warm up the ICs. %PrepareFunctionForOptimization(testAddFn); for (let i = 0; i < 100000; i++) { result = testAddFn(i, i); }; // Enforce optimization. %OptimizeFunctionOnNextCall(testAddFn); result = testAddFn(1, 1); // Cause deopt. testAddFn('1', 1) for (let i = 0; i < 100000; i++) { result = testAddFn('1', i); } )"; CompileRun(source_text); logger.StopLogging(); // We should find at least one code-creation even for testAddFn(); CHECK(logger.ContainsLine({"api,v8::Context::New"})); CHECK(logger.ContainsLine({"timer-event-start", "V8.CompileCode"})); CHECK(logger.ContainsLine({"timer-event-end", "V8.CompileCode"})); CHECK(logger.ContainsLine({"code-creation,Script", ":1:1"})); CHECK(logger.ContainsLine({"api,v8::Script::Run"})); CHECK(logger.ContainsLine({"code-creation,LazyCompile,", "testAddFn"})); if (i::FLAG_opt && !i::FLAG_always_opt) { CHECK(logger.ContainsLine({"code-deopt,", "not a Smi"})); CHECK(logger.ContainsLine({"timer-event-start", "V8.DeoptimizeCode"})); CHECK(logger.ContainsLine({"timer-event-end", "V8.DeoptimizeCode"})); } } isolate->Dispose(); } #ifndef V8_TARGET_ARCH_ARM UNINITIALIZED_TEST(LogInterpretedFramesNativeStack) { SETUP_FLAGS(); i::FLAG_interpreted_frames_native_stack = true; v8::Isolate::CreateParams create_params; create_params.array_buffer_allocator = CcTest::array_buffer_allocator(); v8::Isolate* isolate = v8::Isolate::New(create_params); { ScopedLoggerInitializer logger(isolate); const char* source_text = "function testLogInterpretedFramesNativeStack(a,b) { return a + b };" "testLogInterpretedFramesNativeStack('1', 1);"; CompileRun(source_text); logger.StopLogging(); CHECK(logger.ContainsLinesInOrder( {{"LazyCompile", "testLogInterpretedFramesNativeStack"}, {"LazyCompile", "testLogInterpretedFramesNativeStack"}})); } isolate->Dispose(); } UNINITIALIZED_TEST(LogInterpretedFramesNativeStackWithSerialization) { SETUP_FLAGS(); i::FLAG_interpreted_frames_native_stack = true; i::FLAG_always_opt = false; v8::Isolate::CreateParams create_params; create_params.array_buffer_allocator = CcTest::array_buffer_allocator(); v8::ScriptCompiler::CachedData* cache = nullptr; bool has_cache = cache != nullptr; // NOTE(mmarchini): Runs the test two times. The first time it will compile // our script and will create a code cache for it. The second time we'll // deserialize the cache and check if our function was logged correctly. // We disallow compilation on the second run to ensure we're loading from // cache. do { v8::Isolate* isolate = v8::Isolate::New(create_params); { ScopedLoggerInitializer logger(isolate); has_cache = cache != nullptr; v8::ScriptCompiler::CompileOptions options = has_cache ? v8::ScriptCompiler::kConsumeCodeCache : v8::ScriptCompiler::kEagerCompile; v8::HandleScope scope(isolate); v8::Isolate::Scope isolate_scope(isolate); v8::Local context = v8::Context::New(isolate); v8::Local source = v8_str( "function eyecatcher() { return a * a; } return eyecatcher();"); v8::Local arg_str = v8_str("a"); v8::ScriptOrigin origin(isolate, v8_str("filename")); i::DisallowCompilation* no_compile_expected = has_cache ? new i::DisallowCompilation( reinterpret_cast(isolate)) : nullptr; v8::ScriptCompiler::Source script_source(source, origin, cache); v8::Local fun = v8::ScriptCompiler::CompileFunctionInContext( context, &script_source, 1, &arg_str, 0, nullptr, options) .ToLocalChecked(); if (has_cache) { logger.StopLogging(); logger.PrintLog(); // Function is logged twice: once as interpreted, and once as the // interpreter entry trampoline builtin. CHECK(logger.ContainsLinesInOrder( {{"Function", "eyecatcher"}, {"Function", "eyecatcher"}})); } v8::Local arg = v8_num(3); v8::Local result = fun->Call(context, v8::Undefined(isolate), 1, &arg).ToLocalChecked(); CHECK_EQ(9, result->Int32Value(context).FromJust()); cache = v8::ScriptCompiler::CreateCodeCacheForFunction(fun); if (no_compile_expected != nullptr) delete no_compile_expected; } isolate->Dispose(); } while (!has_cache); delete cache; } #endif // V8_TARGET_ARCH_ARM UNINITIALIZED_TEST(ExternalCodeEventListener) { i::FLAG_log = false; i::FLAG_prof = false; v8::Isolate::CreateParams create_params; create_params.array_buffer_allocator = CcTest::array_buffer_allocator(); v8::Isolate* isolate = v8::Isolate::New(create_params); { v8::HandleScope scope(isolate); v8::Isolate::Scope isolate_scope(isolate); v8::Local context = v8::Context::New(isolate); v8::Context::Scope context_scope(context); TestCodeEventHandler code_event_handler(isolate); const char* source_text_before_start = "function testCodeEventListenerBeforeStart(a,b) { return a + b };" "testCodeEventListenerBeforeStart('1', 1);"; CompileRun(source_text_before_start); CHECK_EQ(code_event_handler.CountLines("Function", "testCodeEventListenerBeforeStart"), 0); CHECK_EQ(code_event_handler.CountLines("LazyCompile", "testCodeEventListenerBeforeStart"), 0); code_event_handler.Enable(); CHECK_GE(code_event_handler.CountLines("Function", "testCodeEventListenerBeforeStart"), 1); const char* source_text_after_start = "function testCodeEventListenerAfterStart(a,b) { return a + b };" "testCodeEventListenerAfterStart('1', 1);"; CompileRun(source_text_after_start); CHECK_GE(code_event_handler.CountLines("LazyCompile", "testCodeEventListenerAfterStart"), 1); } isolate->Dispose(); } UNINITIALIZED_TEST(ExternalCodeEventListenerInnerFunctions) { i::FLAG_log = false; i::FLAG_prof = false; v8::ScriptCompiler::CachedData* cache; static const char* source_cstring = "(function f1() { return (function f2() {}); })()"; v8::Isolate::CreateParams create_params; create_params.array_buffer_allocator = CcTest::array_buffer_allocator(); v8::Isolate* isolate1 = v8::Isolate::New(create_params); { // Test that we emit the correct code events from eagerly compiling. v8::HandleScope scope(isolate1); v8::Isolate::Scope isolate_scope(isolate1); v8::Local context = v8::Context::New(isolate1); v8::Context::Scope context_scope(context); TestCodeEventHandler code_event_handler(isolate1); code_event_handler.Enable(); v8::Local source_string = v8_str(source_cstring); v8::ScriptOrigin origin(isolate1, v8_str("test")); v8::ScriptCompiler::Source source(source_string, origin); v8::Local script = v8::ScriptCompiler::CompileUnboundScript(isolate1, &source) .ToLocalChecked(); CHECK_EQ(code_event_handler.CountLines("Function", "f1"), 1 + (i::FLAG_stress_background_compile ? 1 : 0) + (i::FLAG_always_sparkplug ? 1 : 0)); CHECK_EQ(code_event_handler.CountLines("Function", "f2"), 1 + (i::FLAG_stress_background_compile ? 1 : 0) + (i::FLAG_always_sparkplug ? 1 : 0)); cache = v8::ScriptCompiler::CreateCodeCache(script); } isolate1->Dispose(); v8::Isolate* isolate2 = v8::Isolate::New(create_params); { // Test that we emit the correct code events from deserialization. v8::HandleScope scope(isolate2); v8::Isolate::Scope isolate_scope(isolate2); v8::Local context = v8::Context::New(isolate2); v8::Context::Scope context_scope(context); TestCodeEventHandler code_event_handler(isolate2); code_event_handler.Enable(); v8::Local source_string = v8_str(source_cstring); v8::ScriptOrigin origin(isolate2, v8_str("test")); v8::ScriptCompiler::Source source(source_string, origin, cache); { i::DisallowCompilation no_compile_expected( reinterpret_cast(isolate2)); v8::ScriptCompiler::CompileUnboundScript( isolate2, &source, v8::ScriptCompiler::kConsumeCodeCache) .ToLocalChecked(); } CHECK_EQ(code_event_handler.CountLines("Function", "f1"), 1); CHECK_EQ(code_event_handler.CountLines("Function", "f2"), 1); } isolate2->Dispose(); } #ifndef V8_TARGET_ARCH_ARM UNINITIALIZED_TEST(ExternalCodeEventListenerWithInterpretedFramesNativeStack) { i::FLAG_log = false; i::FLAG_prof = false; i::FLAG_interpreted_frames_native_stack = true; v8::Isolate::CreateParams create_params; create_params.array_buffer_allocator = CcTest::array_buffer_allocator(); v8::Isolate* isolate = v8::Isolate::New(create_params); { v8::HandleScope scope(isolate); v8::Isolate::Scope isolate_scope(isolate); v8::Local context = v8::Context::New(isolate); context->Enter(); TestCodeEventHandler code_event_handler(isolate); const char* source_text_before_start = "function testCodeEventListenerBeforeStart(a,b) { return a + b };" "testCodeEventListenerBeforeStart('1', 1);"; CompileRun(source_text_before_start); CHECK_EQ(code_event_handler.CountLines("Function", "testCodeEventListenerBeforeStart"), 0); code_event_handler.Enable(); CHECK_GE(code_event_handler.CountLines("Function", "testCodeEventListenerBeforeStart"), 2); const char* source_text_after_start = "function testCodeEventListenerAfterStart(a,b) { return a + b };" "testCodeEventListenerAfterStart('1', 1);"; CompileRun(source_text_after_start); CHECK_GE(code_event_handler.CountLines("LazyCompile", "testCodeEventListenerAfterStart"), 2); CHECK_EQ( code_event_handler.CountLines("Builtin", "InterpreterEntryTrampoline"), 1); context->Exit(); } isolate->Dispose(); } #endif // V8_TARGET_ARCH_ARM UNINITIALIZED_TEST(TraceMaps) { SETUP_FLAGS(); i::FLAG_log_maps = true; v8::Isolate::CreateParams create_params; create_params.array_buffer_allocator = CcTest::array_buffer_allocator(); v8::Isolate* isolate = v8::Isolate::New(create_params); { ScopedLoggerInitializer logger(isolate); // Try to create many different kind of maps to make sure the logging won't // crash. More detailed tests are implemented separately. const char* source_text = R"( let a = {}; for (let i = 0; i < 500; i++) { a['p'+i] = i }; class Test { constructor(i) { this.a = 1; this['p'+i] = 1; } }; let t = new Test(); t.b = 1; t.c = 1; t.d = 3; for (let i = 0; i < 100; i++) { t = new Test(i) }; t.b = {}; )"; CompileRunChecked(isolate, source_text); logger.StopLogging(); // Mostly superficial checks. CHECK(logger.ContainsLine({"map,InitialMap", ",0x"})); CHECK(logger.ContainsLine({"map,Transition", ",0x"})); CHECK(logger.ContainsLine({"map-details", ",0x"})); } i::FLAG_log_maps = false; isolate->Dispose(); } namespace { // Ensure that all Maps found on the heap have a single corresponding map-create // and map-details entry in the v8.log. void ValidateMapDetailsLogging(v8::Isolate* isolate, ScopedLoggerInitializer* logger) { // map-create might have duplicates if a Map address is reused after a gc. std::unordered_set map_create_addresses = logger->ExtractLogAddresses("map-create", 2, true); std::unordered_set map_details_addresses = logger->ExtractLogAddresses("map-details", 2, false); // Iterate over all maps on the heap. i::Heap* heap = reinterpret_cast(isolate)->heap(); i::HeapObjectIterator iterator(heap); i::DisallowGarbageCollection no_gc; size_t i = 0; for (i::HeapObject obj = iterator.Next(); !obj.is_null(); obj = iterator.Next()) { if (!obj.IsMap()) continue; i++; uintptr_t address = obj.ptr(); if (map_create_addresses.find(address) == map_create_addresses.end()) { // logger->PrintLog(); i::Map::cast(obj).Print(); FATAL( "Map (%p, #%zu) creation not logged during startup with " "--log-maps!" "\n# Expected Log Line: map-create, ... %p", reinterpret_cast(obj.ptr()), i, reinterpret_cast(obj.ptr())); } else if (map_details_addresses.find(address) == map_details_addresses.end()) { // logger->PrintLog(); i::Map::cast(obj).Print(); FATAL( "Map (%p, #%zu) details not logged during startup with " "--log-maps!" "\n# Expected Log Line: map-details, ... %p", reinterpret_cast(obj.ptr()), i, reinterpret_cast(obj.ptr())); } } } } // namespace UNINITIALIZED_TEST(LogMapsDetailsStartup) { // Reusing map addresses might cause these tests to fail. if (i::FLAG_gc_global || i::FLAG_stress_compaction || i::FLAG_stress_incremental_marking || i::FLAG_enable_third_party_heap) { return; } // Test that all Map details from Maps in the snapshot are logged properly. SETUP_FLAGS(); i::FLAG_log_maps = true; v8::Isolate::CreateParams create_params; create_params.array_buffer_allocator = CcTest::array_buffer_allocator(); v8::Isolate* isolate = v8::Isolate::New(create_params); { ScopedLoggerInitializer logger(isolate); logger.StopLogging(); ValidateMapDetailsLogging(isolate, &logger); } i::FLAG_log_function_events = false; isolate->Dispose(); } UNINITIALIZED_TEST(LogMapsDetailsCode) { // Reusing map addresses might cause these tests to fail. if (i::FLAG_gc_global || i::FLAG_stress_compaction || i::FLAG_stress_incremental_marking || i::FLAG_enable_third_party_heap) { return; } SETUP_FLAGS(); i::FLAG_retain_maps_for_n_gc = 0xFFFFFFF; i::FLAG_log_maps = true; v8::Isolate::CreateParams create_params; create_params.array_buffer_allocator = CcTest::array_buffer_allocator(); v8::Isolate* isolate = v8::Isolate::New(create_params); const char* source = R"( // Normal properties overflowing into dict-mode. let a = {}; for (let i = 0; i < 500; i++) { a['p'+i] = i }; // Constructor / initial maps function Constructor(dictElements=false) { this.a = 1; this.b = 2; this.c = 3; if (dictElements) { this[0xFFFFF] = 1; } this.d = 4; this.e = 5; this.f = 5; } // Keep objects and their maps alive to avoid reusing map addresses. let instances = []; let instance; for (let i =0; i < 500; i++) { instances.push(new Constructor()); } // Map deprecation. for (let i =0; i < 500; i++) { instance = new Constructor(); instance.d = 1.1; instances.push(instance); } for (let i =0; i < 500; i++) { instance = new Constructor(); instance.b = 1.1; instances.push(instance); } for (let i =0; i < 500; i++) { instance = new Constructor(); instance.c = Object; instances.push(instance); } // Create instance with dict-elements. instances.push(new Constructor(true)); // Class class Test { constructor(i) { this.a = 1; this['p'+i] = 1; } }; let t = new Test(); t.b = 1; t.c = 1; t.d = 3; for (let i = 0; i < 100; i++) { t = new Test(i); instances.push(t); } t.b = {}; // Anonymous classes function create(value) { return new class { constructor() { this.value = value; } } } for (let i = 0; i < 100; i++) { instances.push(create(i)); }; // Modifying some protoypes. Array.prototype.helper = () => 1; [1,2,3].helper(); )"; { ScopedLoggerInitializer logger(isolate); CompileRunChecked(isolate, source); logger.StopLogging(); ValidateMapDetailsLogging(isolate, &logger); } i::FLAG_log_function_events = false; isolate->Dispose(); } UNINITIALIZED_TEST(LogMapsDetailsContexts) { // Reusing map addresses might cause these tests to fail. if (i::FLAG_gc_global || i::FLAG_stress_compaction || i::FLAG_stress_incremental_marking || i::FLAG_enable_third_party_heap) { return; } // Test that all Map details from Maps in the snapshot are logged properly. SETUP_FLAGS(); i::FLAG_log_maps = true; v8::Isolate::CreateParams create_params; create_params.array_buffer_allocator = CcTest::array_buffer_allocator(); v8::Isolate* isolate = v8::Isolate::New(create_params); { ScopedLoggerInitializer logger(isolate); // Use the default context. CompileRunChecked(isolate, "{a:1}"); // Create additional contexts. v8::Local env1 = v8::Context::New(isolate); env1->Enter(); CompileRun(env1, "{b:1}").ToLocalChecked(); v8::Local env2 = v8::Context::New(isolate); env2->Enter(); CompileRun(env2, "{c:1}").ToLocalChecked(); env2->Exit(); env1->Exit(); logger.StopLogging(); ValidateMapDetailsLogging(isolate, &logger); } i::FLAG_log_function_events = false; isolate->Dispose(); } UNINITIALIZED_TEST(ConsoleTimeEvents) { SETUP_FLAGS(); v8::Isolate::CreateParams create_params; create_params.array_buffer_allocator = CcTest::array_buffer_allocator(); v8::Isolate* isolate = v8::Isolate::New(create_params); { ScopedLoggerInitializer logger(isolate); // Test that console time events are properly logged const char* source_text = "console.time();" "console.timeEnd();" "console.timeStamp();" "console.time('timerEvent1');" "console.timeEnd('timerEvent1');" "console.timeStamp('timerEvent2');" "console.timeStamp('timerEvent3');"; CompileRun(source_text); logger.StopLogging(); std::vector> lines = { {"timer-event-start,default,"}, {"timer-event-end,default,"}, {"timer-event,default,"}, {"timer-event-start,timerEvent1,"}, {"timer-event-end,timerEvent1,"}, {"timer-event,timerEvent2,"}, {"timer-event,timerEvent3,"}}; CHECK(logger.ContainsLinesInOrder(lines)); } isolate->Dispose(); } UNINITIALIZED_TEST(LogFunctionEvents) { // Always opt and stress opt will break the fine-grained log order. if (i::FLAG_always_opt) return; SETUP_FLAGS(); i::FLAG_log_function_events = true; v8::Isolate::CreateParams create_params; create_params.array_buffer_allocator = CcTest::array_buffer_allocator(); v8::Isolate* isolate = v8::Isolate::New(create_params); { ScopedLoggerInitializer logger(isolate); // Run some warmup code to help ignoring existing log entries. CompileRun( "function warmUp(a) {" " let b = () => 1;" " return function(c) { return a+b+c; };" "};" "warmUp(1)(2);" "(function warmUpEndMarkerFunction(){})();"); const char* source_text = "function lazyNotExecutedFunction() { return 'lazy' };" "function lazyFunction() { " " function lazyInnerFunction() { return 'lazy' };" " return lazyInnerFunction;" "};" "let innerFn = lazyFunction();" "innerFn();" "(function eagerFunction(){ return 'eager' })();" "function Foo() { this.foo = function(){}; };" "let i = new Foo(); i.foo();"; CompileRun(source_text); logger.StopLogging(); // Ignore all the log entries that happened before warmup size_t start = logger.IndexOfLine( {"function,first-execution", "warmUpEndMarkerFunction"}); CHECK(start != std::string::npos); std::vector> lines = { // Create a new script {"script,create"}, {"script-details"}, // Step 1: parsing top-level script, preparsing functions {"function,preparse-", ",lazyNotExecutedFunction"}, // Missing name for preparsing lazyInnerFunction // {"function,preparse-", nullptr}, {"function,preparse-", ",lazyFunction"}, {"function,full-parse,", ",eagerFunction"}, {"function,preparse-", ",Foo"}, // Missing name for inner preparsing of Foo.foo // {"function,preparse-", nullptr}, // Missing name for top-level script. {"function,parse-script,"}, // Step 2: compiling top-level script and eager functions // - Compiling script without name. {"function,interpreter,"}, {"function,interpreter,", ",eagerFunction"}, // Step 3: start executing script // Step 4. - lazy parse, lazy compiling and execute skipped functions // - execute eager functions. {"function,parse-function,", ",lazyFunction"}, {"function,interpreter-lazy,", ",lazyFunction"}, {"function,first-execution,", ",lazyFunction"}, {"function,parse-function,", ",lazyInnerFunction"}, {"function,interpreter-lazy,", ",lazyInnerFunction"}, {"function,first-execution,", ",lazyInnerFunction"}, {"function,first-execution,", ",eagerFunction"}, {"function,parse-function,", ",Foo"}, {"function,interpreter-lazy,", ",Foo"}, {"function,first-execution,", ",Foo"}, {"function,parse-function,", ",Foo.foo"}, {"function,interpreter-lazy,", ",Foo.foo"}, {"function,first-execution,", ",Foo.foo"}, }; CHECK(logger.ContainsLinesInOrder(lines, start)); } i::FLAG_log_function_events = false; isolate->Dispose(); } UNINITIALIZED_TEST(BuiltinsNotLoggedAsLazyCompile) { SETUP_FLAGS(); v8::Isolate::CreateParams create_params; create_params.array_buffer_allocator = CcTest::array_buffer_allocator(); v8::Isolate* isolate = v8::Isolate::New(create_params); { ScopedLoggerInitializer logger(isolate); logger.LogCodeObjects(); logger.LogCompiledFunctions(); logger.StopLogging(); i::Handle builtin = logger.i_isolate()->builtins()->code_handle( i::Builtin::kBooleanConstructor); v8::base::EmbeddedVector buffer; // Should only be logged as "Builtin" with a name, never as "LazyCompile". v8::base::SNPrintF(buffer, ",0x%" V8PRIxPTR ",%d,BooleanConstructor", builtin->InstructionStart(), builtin->InstructionSize()); CHECK(logger.ContainsLine( {"code-creation,Builtin,2,", std::string(buffer.begin())})); v8::base::SNPrintF(buffer, ",0x%" V8PRIxPTR ",%d,", builtin->InstructionStart(), builtin->InstructionSize()); CHECK(!logger.ContainsLine( {"code-creation,LazyCompile,2,", std::string(buffer.begin())})); } isolate->Dispose(); }