blob: f1cd1e0901e42b98e0cc9f0456ab3f66cee85238 [file] [log] [blame]
// Copyright (c) 2014, 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 "vm/tags.h"
#include "vm/isolate.h"
#include "vm/json_stream.h"
#include "vm/native_entry.h"
#include "vm/object.h"
#include "vm/runtime_entry.h"
namespace dart {
const char* VMTag::TagName(uword tag) {
if (IsNativeEntryTag(tag)) {
const uint8_t* native_reverse_lookup = NativeEntry::ResolveSymbol(tag);
if (native_reverse_lookup != NULL) {
return reinterpret_cast<const char*>(native_reverse_lookup);
}
return "Unknown native entry";
} else if (IsRuntimeEntryTag(tag)) {
const char* runtime_entry_name = RuntimeEntryTagName(tag);
ASSERT(runtime_entry_name != NULL);
return runtime_entry_name;
}
ASSERT(tag != kInvalidTagId);
ASSERT(tag < kNumVMTags);
const TagEntry& entry = entries_[tag];
ASSERT(entry.id == tag);
return entry.name;
}
bool VMTag::IsNativeEntryTag(uword tag) {
return (tag > kLastTagId) && !IsRuntimeEntryTag(tag);
}
bool VMTag::IsDartTag(uword id) {
return id == kDartTagId;
}
bool VMTag::IsExitFrameTag(uword id) {
return (id != 0) && !IsDartTag(id) && (id != kIdleTagId) &&
(id != kVMTagId) && (id != kEmbedderTagId);
}
static RuntimeEntry* runtime_entry_list = NULL;
bool VMTag::IsRuntimeEntryTag(uword id) {
const RuntimeEntry* current = runtime_entry_list;
while (current != NULL) {
if (reinterpret_cast<uword>(current->function()) == id) {
return true;
}
current = current->next();
}
return false;
}
const char* VMTag::RuntimeEntryTagName(uword id) {
const RuntimeEntry* current = runtime_entry_list;
while (current != NULL) {
if (reinterpret_cast<uword>(current->function()) == id) {
return current->name();
}
current = current->next();
}
return NULL;
}
void VMTag::RegisterRuntimeEntry(RuntimeEntry* runtime_entry) {
ASSERT(runtime_entry != NULL);
runtime_entry->set_next(runtime_entry_list);
runtime_entry_list = runtime_entry;
}
VMTag::TagEntry VMTag::entries_[] = {
{
"InvalidTag", kInvalidTagId,
},
#define DEFINE_VM_TAG_ENTRY(tag) {"" #tag, k##tag##TagId},
VM_TAG_LIST(DEFINE_VM_TAG_ENTRY)
#undef DEFINE_VM_TAG_ENTRY
{"kNumVMTags", kNumVMTags},
};
VMTagScope::VMTagScope(Thread* thread, uword tag, bool conditional_set)
: StackResource(thread) {
ASSERT(isolate() != NULL);
previous_tag_ = thread->vm_tag();
if (conditional_set) {
thread->set_vm_tag(tag);
}
}
VMTagScope::~VMTagScope() {
ASSERT(isolate() != NULL);
thread()->set_vm_tag(previous_tag_);
}
VMTagCounters::VMTagCounters() {
for (intptr_t i = 0; i < VMTag::kNumVMTags; i++) {
counters_[i] = 0;
}
}
void VMTagCounters::Increment(uword tag) {
if (VMTag::IsRuntimeEntryTag(tag)) {
counters_[VMTag::kRuntimeTagId]++;
return;
} else if (tag > VMTag::kNumVMTags) {
// Assume native entry.
counters_[VMTag::kNativeTagId]++;
return;
}
ASSERT(tag != VMTag::kInvalidTagId);
ASSERT(tag < VMTag::kNumVMTags);
counters_[tag]++;
}
int64_t VMTagCounters::count(uword tag) {
ASSERT(tag != VMTag::kInvalidTagId);
ASSERT(tag < VMTag::kNumVMTags);
return counters_[tag];
}
#ifndef PRODUCT
void VMTagCounters::PrintToJSONObject(JSONObject* obj) {
if (!FLAG_support_service) {
return;
}
{
JSONArray arr(obj, "names");
for (intptr_t i = 1; i < VMTag::kNumVMTags; i++) {
arr.AddValue(VMTag::TagName(i));
}
}
{
JSONArray arr(obj, "counters");
for (intptr_t i = 1; i < VMTag::kNumVMTags; i++) {
arr.AddValue64(counters_[i]);
}
}
}
#endif // !PRODUCT
const char* UserTags::TagName(uword tag_id) {
ASSERT(tag_id >= kUserTagIdOffset);
ASSERT(tag_id < kUserTagIdOffset + kMaxUserTags);
Zone* zone = Thread::Current()->zone();
const UserTag& tag = UserTag::Handle(zone, UserTag::FindTagById(tag_id));
ASSERT(!tag.IsNull());
const String& label = String::Handle(zone, tag.label());
return label.ToCString();
}
} // namespace dart