blob: c7cf66effafdd9a58732d188d343326b2594d983 [file] [log] [blame]
// Copyright (c) 2012, 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/assert.h"
#include "vm/heap_profiler.h"
#include "vm/growable_array.h"
#include "vm/unit_test.h"
namespace dart {
static void WriteCallback(const void* data, intptr_t length, void* stream) {
GrowableArray<uint8_t>* array =
reinterpret_cast<GrowableArray<uint8_t>*>(stream);
for (intptr_t i = 0; i < length; ++i) {
array->Add(reinterpret_cast<const uint8_t*>(data)[i]);
}
}
static uint8_t Read8(GrowableArray<uint8_t>* array, intptr_t* i) {
EXPECT(array != NULL);
EXPECT(i != NULL);
EXPECT_LE(*i + 1, array->length());
return (*array)[(*i)++];
}
static uint32_t Read32(GrowableArray<uint8_t>* array, intptr_t* i) {
EXPECT(array != NULL);
EXPECT(i != NULL);
EXPECT_LE(*i + 4, array->length());
uint32_t result = 0;
result |= ((*array)[(*i)++] << 0);
result |= ((*array)[(*i)++] << 8);
result |= ((*array)[(*i)++] << 16);
result |= ((*array)[(*i)++] << 24);
return ntohl(result);
}
static bool IsTagValid(uint8_t tag) {
switch (static_cast<HeapProfiler::Tag>(tag)) {
case HeapProfiler::kStringInUtf8:
case HeapProfiler::kLoadClass:
case HeapProfiler::kUnloadClass:
case HeapProfiler::kStackFrame:
case HeapProfiler::kStackTrace:
case HeapProfiler::kAllocSites:
case HeapProfiler::kHeapSummary:
case HeapProfiler::kStartThread:
case HeapProfiler::kEndThread:
case HeapProfiler::kHeapDump:
case HeapProfiler::kCpuSamples:
case HeapProfiler::kControlSettings:
case HeapProfiler::kHeapDumpSummary:
case HeapProfiler::kHeapDumpEnd:
return true;
default:
return false;
}
}
// Write an empty profile. Validate the presence of a header and a
// minimal set of records.
TEST_CASE(HeapProfileEmpty) {
uint64_t before = OS::GetCurrentTimeMillis();
GrowableArray<uint8_t> array;
{
HeapProfiler(WriteCallback, &array);
}
uint64_t after = OS::GetCurrentTimeMillis();
intptr_t i = 0;
EXPECT_LE(i + 19, array.length());
EXPECT_EQ('J', array[i++]);
EXPECT_EQ('A', array[i++]);
EXPECT_EQ('V', array[i++]);
EXPECT_EQ('A', array[i++]);
EXPECT_EQ(' ', array[i++]);
EXPECT_EQ('P', array[i++]);
EXPECT_EQ('R', array[i++]);
EXPECT_EQ('O', array[i++]);
EXPECT_EQ('F', array[i++]);
EXPECT_EQ('I', array[i++]);
EXPECT_EQ('L', array[i++]);
EXPECT_EQ('E', array[i++]);
EXPECT_EQ(' ', array[i++]);
EXPECT_EQ('1', array[i++]);
EXPECT_EQ('.', array[i++]);
EXPECT_EQ('0', array[i++]);
EXPECT_EQ('.', array[i++]);
EXPECT_EQ('1', array[i++]);
EXPECT_EQ('\0', array[i++]);
uint32_t size = Read32(&array, &i);
EXPECT_EQ(8u, size);
uint64_t hi = Read32(&array, &i);
uint64_t lo = Read32(&array, &i);
uint64_t time = (hi << 32) | lo;
EXPECT_LE(before, time);
EXPECT_GE(after, time);
while (i != array.length()) {
// Check tag
uint8_t tag = Read8(&array, &i);
EXPECT(IsTagValid(tag));
// Check time diff
uint32_t time_diff = Read32(&array, &i);
EXPECT_LE(before, time + time_diff);
EXPECT_GE(after, time + time_diff);
// Check length diff
uint32_t length = Read32(&array, &i);
EXPECT_LE((intptr_t)length + i , array.length());
// skip body
i += length;
}
}
} // namespace dart