Index: runtime/vm/coverage.cc |
=================================================================== |
--- runtime/vm/coverage.cc (revision 0) |
+++ runtime/vm/coverage.cc (revision 0) |
@@ -0,0 +1,99 @@ |
+// Copyright (c) 2013, 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/coverage.h" |
+ |
+#include "vm/isolate.h" |
+#include "vm/json_stream.h" |
+#include "vm/object.h" |
+#include "vm/object_store.h" |
+ |
+namespace dart { |
+ |
+DEFINE_FLAG(bool, print_coverage, false, "Print code coverage."); |
+ |
+void CodeCoverage::PrintClass(const Class& cls, const JSONArray& jsarr) { |
+ const Array& functions = Array::Handle(cls.functions()); |
+ ASSERT(!functions.IsNull()); |
+ Function& function = Function::Handle(); |
+ Code& code = Code::Handle(); |
+ Script& script = Script::Handle(); |
+ String& url = String::Handle(); |
+ String& name = String::Handle(); |
+ PcDescriptors& descriptors = PcDescriptors::Handle(); |
+ Array& ic_array = Array::Handle(); |
+ ICData& ic_data = ICData::Handle(); |
+ for (int i = 0; i < functions.Length(); i++) { |
+ function ^= functions.At(i); |
+ if (function.HasCode()) { |
+ JSONObject jsobj(jsarr); |
+ |
+ script = function.script(); |
+ url = script.url(); |
+ name = function.QualifiedUserVisibleName(); |
+ jsobj.AddProperty("source", url.ToCString()); |
+ jsobj.AddProperty("function", name.ToCString()); |
+ |
+ code = function.unoptimized_code(); |
+ ic_array = code.ExtractTypeFeedbackArray(); |
+ descriptors = code.pc_descriptors(); |
+ |
+ JSONArray jsarr(jsobj, "hits"); |
+ for (int j = 0; j < descriptors.Length(); j++) { |
+ PcDescriptors::Kind kind = descriptors.DescriptorKind(j); |
+ // Only IC based calls have counting. |
+ if ((kind == PcDescriptors::kIcCall) || |
+ (kind == PcDescriptors::kUnoptStaticCall)) { |
+ intptr_t deopt_id = descriptors.DeoptId(j); |
+ ic_data ^= ic_array.At(deopt_id); |
+ if (!ic_data.IsNull() && (ic_data.AggregateCount() > 0)) { |
+ intptr_t token_pos = descriptors.TokenPos(j); |
+ intptr_t line = -1; |
+ intptr_t col = -1; |
+ script.GetTokenLocation(token_pos, &line, &col); |
+ // OS::Print(" %s -> %"Pd" @ %"Pd" %"Pd"\n", |
+ // ic_data.ToCString(), |
+ // ic_data.AggregateCount(), |
+ // token_pos, line); |
+ JSONObject ic_info(jsarr); |
+ ic_info.AddProperty("line", line); |
+ ic_info.AddProperty("col", col); |
+ ic_info.AddProperty("count", ic_data.AggregateCount()); |
+ } |
+ } |
+ } |
+ } |
+ } |
+} |
+ |
+ |
+void CodeCoverage::Print(Isolate* isolate) { |
+ JSONStream stream; |
+ |
+ { |
+ const GrowableObjectArray& libs = GrowableObjectArray::Handle( |
+ isolate, isolate->object_store()->libraries()); |
+ Library& lib = Library::Handle(); |
+ Class& cls = Class::Handle(); |
+ JSONArray jsarr(&stream); |
+ for (int i = 0; i < libs.Length(); i++) { |
+ lib ^= libs.At(i); |
+ ClassDictionaryIterator it(lib, ClassDictionaryIterator::kIteratePrivate); |
+ while (it.HasNext()) { |
+ cls = it.GetNextClass(); |
+ if (cls.is_finalized()) { |
+ // Only classes that have been finalized do have a meaningful list of |
+ // functions. |
+ PrintClass(cls, jsarr); |
+ } |
+ } |
+ } |
+ } |
+ |
+ OS::Print("### COVERAGE DATA ###\n" |
+ "%s\n" |
+ "### END ###\n", stream.ToCString()); |
+} |
+ |
+} // namespace dart |