1 /* 2 * Copyright 2014 Google Inc. 3 * 4 * Use of this source code is governed by a BSD-style license that can be 5 * found in the LICENSE file. 6 */ 7 8 #include "DMJsonWriter.h" 9 10 #include "ProcStats.h" 11 #include "SkCommonFlags.h" 12 #include "SkData.h" 13 #include "SkJSONCPP.h" 14 #include "SkMutex.h" 15 #include "SkOSFile.h" 16 #include "SkOSPath.h" 17 #include "SkStream.h" 18 #include "SkTArray.h" 19 20 namespace DM { 21 22 SkTArray<JsonWriter::BitmapResult> gBitmapResults; 23 SK_DECLARE_STATIC_MUTEX(gBitmapResultLock); 24 25 void JsonWriter::AddBitmapResult(const BitmapResult& result) { 26 SkAutoMutexAcquire lock(&gBitmapResultLock); 27 gBitmapResults.push_back(result); 28 } 29 30 SkTArray<skiatest::Failure> gFailures; 31 SK_DECLARE_STATIC_MUTEX(gFailureLock); 32 33 void JsonWriter::AddTestFailure(const skiatest::Failure& failure) { 34 SkAutoMutexAcquire lock(gFailureLock); 35 gFailures.push_back(failure); 36 } 37 38 void JsonWriter::DumpJson() { 39 if (FLAGS_writePath.isEmpty()) { 40 return; 41 } 42 43 Json::Value root; 44 45 for (int i = 1; i < FLAGS_properties.count(); i += 2) { 46 root[FLAGS_properties[i-1]] = FLAGS_properties[i]; 47 } 48 for (int i = 1; i < FLAGS_key.count(); i += 2) { 49 root["key"][FLAGS_key[i-1]] = FLAGS_key[i]; 50 } 51 52 { 53 SkAutoMutexAcquire lock(&gBitmapResultLock); 54 for (int i = 0; i < gBitmapResults.count(); i++) { 55 Json::Value result; 56 result["key"]["name"] = gBitmapResults[i].name.c_str(); 57 result["key"]["config"] = gBitmapResults[i].config.c_str(); 58 result["key"]["source_type"] = gBitmapResults[i].sourceType.c_str(); 59 result["options"]["ext"] = gBitmapResults[i].ext.c_str(); 60 result["options"]["gamma_correct"] = gBitmapResults[i].gammaCorrect ? "yes" : "no"; 61 result["md5"] = gBitmapResults[i].md5.c_str(); 62 63 // Source options only need to be part of the key if they exist. 64 // Source type by source type, we either always set options or never set options. 65 if (!gBitmapResults[i].sourceOptions.isEmpty()) { 66 result["key"]["source_options"] = gBitmapResults[i].sourceOptions.c_str(); 67 } 68 69 root["results"].append(result); 70 } 71 } 72 73 { 74 SkAutoMutexAcquire lock(gFailureLock); 75 for (int i = 0; i < gFailures.count(); i++) { 76 Json::Value result; 77 result["file_name"] = gFailures[i].fileName; 78 result["line_no"] = gFailures[i].lineNo; 79 result["condition"] = gFailures[i].condition; 80 result["message"] = gFailures[i].message.c_str(); 81 82 root["test_results"]["failures"].append(result); 83 } 84 } 85 86 int maxResidentSetSizeMB = sk_tools::getMaxResidentSetSizeMB(); 87 if (maxResidentSetSizeMB != -1) { 88 root["max_rss_MB"] = sk_tools::getMaxResidentSetSizeMB(); 89 } 90 91 SkString path = SkOSPath::Join(FLAGS_writePath[0], "dm.json"); 92 sk_mkdir(FLAGS_writePath[0]); 93 SkFILEWStream stream(path.c_str()); 94 stream.writeText(Json::StyledWriter().write(root).c_str()); 95 stream.flush(); 96 } 97 98 bool JsonWriter::ReadJson(const char* path, void(*callback)(BitmapResult)) { 99 sk_sp<SkData> json(SkData::MakeFromFileName(path)); 100 if (!json) { 101 return false; 102 } 103 104 Json::Reader reader; 105 Json::Value root; 106 const char* data = (const char*)json->data(); 107 if (!reader.parse(data, data+json->size(), root)) { 108 return false; 109 } 110 111 const Json::Value& results = root["results"]; 112 BitmapResult br; 113 for (unsigned i = 0; i < results.size(); i++) { 114 const Json::Value& r = results[i]; 115 br.name = r["key"]["name"].asCString(); 116 br.config = r["key"]["config"].asCString(); 117 br.sourceType = r["key"]["source_type"].asCString(); 118 br.ext = r["options"]["ext"].asCString(); 119 br.gammaCorrect = 0 == strcmp("yes", r["options"]["gamma_correct"].asCString()); 120 br.md5 = r["md5"].asCString(); 121 122 if (!r["key"]["source_options"].isNull()) { 123 br.sourceOptions = r["key"]["source_options"].asCString(); 124 } 125 callback(br); 126 } 127 return true; 128 } 129 130 } // namespace DM 131