 Chromium Code Reviews
 Chromium Code Reviews Issue 83863002:
  Add JSON output option to bench.  (Closed) 
  Base URL: http://skia.googlecode.com/svn/trunk/
    
  
    Issue 83863002:
  Add JSON output option to bench.  (Closed) 
  Base URL: http://skia.googlecode.com/svn/trunk/| Index: bench/ResultsWriter.h | 
| =================================================================== | 
| --- bench/ResultsWriter.h (revision 0) | 
| +++ bench/ResultsWriter.h (revision 0) | 
| @@ -0,0 +1,186 @@ | 
| +/* | 
| + * Copyright 2013 Google Inc. | 
| + * | 
| + * Use of this source code is governed by a BSD-style license that can be | 
| + * found in the LICENSE file. | 
| + * | 
| + * Classes for writing out bench results in various formats. | 
| + */ | 
| +#ifndef SkResultsWriter_DEFINED | 
| +#define SkResultsWriter_DEFINED | 
| + | 
| +#include "SkBenchLogger.h" | 
| +#include "SkJSONCPP.h" | 
| +#include "SkStream.h" | 
| +#include "SkString.h" | 
| +#include "SkTArray.h" | 
| +#include "SkTypes.h" | 
| + | 
| + | 
| +/** | 
| + * Base class for writing out the bench results. | 
| + * | 
| + * TODO(jcgregorio) Add info if tests fail to converge? | 
| + */ | 
| +class ResultsWriter : SkNoncopyable { | 
| +public: | 
| + virtual ~ResultsWriter() {}; | 
| + | 
| + // Records one option set for this run. All options must be set before | 
| + // calling bench(). | 
| + virtual void option(const char name[], const char value[]) = 0; | 
| + | 
| + // Denotes the start of a specific benchmark. Once bench is called, | 
| + // then config and timer can be called multiple times to record runs. | 
| + virtual void bench(const char name[], int32_t x, int32_t y) = 0; | 
| + | 
| + // Records the specific configuration a bench is run under, such as "8888". | 
| + virtual void config(const char name[]) = 0; | 
| + | 
| + // Records a single test metric. | 
| + virtual void timer(const char name[], double ms) = 0; | 
| + | 
| + // Call when all results are finished. | 
| + virtual void end() = 0; | 
| +}; | 
| + | 
| +/** | 
| + * This ResultsWriter handles writing out the human readable format of the | 
| + * bench results. | 
| + */ | 
| +class LoggerResultsWriter : public ResultsWriter { | 
| +public: | 
| + explicit LoggerResultsWriter(SkBenchLogger& logger, const char* timeFormat) | 
| + : fLogger(logger) | 
| + , fTimeFormat(timeFormat) { | 
| + fLogger.logProgress("skia bench:"); | 
| + } | 
| + virtual void option(const char name[], const char value[]) { | 
| + fLogger.logProgress(SkStringPrintf(" %s=%s", name, value)); | 
| + } | 
| + virtual void bench(const char name[], int32_t x, int32_t y) { | 
| + fLogger.logProgress(SkStringPrintf( | 
| + "\nrunning bench [%3d %3d] %40s", x, y, name)); | 
| 
jcgregorio
2013/12/03 14:21:10
%040s produces 0-padded strings on a Mac:
http://
 | 
| + } | 
| + virtual void config(const char name[]) { | 
| + fLogger.logProgress(SkStringPrintf(" %s:", name)); | 
| + } | 
| + virtual void timer(const char name[], double ms) { | 
| + fLogger.logProgress(SkStringPrintf(" %s = ", name)); | 
| + fLogger.logProgress(SkStringPrintf(fTimeFormat, ms)); | 
| + } | 
| + virtual void end() { | 
| + fLogger.logProgress("\n"); | 
| + } | 
| +private: | 
| + SkBenchLogger& fLogger; | 
| + const char* fTimeFormat; | 
| +}; | 
| + | 
| +/** | 
| + * This ResultsWriter handles writing out the results in JSON. | 
| + * | 
| + * The output looks like: | 
| + * | 
| + * { | 
| + * "options" : { | 
| + * "alpha" : "0xFF", | 
| + * "scale" : "0", | 
| + * ... | 
| + * "system" : "UNIX" | 
| + * }, | 
| + * "results" : { | 
| + * "Xfermode_Luminosity_640_480" : { | 
| + * "565" : { | 
| + * "cmsecs" : 143.188128906250, | 
| + * "msecs" : 143.835957031250 | 
| + * }, | 
| + * ... | 
| + */ | 
| +class JSONResultsWriter : public ResultsWriter { | 
| +public: | 
| + explicit JSONResultsWriter(const char filename[]) | 
| + : fFilename(filename) | 
| + , fRoot() | 
| + , fResults(fRoot["results"]) | 
| + , fBench(NULL) | 
| + , fConfig(NULL) { | 
| + } | 
| + virtual void option(const char name[], const char value[]) { | 
| + fRoot["options"][name] = value; | 
| + } | 
| + virtual void bench(const char name[], int32_t x, int32_t y) { | 
| + fBench = &fResults[SkStringPrintf( "%s_%d_%d", name, x, y).c_str()]; | 
| + } | 
| + virtual void config(const char name[]) { | 
| + SkASSERT(NULL != fBench); | 
| + fConfig = &(*fBench)[name]; | 
| + } | 
| + virtual void timer(const char name[], double ms) { | 
| + SkASSERT(NULL != fConfig); | 
| + (*fConfig)[name] = ms; | 
| + } | 
| + virtual void end() { | 
| + SkFILEWStream stream(fFilename.c_str()); | 
| + stream.writeText(fRoot.toStyledString().c_str()); | 
| + stream.flush(); | 
| + } | 
| +private: | 
| + SkString fFilename; | 
| + Json::Value fRoot; | 
| + Json::Value& fResults; | 
| + Json::Value* fBench; | 
| + Json::Value* fConfig; | 
| +}; | 
| + | 
| +/** | 
| + * This ResultsWriter writes out to multiple ResultsWriters. | 
| + */ | 
| +class MultiResultsWriter : public ResultsWriter { | 
| +public: | 
| + MultiResultsWriter() : writers() { | 
| + }; | 
| + void add(ResultsWriter* writer) { | 
| + writers.push_back(writer); | 
| + } | 
| + virtual void option(const char name[], const char value[]) { | 
| + for (int i = 0; i < writers.count(); ++i) { | 
| + writers[i]->option(name, value); | 
| + } | 
| + } | 
| + virtual void bench(const char name[], int32_t x, int32_t y) { | 
| + for (int i = 0; i < writers.count(); ++i) { | 
| + writers[i]->bench(name, x, y); | 
| + } | 
| + } | 
| + virtual void config(const char name[]) { | 
| + for (int i = 0; i < writers.count(); ++i) { | 
| + writers[i]->config(name); | 
| + } | 
| + } | 
| + virtual void timer(const char name[], double ms) { | 
| + for (int i = 0; i < writers.count(); ++i) { | 
| + writers[i]->timer(name, ms); | 
| + } | 
| + } | 
| + virtual void end() { | 
| + for (int i = 0; i < writers.count(); ++i) { | 
| + writers[i]->end(); | 
| + } | 
| + } | 
| +private: | 
| + SkTArray<ResultsWriter *> writers; | 
| +}; | 
| + | 
| +/** | 
| + * Calls the end() method of T on destruction. | 
| + */ | 
| +template <typename T> class CallEnd : SkNoncopyable { | 
| +public: | 
| + CallEnd(T& obj) : fObj(obj) {} | 
| + ~CallEnd() { fObj.end(); } | 
| +private: | 
| + T& fObj; | 
| +}; | 
| + | 
| +#endif |