| // Copyright (c) 2017, the Dart project authors. Please see the AUTHORS file |
| // for details. All rights reserved. Use of this source code is governed by a |
| // BSD-style license that can be found in the LICENSE file. |
| |
| #include "platform/globals.h" |
| |
| #if defined(DART_USE_TCMALLOC) && defined(DEBUG) |
| |
| #include "platform/assert.h" |
| #include "vm/globals.h" |
| #include "vm/malloc_hooks.h" |
| #include "vm/os.h" |
| #include "vm/profiler.h" |
| #include "vm/profiler_service.h" |
| #include "vm/unit_test.h" |
| |
| namespace dart { |
| |
| static void MallocHookTestBufferInitializer(volatile char* buffer, |
| uintptr_t size) { |
| // Run through the buffer and do something. If we don't do this and the memory |
| // in buffer isn't touched, the tcmalloc hooks won't be called. |
| for (uintptr_t i = 0; i < size; ++i) { |
| buffer[i] = i; |
| } |
| } |
| |
| class EnableMallocHooksScope : public ValueObject { |
| public: |
| EnableMallocHooksScope() { |
| OSThread::Current(); // Ensure not allocated during test. |
| saved_enable_malloc_hooks_ = FLAG_profiler_native_memory; |
| FLAG_profiler_native_memory = true; |
| MallocHooks::Init(); |
| MallocHooks::ResetStats(); |
| } |
| |
| ~EnableMallocHooksScope() { |
| MallocHooks::Cleanup(); |
| FLAG_profiler_native_memory = saved_enable_malloc_hooks_; |
| } |
| |
| private: |
| bool saved_enable_malloc_hooks_; |
| }; |
| |
| class EnableMallocHooksAndStacksScope : public EnableMallocHooksScope { |
| public: |
| EnableMallocHooksAndStacksScope() { |
| OSThread::Current(); // Ensure not allocated during test. |
| saved_enable_stack_traces_ = MallocHooks::stack_trace_collection_enabled(); |
| MallocHooks::set_stack_trace_collection_enabled(true); |
| if (!FLAG_profiler) { |
| FLAG_profiler = true; |
| Profiler::Init(); |
| } |
| MallocHooks::ResetStats(); |
| } |
| |
| ~EnableMallocHooksAndStacksScope() { |
| MallocHooks::set_stack_trace_collection_enabled(saved_enable_stack_traces_); |
| } |
| |
| private: |
| bool saved_enable_stack_traces_; |
| }; |
| |
| UNIT_TEST_CASE(BasicMallocHookTest) { |
| EnableMallocHooksScope scope; |
| |
| EXPECT_EQ(0L, MallocHooks::allocation_count()); |
| EXPECT_EQ(0L, MallocHooks::heap_allocated_memory_in_bytes()); |
| const intptr_t buffer_size = 10; |
| char* buffer = new char[buffer_size]; |
| MallocHookTestBufferInitializer(buffer, buffer_size); |
| |
| EXPECT_EQ(1L, MallocHooks::allocation_count()); |
| EXPECT_EQ(static_cast<intptr_t>(sizeof(char) * buffer_size), |
| MallocHooks::heap_allocated_memory_in_bytes()); |
| |
| delete[] buffer; |
| EXPECT_EQ(0L, MallocHooks::allocation_count()); |
| EXPECT_EQ(0L, MallocHooks::heap_allocated_memory_in_bytes()); |
| } |
| |
| UNIT_TEST_CASE(FreeUnseenMemoryMallocHookTest) { |
| EnableMallocHooksScope scope; |
| |
| const intptr_t pre_hook_buffer_size = 3; |
| char* pre_hook_buffer = new char[pre_hook_buffer_size]; |
| MallocHookTestBufferInitializer(pre_hook_buffer, pre_hook_buffer_size); |
| |
| MallocHooks::ResetStats(); |
| EXPECT_EQ(0L, MallocHooks::allocation_count()); |
| EXPECT_EQ(0L, MallocHooks::heap_allocated_memory_in_bytes()); |
| |
| const intptr_t buffer_size = 10; |
| char* buffer = new char[buffer_size]; |
| MallocHookTestBufferInitializer(buffer, buffer_size); |
| |
| EXPECT_EQ(1L, MallocHooks::allocation_count()); |
| EXPECT_EQ(static_cast<intptr_t>(sizeof(char) * buffer_size), |
| MallocHooks::heap_allocated_memory_in_bytes()); |
| |
| delete[] pre_hook_buffer; |
| EXPECT_EQ(1L, MallocHooks::allocation_count()); |
| EXPECT_EQ(static_cast<intptr_t>(sizeof(char) * buffer_size), |
| MallocHooks::heap_allocated_memory_in_bytes()); |
| |
| delete[] buffer; |
| EXPECT_EQ(0L, MallocHooks::allocation_count()); |
| EXPECT_EQ(0L, MallocHooks::heap_allocated_memory_in_bytes()); |
| } |
| |
| VM_UNIT_TEST_CASE(StackTraceMallocHookSimpleTest) { |
| EnableMallocHooksAndStacksScope scope; |
| |
| char* var = static_cast<char*>(malloc(16 * sizeof(char))); |
| Sample* sample = MallocHooks::GetSample(var); |
| EXPECT(sample != NULL); |
| |
| free(var); |
| sample = MallocHooks::GetSample(var); |
| EXPECT(sample == NULL); |
| } |
| |
| static char* DART_NOINLINE StackTraceLengthHelper(uintptr_t* end_address) { |
| char* var = static_cast<char*>(malloc(16 * sizeof(char))); |
| *end_address = OS::GetProgramCounter(); |
| return var; |
| } |
| |
| VM_UNIT_TEST_CASE(StackTraceMallocHookLengthTest) { |
| EnableMallocHooksAndStacksScope scope; |
| |
| uintptr_t test_start_address = |
| reinterpret_cast<uintptr_t>(Dart_TestStackTraceMallocHookLengthTest); |
| uintptr_t helper_start_address = |
| reinterpret_cast<uintptr_t>(StackTraceLengthHelper); |
| uintptr_t helper_end_address = 0; |
| |
| char* var = StackTraceLengthHelper(&helper_end_address); |
| Sample* sample = MallocHooks::GetSample(var); |
| EXPECT(sample != NULL); |
| uintptr_t test_end_address = OS::GetProgramCounter(); |
| |
| // Ensure that all stack frames are where we expect them to be in the sample. |
| // If they aren't, the kSkipCount constant in malloc_hooks.cc is likely |
| // incorrect. |
| uword address = sample->At(0); |
| bool first_result = |
| (helper_start_address <= address) && (helper_end_address >= address); |
| EXPECT(first_result); |
| address = sample->At(1); |
| bool second_result = |
| (test_start_address <= address) && (test_end_address >= address); |
| EXPECT(second_result); |
| |
| if (!(first_result && second_result)) { |
| OS::PrintErr( |
| "If this test is failing, it's likely that the value set for" |
| " the number of frames to skip in malloc_hooks.cc is " |
| "incorrect for this configuration/platform. This value can be" |
| " found in malloc_hooks.cc in the AllocationInfo class, and " |
| "is stored in the kSkipCount constant.\n"); |
| OS::PrintErr("First result: %d Second Result: %d\n", first_result, |
| second_result); |
| OS::PrintErr("Dumping sample stack trace:\n"); |
| sample->DumpStackTrace(); |
| } |
| |
| free(var); |
| } |
| |
| ISOLATE_UNIT_TEST_CASE(StackTraceMallocHookSimpleJSONTest) { |
| EnableMallocHooksAndStacksScope scope; |
| |
| ClearProfileVisitor cpv(Isolate::Current()); |
| Profiler::sample_block_buffer()->VisitSamples(&cpv); |
| |
| char* var = static_cast<char*>(malloc(16 * sizeof(char))); |
| JSONStream js; |
| ProfilerService::PrintNativeAllocationJSON(&js, -1, -1, false); |
| const char* json = js.ToCString(); |
| |
| // Check that all the stack frames from the current down to main are actually |
| // present in the profile. This is just a simple sanity check to make sure |
| // that the ProfileTrie has a representation of the stack trace collected when |
| // var is allocated. More intense testing is already done in profiler_test.cc. |
| // This is brittle: inlining and ICF in the C compiler and linker will affect |
| // the frames we see. |
| EXPECT_SUBSTRING("\"dart::Dart_TestStackTraceMallocHookSimpleJSONTest()\"", |
| json); |
| EXPECT_SUBSTRING("\"dart::TestCase::Run()\"", json); |
| EXPECT_SUBSTRING("\"dart::TestCaseBase::RunTest()\"", json); |
| EXPECT_SUBSTRING("\"main\"", json); |
| |
| free(var); |
| } |
| |
| }; // namespace dart |
| |
| #endif // defined(DART_USE_TCMALLOC) && !defined(DEBUG) |