| /* |
| * Copyright 2012 Google Inc. |
| * |
| * Use of this source code is governed by a BSD-style license that can be |
| * found in the LICENSE file. |
| */ |
| |
| #ifndef skdiff_DEFINED |
| #define skdiff_DEFINED |
| |
| #include "SkBitmap.h" |
| #include "SkColor.h" |
| #include "SkColorPriv.h" |
| #include "SkString.h" |
| #include "SkTDArray.h" |
| |
| #if SK_BUILD_FOR_WIN32 |
| #define PATH_DIV_STR "\\" |
| #define PATH_DIV_CHAR '\\' |
| #else |
| #define PATH_DIV_STR "/" |
| #define PATH_DIV_CHAR '/' |
| #endif |
| |
| #define MAX2(a,b) (((b) < (a)) ? (a) : (b)) |
| #define MAX3(a,b,c) (((b) < (a)) ? MAX2((a), (c)) : MAX2((b), (c))) |
| |
| |
| struct DiffResource { |
| enum Status { |
| /** The resource was specified, exists, read, and decoded. */ |
| kDecoded_Status, |
| /** The resource was specified, exists, read, but could not be decoded. */ |
| kCouldNotDecode_Status, |
| |
| /** The resource was specified, exists, and read. */ |
| kRead_Status, |
| /** The resource was specified, exists, but could not be read. */ |
| kCouldNotRead_Status, |
| |
| /** The resource was specified and exists. */ |
| kExists_Status, |
| /** The resource was specified, but does not exist. */ |
| kDoesNotExist_Status, |
| |
| /** The resource was specified. */ |
| kSpecified_Status, |
| /** The resource was not specified. */ |
| kUnspecified_Status, |
| |
| /** Nothing is yet known about the resource. */ |
| kUnknown_Status, |
| |
| /** NOT A VALID VALUE -- used to set up arrays and to represent an unknown value. */ |
| kStatusCount |
| }; |
| static char const * const StatusNames[DiffResource::kStatusCount]; |
| |
| /** Returns the Status with this name. |
| * If there is no Status with this name, returns kStatusCount. |
| */ |
| static Status getStatusByName(const char *name); |
| |
| /** Returns a text description of the given Status type. */ |
| static const char *getStatusDescription(Status status); |
| |
| /** Returns true if the Status indicates some kind of failure. */ |
| static bool isStatusFailed(Status status); |
| |
| /** Sets statuses[i] if it is implied by selector, unsets it if not. |
| * Selector may be a comma delimited list of status names, "any", or "failed". |
| * Returns true if the selector was entirely understood, false otherwise. |
| */ |
| static bool getMatchingStatuses(char* selector, bool statuses[kStatusCount]); |
| |
| DiffResource() : fFilename(), fFullPath(), fBitmap(), fStatus(kUnknown_Status) { }; |
| |
| /** If isEmpty() indicates no filename available. */ |
| SkString fFilename; |
| /** If isEmpty() indicates no path available. */ |
| SkString fFullPath; |
| /** If empty() indicates the bitmap could not be created. */ |
| SkBitmap fBitmap; |
| Status fStatus; |
| }; |
| |
| struct DiffRecord { |
| |
| // Result of comparison for each pair of files. |
| // Listed from "better" to "worse", for sorting of results. |
| enum Result { |
| kEqualBits_Result, |
| kEqualPixels_Result, |
| kDifferentPixels_Result, |
| kDifferentSizes_Result, |
| kCouldNotCompare_Result, |
| kUnknown_Result, |
| |
| kResultCount // NOT A VALID VALUE--used to set up arrays. Must be last. |
| }; |
| static char const * const ResultNames[DiffRecord::kResultCount]; |
| |
| /** Returns the Result with this name. |
| * If there is no Result with this name, returns kResultCount. |
| */ |
| static Result getResultByName(const char *name); |
| |
| /** Returns a text description of the given Result type. */ |
| static const char *getResultDescription(Result result); |
| |
| DiffRecord() |
| : fBase() |
| , fComparison() |
| , fDifference() |
| , fWhite() |
| , fFractionDifference(0) |
| , fWeightedFraction(0) |
| , fAverageMismatchR(0) |
| , fAverageMismatchG(0) |
| , fAverageMismatchB(0) |
| , fMaxMismatchR(0) |
| , fMaxMismatchG(0) |
| , fMaxMismatchB(0) |
| , fResult(kUnknown_Result) { |
| }; |
| |
| DiffResource fBase; |
| DiffResource fComparison; |
| DiffResource fDifference; |
| DiffResource fWhite; |
| |
| /// Arbitrary floating-point metric to be used to sort images from most |
| /// to least different from baseline; values of 0 will be omitted from the |
| /// summary webpage. |
| float fFractionDifference; |
| float fWeightedFraction; |
| |
| float fAverageMismatchR; |
| float fAverageMismatchG; |
| float fAverageMismatchB; |
| |
| uint32_t fMaxMismatchR; |
| uint32_t fMaxMismatchG; |
| uint32_t fMaxMismatchB; |
| |
| /// Which category of diff result. |
| Result fResult; |
| }; |
| |
| typedef SkTDArray<DiffRecord*> RecordArray; |
| |
| /// A wrapper for any sortProc (comparison routine) which applies a first-order |
| /// sort beforehand, and a tiebreaker if the sortProc returns 0. |
| template<typename T> static int compare(const void* untyped_lhs, const void* untyped_rhs) { |
| const DiffRecord* lhs = *reinterpret_cast<DiffRecord* const *>(untyped_lhs); |
| const DiffRecord* rhs = *reinterpret_cast<DiffRecord* const *>(untyped_rhs); |
| |
| // First-order sort... these comparisons should be applied before comparing |
| // pixel values, no matter what. |
| if (lhs->fResult != rhs->fResult) { |
| return (lhs->fResult < rhs->fResult) ? 1 : -1; |
| } |
| |
| // Passed first-order sort, so call the pixel comparison routine. |
| int result = T::comparePixels(lhs, rhs); |
| if (result != 0) { |
| return result; |
| } |
| |
| // Tiebreaker... if we got to this point, we don't really care |
| // which order they are sorted in, but let's at least be consistent. |
| return strcmp(lhs->fBase.fFilename.c_str(), rhs->fBase.fFilename.c_str()); |
| } |
| |
| /// Comparison routine for qsort; sorts by fFractionDifference |
| /// from largest to smallest. |
| class CompareDiffMetrics { |
| public: |
| static int comparePixels(const DiffRecord* lhs, const DiffRecord* rhs) { |
| if (lhs->fFractionDifference < rhs->fFractionDifference) { |
| return 1; |
| } |
| if (rhs->fFractionDifference < lhs->fFractionDifference) { |
| return -1; |
| } |
| return 0; |
| } |
| }; |
| |
| class CompareDiffWeighted { |
| public: |
| static int comparePixels(const DiffRecord* lhs, const DiffRecord* rhs) { |
| if (lhs->fWeightedFraction < rhs->fWeightedFraction) { |
| return 1; |
| } |
| if (lhs->fWeightedFraction > rhs->fWeightedFraction) { |
| return -1; |
| } |
| return 0; |
| } |
| }; |
| |
| /// Comparison routine for qsort; sorts by max(fAverageMismatch{RGB}) |
| /// from largest to smallest. |
| class CompareDiffMeanMismatches { |
| public: |
| static int comparePixels(const DiffRecord* lhs, const DiffRecord* rhs) { |
| float leftValue = MAX3(lhs->fAverageMismatchR, |
| lhs->fAverageMismatchG, |
| lhs->fAverageMismatchB); |
| float rightValue = MAX3(rhs->fAverageMismatchR, |
| rhs->fAverageMismatchG, |
| rhs->fAverageMismatchB); |
| if (leftValue < rightValue) { |
| return 1; |
| } |
| if (rightValue < leftValue) { |
| return -1; |
| } |
| return 0; |
| } |
| }; |
| |
| /// Comparison routine for qsort; sorts by max(fMaxMismatch{RGB}) |
| /// from largest to smallest. |
| class CompareDiffMaxMismatches { |
| public: |
| static int comparePixels(const DiffRecord* lhs, const DiffRecord* rhs) { |
| uint32_t leftValue = MAX3(lhs->fMaxMismatchR, |
| lhs->fMaxMismatchG, |
| lhs->fMaxMismatchB); |
| uint32_t rightValue = MAX3(rhs->fMaxMismatchR, |
| rhs->fMaxMismatchG, |
| rhs->fMaxMismatchB); |
| if (leftValue < rightValue) { |
| return 1; |
| } |
| if (rightValue < leftValue) { |
| return -1; |
| } |
| |
| return CompareDiffMeanMismatches::comparePixels(lhs, rhs); |
| } |
| }; |
| |
| |
| /// Parameterized routine to compute the color of a pixel in a difference image. |
| typedef SkPMColor (*DiffMetricProc)(SkPMColor, SkPMColor); |
| |
| // from gm |
| static inline SkPMColor compute_diff_pmcolor(SkPMColor c0, SkPMColor c1) { |
| int dr = SkGetPackedR32(c0) - SkGetPackedR32(c1); |
| int dg = SkGetPackedG32(c0) - SkGetPackedG32(c1); |
| int db = SkGetPackedB32(c0) - SkGetPackedB32(c1); |
| |
| return SkPackARGB32(0xFF, SkAbs32(dr), SkAbs32(dg), SkAbs32(db)); |
| } |
| |
| /** When finished, dr->fResult should have some value other than kUnknown_Result. |
| * Expects dr->fWhite.fBitmap and dr->fDifference.fBitmap to have the same bounds as |
| * dr->fBase.fBitmap and have a valid pixelref. |
| */ |
| void compute_diff(DiffRecord* dr, DiffMetricProc diffFunction, const int colorThreshold); |
| |
| #endif |