OLD | NEW |
1 /* | 1 /* |
2 * Copyright 2013 Google Inc. | 2 * Copyright 2013 Google Inc. |
3 * | 3 * |
4 * Use of this source code is governed by a BSD-style license that can be | 4 * Use of this source code is governed by a BSD-style license that can be |
5 * found in the LICENSE file. | 5 * found in the LICENSE file. |
6 */ | 6 */ |
7 | 7 |
8 #define __NO_STD_VECTOR // Uses cl::vectpr instead of std::vectpr | 8 #define __NO_STD_VECTOR // Uses cl::vectpr instead of std::vectpr |
9 #define __NO_STD_STRING // Uses cl::STRING_CLASS instead of std::string | 9 #define __NO_STD_STRING // Uses cl::STRING_CLASS instead of std::string |
10 #include <CL/cl.hpp> | 10 #include <CL/cl.hpp> |
11 | 11 |
12 #include "SkCommandLineFlags.h" | 12 #include "SkCommandLineFlags.h" |
13 #include "SkGraphics.h" | 13 #include "SkGraphics.h" |
14 #include "SkPoint.h" | 14 #include "SkStream.h" |
15 #include "SkOSFile.h" | |
16 #include "SkString.h" | |
17 #include "SkTArray.h" | |
18 #include "SkTDArray.h" | 15 #include "SkTDArray.h" |
19 | 16 |
| 17 #include "SkCLImageDiffer.h" |
| 18 #include "SkDiffContext.h" |
20 #include "SkImageDiffer.h" | 19 #include "SkImageDiffer.h" |
21 #include "SkCLImageDiffer.h" | |
22 #include "SkPMetric.h" | 20 #include "SkPMetric.h" |
23 #include "skpdiff_util.h" | 21 #include "skpdiff_util.h" |
24 | 22 |
25 #include "SkForceLinking.h" | 23 #include "SkForceLinking.h" |
26 __SK_FORCE_IMAGE_DECODER_LINKING; | 24 __SK_FORCE_IMAGE_DECODER_LINKING; |
27 | 25 |
28 // Command line argument definitions go here | 26 // Command line argument definitions go here |
29 DEFINE_bool2(list, l, false, "List out available differs"); | 27 DEFINE_bool2(list, l, false, "List out available differs"); |
30 DEFINE_string2(differs, d, "", "The names of the differs to use or all of them b
y default"); | 28 DEFINE_string2(differs, d, "", "The names of the differs to use or all of them b
y default"); |
31 DEFINE_string2(folders, f, "", "Compare two folders with identical subfile names
: <baseline folder> <test folder>"); | 29 DEFINE_string2(folders, f, "", "Compare two folders with identical subfile names
: <baseline folder> <test folder>"); |
32 DEFINE_string2(patterns, p, "", "Use two patterns to compare images: <baseline>
<test>"); | 30 DEFINE_string2(patterns, p, "", "Use two patterns to compare images: <baseline>
<test>"); |
| 31 DEFINE_string2(output, o, "skpdiff_output.json", "Writes the output of these dif
fs to output: <output>"); |
33 | 32 |
34 /// A callback for any OpenCL errors | 33 /// A callback for any OpenCL errors |
35 CL_CALLBACK void error_notify(const char* errorInfo, const void* privateInfoSize
, ::size_t cb, void* userData) { | 34 CL_CALLBACK void error_notify(const char* errorInfo, const void* privateInfoSize
, ::size_t cb, void* userData) { |
36 SkDebugf("OpenCL error notify: %s\n", errorInfo); | 35 SkDebugf("OpenCL error notify: %s\n", errorInfo); |
37 exit(1); | 36 exit(1); |
38 } | 37 } |
39 | 38 |
40 /// Creates a device and context with OpenCL | 39 /// Creates a device and context with OpenCL |
41 static bool init_device_and_context(cl::Device* device, cl::Context* context) { | 40 static bool init_device_and_context(cl::Device* device, cl::Context* context) { |
42 // Query for a platform | 41 // Query for a platform |
(...skipping 22 matching lines...) Expand all Loading... |
65 cl_int contextErr = CL_SUCCESS; | 64 cl_int contextErr = CL_SUCCESS; |
66 *context = cl::Context(deviceList, NULL, error_notify, NULL, &contextErr); | 65 *context = cl::Context(deviceList, NULL, error_notify, NULL, &contextErr); |
67 if (contextErr != CL_SUCCESS) { | 66 if (contextErr != CL_SUCCESS) { |
68 SkDebugf("Context creation failed: %s\n", cl_error_to_string(contextErr)
); | 67 SkDebugf("Context creation failed: %s\n", cl_error_to_string(contextErr)
); |
69 return false; | 68 return false; |
70 } | 69 } |
71 | 70 |
72 return true; | 71 return true; |
73 } | 72 } |
74 | 73 |
75 /// Compares two directories of images with the given differ | |
76 static void diff_directories(const char baselinePath[], const char testPath[], S
kImageDiffer* differ) { | |
77 // Get the files in the baseline, we will then look for those inside the tes
t path | |
78 SkTArray<SkString> baselineEntries; | |
79 if (!get_directory(baselinePath, &baselineEntries)) { | |
80 SkDebugf("Unable to open path \"%s\"\n", baselinePath); | |
81 return; | |
82 } | |
83 | |
84 SkTDArray<int> queuedDiffIDs; | |
85 for (int baselineIndex = 0; baselineIndex < baselineEntries.count(); baselin
eIndex++) { | |
86 const char* baseFilename = baselineEntries[baselineIndex].c_str(); | |
87 SkDebugf("\n%s\n", baseFilename); | |
88 | |
89 // Find the real location of each file to compare | |
90 SkString baselineFile = SkOSPath::SkPathJoin(baselinePath, baseFilename)
; | |
91 SkString testFile = SkOSPath::SkPathJoin(testPath, baseFilename); | |
92 | |
93 // Check that the test file exists and is a file | |
94 if (sk_exists(testFile.c_str()) && !sk_isdir(testFile.c_str())) { | |
95 // Queue up the comparison with the differ | |
96 int diffID = differ->queueDiffOfFile(baselineFile.c_str(), testFile.
c_str()); | |
97 if (diffID >= 0) { | |
98 queuedDiffIDs.push(diffID); | |
99 SkDebugf("Result: %f\n", differ->getResult(diffID)); | |
100 SkDebugf("POI Count: %i\n", differ->getPointsOfInterestCount(dif
fID)); | |
101 differ->deleteDiff(diffID); | |
102 } | |
103 } else { | |
104 SkDebugf("Baseline file \"%s\" has no corresponding test file\n", ba
selineFile.c_str()); | |
105 } | |
106 } | |
107 } | |
108 | |
109 | |
110 /// Compares two sets of images identified by glob style patterns with the given
differ | |
111 static void diff_patterns(const char baselinePattern[], const char testPattern[]
, SkImageDiffer* differ) { | |
112 // Get the files in the baseline and test patterns. Because they are in sort
ed order, it's easy | |
113 // to find corresponding images by matching entry indices. | |
114 | |
115 SkTArray<SkString> baselineEntries; | |
116 if (!glob_files(baselinePattern, &baselineEntries)) { | |
117 SkDebugf("Unable to get pattern \"%s\"\n", baselinePattern); | |
118 return; | |
119 } | |
120 | |
121 SkTArray<SkString> testEntries; | |
122 if (!glob_files(testPattern, &testEntries)) { | |
123 SkDebugf("Unable to get pattern \"%s\"\n", testPattern); | |
124 return; | |
125 } | |
126 | |
127 if (baselineEntries.count() != testEntries.count()) { | |
128 SkDebugf("Baseline and test patterns do not yield corresponding number o
f files\n"); | |
129 return; | |
130 } | |
131 | |
132 SkTDArray<int> queuedDiffIDs; | |
133 for (int entryIndex = 0; entryIndex < baselineEntries.count(); entryIndex++)
{ | |
134 const char* baselineFilename = baselineEntries[entryIndex].c_str(); | |
135 const char* testFilename = testEntries [entryIndex].c_str(); | |
136 SkDebugf("\n%s %s\n", baselineFilename, testFilename); | |
137 | |
138 int diffID = differ->queueDiffOfFile(baselineFilename, testFilename); | |
139 if (diffID >= 0) { | |
140 queuedDiffIDs.push(diffID); | |
141 SkDebugf("Result: %f\n", differ->getResult(diffID)); | |
142 SkDebugf("POI Count: %i\n", differ->getPointsOfInterestCount(diffID)
); | |
143 differ->deleteDiff(diffID); | |
144 } | |
145 } | |
146 } | |
147 | 74 |
148 | 75 |
149 static bool init_cl_diff(SkImageDiffer* differ) { | 76 static bool init_cl_diff(SkImageDiffer* differ) { |
150 // Setup OpenCL | 77 // Setup OpenCL |
151 cl::Device device; | 78 cl::Device device; |
152 cl::Context context; | 79 cl::Context context; |
153 if (!init_device_and_context(&device, &context)) { | 80 if (!init_device_and_context(&device, &context)) { |
154 return false; | 81 return false; |
155 } | 82 } |
156 | 83 |
(...skipping 30 matching lines...) Expand all Loading... |
187 SkCommandLineFlags::Parse(argc, argv); | 114 SkCommandLineFlags::Parse(argc, argv); |
188 | 115 |
189 // Needed by various Skia components | 116 // Needed by various Skia components |
190 SkAutoGraphics ag; | 117 SkAutoGraphics ag; |
191 | 118 |
192 if (FLAGS_list) { | 119 if (FLAGS_list) { |
193 SkDebugf("Available Metrics:\n"); | 120 SkDebugf("Available Metrics:\n"); |
194 } | 121 } |
195 | 122 |
196 // Figure which differs the user chose, and optionally print them if the use
r requests it | 123 // Figure which differs the user chose, and optionally print them if the use
r requests it |
197 SkTDArray<int> chosenDiffers; | 124 SkTDArray<SkImageDiffer*> chosenDiffers; |
198 for (int differIndex = 0; NULL != gDiffers[differIndex]; differIndex++) { | 125 for (int differIndex = 0; NULL != gDiffers[differIndex]; differIndex++) { |
| 126 SkImageDiffer* differ = gDiffers[differIndex]; |
199 if (FLAGS_list) { | 127 if (FLAGS_list) { |
200 SkDebugf(" %s", gDiffers[differIndex]->getName()); | 128 SkDebugf(" %s", differ->getName()); |
201 SkDebugf("\n"); | 129 SkDebugf("\n"); |
202 } | 130 } |
203 | 131 |
204 // Check if this differ was chosen by any of the flags | 132 // Check if this differ was chosen by any of the flags. Initialize them
if they were chosen. |
205 if (FLAGS_differs.isEmpty()) { | 133 if (FLAGS_differs.isEmpty()) { |
206 // If no differs were chosen, they all get added | 134 // If no differs were chosen, they all get added |
207 chosenDiffers.push(differIndex); | 135 chosenDiffers.push(differ); |
| 136 gDiffInits[differIndex](differ); |
208 } else { | 137 } else { |
209 for (int flagIndex = 0; flagIndex < FLAGS_differs.count(); flagIndex
++) { | 138 for (int flagIndex = 0; flagIndex < FLAGS_differs.count(); flagIndex
++) { |
210 if (SkString(FLAGS_differs[flagIndex]).equals(gDiffers[differInd
ex]->getName())) { | 139 if (SkString(FLAGS_differs[flagIndex]).equals(differ->getName())
) { |
211 chosenDiffers.push(differIndex); | 140 chosenDiffers.push(differ); |
| 141 gDiffInits[differIndex](differ); |
212 break; | 142 break; |
213 } | 143 } |
214 } | 144 } |
215 } | 145 } |
216 } | 146 } |
217 | 147 |
218 // Don't attempt to initialize the differ if we aren't going to use it | 148 // Don't attempt to initialize the differ if we aren't going to use it |
219 if (FLAGS_folders.isEmpty() && FLAGS_patterns.isEmpty()) { | 149 if (FLAGS_folders.isEmpty() && FLAGS_patterns.isEmpty()) { |
220 return 0; | 150 return 0; |
221 } | 151 } |
222 | 152 |
223 // Validate command line flags | 153 // Validate command line flags |
224 if (!FLAGS_folders.isEmpty()) { | 154 if (!FLAGS_folders.isEmpty()) { |
225 if (2 != FLAGS_folders.count()) { | 155 if (2 != FLAGS_folders.count()) { |
226 SkDebugf("Folders flag expects two arguments: <baseline folder> <tes
t folder>\n"); | 156 SkDebugf("Folders flag expects two arguments: <baseline folder> <tes
t folder>\n"); |
227 return 1; | 157 return 1; |
228 } | 158 } |
229 } | 159 } |
230 | 160 |
231 if (!FLAGS_patterns.isEmpty()) { | 161 if (!FLAGS_patterns.isEmpty()) { |
232 if (2 != FLAGS_patterns.count()) { | 162 if (2 != FLAGS_patterns.count()) { |
233 SkDebugf("Patterns flag expects two arguments: <baseline pattern> <t
est pattern>\n"); | 163 SkDebugf("Patterns flag expects two arguments: <baseline pattern> <t
est pattern>\n"); |
234 return 1; | 164 return 1; |
235 } | 165 } |
236 } | 166 } |
237 | 167 |
238 // TODO Move the differ loop to after the bitmaps are decoded and/or uploade
d to the OpenCL | 168 SkDiffContext ctx; |
239 // device. Those are often the slowest processes and should not be done more
than once if it can | 169 ctx.setDiffers(chosenDiffers); |
240 // be helped. | |
241 | 170 |
242 // Perform each requested diff | 171 // Perform a folder diff if one is requested |
243 for (int chosenDifferIndex = 0; chosenDifferIndex < chosenDiffers.count(); c
hosenDifferIndex++) { | 172 if (!FLAGS_folders.isEmpty()) { |
244 int differIndex = chosenDiffers[chosenDifferIndex]; | 173 ctx.diffDirectories(FLAGS_folders[0], FLAGS_folders[1]); |
| 174 } |
245 | 175 |
246 // Get the chosen differ and say which one they chose | 176 // Perform a pattern diff if one is requested |
247 SkImageDiffer * differ = gDiffers[differIndex]; | 177 if (!FLAGS_patterns.isEmpty()) { |
248 SkDebugf("Using metric \"%s\"\n", differ->getName()); | 178 ctx.diffPatterns(FLAGS_patterns[0], FLAGS_patterns[1]); |
| 179 } |
249 | 180 |
250 // Initialize the differ using the global list of init functions that ma
tch the list of | 181 // Output to the file specified |
251 // differs | 182 if (!FLAGS_output.isEmpty()) { |
252 gDiffInits[differIndex](differ); | 183 SkFILEWStream outputStream(FLAGS_output[0]); |
253 | 184 ctx.outputRecords(outputStream); |
254 // Perform a folder diff if one is requested | |
255 if (!FLAGS_folders.isEmpty()) { | |
256 diff_directories(FLAGS_folders[0], FLAGS_folders[1], differ); | |
257 } | |
258 | |
259 // Perform a pattern diff if one is requested | |
260 if (!FLAGS_patterns.isEmpty()) { | |
261 diff_patterns(FLAGS_patterns[0], FLAGS_patterns[1], differ); | |
262 } | |
263 } | 185 } |
264 | 186 |
265 return 0; | 187 return 0; |
266 } | 188 } |
OLD | NEW |