blob: 7113ad839a02e4e61cc028adf608c389e389edea [file] [log] [blame]
mtklein@google.comd36522d2013-10-16 13:02:15 +00001// Main binary for DM.
2// For a high-level overview, please see dm/README.
3
jcgregorio3b27ade2014-11-13 08:06:40 -08004#include "CrashHandler.h"
mtklein72ebb9f2014-08-07 14:27:03 -07005#include "LazyDecodeBitmap.h"
caryclark17f0b6d2014-07-22 10:15:34 -07006#include "SkCommonFlags.h"
mtklein@google.comd36522d2013-10-16 13:02:15 +00007#include "SkForceLinking.h"
8#include "SkGraphics.h"
mtklein1d0f1642014-09-08 08:05:18 -07009#include "SkOSFile.h"
commit-bot@chromium.org90b5a2a2014-05-14 17:55:32 +000010#include "SkPicture.h"
rmistry@google.comd6bab022013-12-02 13:50:38 +000011#include "SkString.h"
mtklein406654b2014-09-03 15:34:37 -070012#include "SkTaskGroup.h"
commit-bot@chromium.org0dc5bd12014-02-26 16:31:22 +000013#include "Test.h"
mtklein@google.comd36522d2013-10-16 13:02:15 +000014#include "gm.h"
Cary Clark992c7b02014-07-31 08:58:44 -040015#include "sk_tool_utils.h"
16#include "sk_tool_utils_flags.h"
mtklein@google.comd36522d2013-10-16 13:02:15 +000017
commit-bot@chromium.orgef57b7e2014-02-28 20:31:31 +000018#include "DMCpuGMTask.h"
19#include "DMGpuGMTask.h"
commit-bot@chromium.org787227d2014-03-26 21:26:15 +000020#include "DMGpuSupport.h"
mtkleinf6139f72014-12-12 16:41:12 -080021#include "DMImageTask.h"
scroggo7a10fb62014-11-04 07:21:10 -080022#include "DMJsonWriter.h"
mtklein30bf3e22014-06-03 13:57:14 -070023#include "DMPDFTask.h"
halcanary9e398f72015-01-10 11:18:04 -080024#include "DMPDFRasterizeTask.h"
mtklein@google.comd36522d2013-10-16 13:02:15 +000025#include "DMReporter.h"
commit-bot@chromium.org90b5a2a2014-05-14 17:55:32 +000026#include "DMSKPTask.h"
mtklein@google.comd36522d2013-10-16 13:02:15 +000027#include "DMTask.h"
28#include "DMTaskRunner.h"
commit-bot@chromium.org0dc5bd12014-02-26 16:31:22 +000029#include "DMTestTask.h"
mtklein@google.comd36522d2013-10-16 13:02:15 +000030
mtklein30bf3e22014-06-03 13:57:14 -070031#ifdef SK_BUILD_POPPLER
32# include "SkPDFRasterizer.h"
33# define RASTERIZE_PDF_PROC SkPopplerRasterizePDF
halcanary3dc5d702014-10-27 06:24:11 -070034#elif defined(SK_BUILD_FOR_MAC) || defined(SK_BUILD_FOR_IOS)
35# include "SkCGUtils.h"
36# define RASTERIZE_PDF_PROC SkPDFDocumentToBitmap
mtklein30bf3e22014-06-03 13:57:14 -070037#else
38# define RASTERIZE_PDF_PROC NULL
39#endif
40
commit-bot@chromium.org120c9992014-05-16 18:11:51 +000041#include <ctype.h>
mtklein@google.comd36522d2013-10-16 13:02:15 +000042
43using skiagm::GM;
44using skiagm::GMRegistry;
commit-bot@chromium.org0dc5bd12014-02-26 16:31:22 +000045using skiatest::Test;
46using skiatest::TestRegistry;
mtklein@google.comd36522d2013-10-16 13:02:15 +000047
kkinnunen80549fc2014-06-30 06:36:31 -070048static const char kGpuAPINameGL[] = "gl";
49static const char kGpuAPINameGLES[] = "gles";
50
commit-bot@chromium.org0dc5bd12014-02-26 16:31:22 +000051DEFINE_bool(gms, true, "Run GMs?");
52DEFINE_bool(tests, true, "Run tests?");
Cary Clark992c7b02014-07-31 08:58:44 -040053DEFINE_bool(reportUsedChars, false, "Output test font construction data to be pasted into"
54 " create_test_font.cpp.");
mtkleinf6139f72014-12-12 16:41:12 -080055DEFINE_string(images, "resources", "Path to directory containing images to decode.");
halcanary9e398f72015-01-10 11:18:04 -080056DEFINE_bool(rasterPDF, true, "Rasterize PDFs?");
mtklein@google.comd36522d2013-10-16 13:02:15 +000057
58__SK_FORCE_IMAGE_DECODER_LINKING;
59
halcanary9e398f72015-01-10 11:18:04 -080060static DM::RasterizePdfProc get_pdf_rasterizer_proc() {
61 return reinterpret_cast<DM::RasterizePdfProc>(
62 FLAGS_rasterPDF ? RASTERIZE_PDF_PROC : NULL);
63}
64
mtklein@google.comd36522d2013-10-16 13:02:15 +000065// "FooBar" -> "foobar". Obviously, ASCII only.
66static SkString lowercase(SkString s) {
67 for (size_t i = 0; i < s.size(); i++) {
68 s[i] = tolower(s[i]);
69 }
70 return s;
71}
72
commit-bot@chromium.org38aeb0f2014-02-26 23:01:57 +000073static const GrContextFactory::GLContextType native = GrContextFactory::kNative_GLContextType;
mtklein60c77072014-08-14 10:36:55 -070074static const GrContextFactory::GLContextType nvpr = GrContextFactory::kNVPR_GLContextType;
commit-bot@chromium.org38aeb0f2014-02-26 23:01:57 +000075static const GrContextFactory::GLContextType null = GrContextFactory::kNull_GLContextType;
76static const GrContextFactory::GLContextType debug = GrContextFactory::kDebug_GLContextType;
commit-bot@chromium.org38aeb0f2014-02-26 23:01:57 +000077#if SK_ANGLE
mtklein60c77072014-08-14 10:36:55 -070078static const GrContextFactory::GLContextType angle = GrContextFactory::kANGLE_GLContextType;
commit-bot@chromium.org38aeb0f2014-02-26 23:01:57 +000079#endif
commit-bot@chromium.org38aeb0f2014-02-26 23:01:57 +000080#if SK_MESA
mtklein60c77072014-08-14 10:36:55 -070081static const GrContextFactory::GLContextType mesa = GrContextFactory::kMESA_GLContextType;
commit-bot@chromium.org38aeb0f2014-02-26 23:01:57 +000082#endif
83
commit-bot@chromium.org0dc5bd12014-02-26 16:31:22 +000084static void kick_off_gms(const SkTDArray<GMRegistry::Factory>& gms,
85 const SkTArray<SkString>& configs,
kkinnunen80549fc2014-06-30 06:36:31 -070086 GrGLStandard gpuAPI,
commit-bot@chromium.org0dc5bd12014-02-26 16:31:22 +000087 DM::Reporter* reporter,
88 DM::TaskRunner* tasks) {
mtkleine4d3e602014-06-06 09:28:43 -070089#define START(name, type, ...) \
90 if (lowercase(configs[j]).equals(name)) { \
91 tasks->add(SkNEW_ARGS(DM::type, (name, reporter, tasks, gms[i], ## __VA_ARGS__))); \
mtklein@google.comd36522d2013-10-16 13:02:15 +000092 }
commit-bot@chromium.org38aeb0f2014-02-26 23:01:57 +000093 for (int i = 0; i < gms.count(); i++) {
mtklein@google.comd36522d2013-10-16 13:02:15 +000094 for (int j = 0; j < configs.count(); j++) {
kkinnunen80549fc2014-06-30 06:36:31 -070095
mtklein197ceda2014-09-09 07:36:57 -070096 START("565", CpuGMTask, kRGB_565_SkColorType);
97 START("8888", CpuGMTask, kN32_SkColorType);
jvanverth4736e142014-11-07 07:12:46 -080098 START("gpu", GpuGMTask, native, gpuAPI, 0, false);
99 START("msaa4", GpuGMTask, native, gpuAPI, 4, false);
100 START("msaa16", GpuGMTask, native, gpuAPI, 16, false);
101 START("nvprmsaa4", GpuGMTask, nvpr, gpuAPI, 4, false);
102 START("nvprmsaa16", GpuGMTask, nvpr, gpuAPI, 16, false);
103 START("gpudft", GpuGMTask, native, gpuAPI, 0, true);
104 START("gpunull", GpuGMTask, null, gpuAPI, 0, false);
105 START("gpudebug", GpuGMTask, debug, gpuAPI, 0, false);
mtklein60c77072014-08-14 10:36:55 -0700106#if SK_ANGLE
jvanverth4736e142014-11-07 07:12:46 -0800107 START("angle", GpuGMTask, angle, gpuAPI, 0, false);
mtklein60c77072014-08-14 10:36:55 -0700108#endif
109#if SK_MESA
jvanverth4736e142014-11-07 07:12:46 -0800110 START("mesa", GpuGMTask, mesa, gpuAPI, 0, false);
mtklein60c77072014-08-14 10:36:55 -0700111#endif
halcanary9e398f72015-01-10 11:18:04 -0800112 START("pdf", PDFTask, get_pdf_rasterizer_proc());
commit-bot@chromium.org38aeb0f2014-02-26 23:01:57 +0000113 }
114 }
115#undef START
116}
117
commit-bot@chromium.org0dc5bd12014-02-26 16:31:22 +0000118static void kick_off_tests(const SkTDArray<TestRegistry::Factory>& tests,
119 DM::Reporter* reporter,
120 DM::TaskRunner* tasks) {
121 for (int i = 0; i < tests.count(); i++) {
commit-bot@chromium.orgef57b7e2014-02-28 20:31:31 +0000122 SkAutoTDelete<Test> test(tests[i](NULL));
123 if (test->isGPUTest()) {
124 tasks->add(SkNEW_ARGS(DM::GpuTestTask, (reporter, tasks, tests[i])));
125 } else {
126 tasks->add(SkNEW_ARGS(DM::CpuTestTask, (reporter, tasks, tests[i])));
127 }
commit-bot@chromium.org0dc5bd12014-02-26 16:31:22 +0000128 }
129}
130
mtkleinf6139f72014-12-12 16:41:12 -0800131static void find_files(const char* dir,
132 const char* suffixes[],
133 size_t num_suffixes,
134 SkTArray<SkString>* files) {
135 if (0 == strcmp(dir, "")) {
commit-bot@chromium.org90b5a2a2014-05-14 17:55:32 +0000136 return;
137 }
138
commit-bot@chromium.org90b5a2a2014-05-14 17:55:32 +0000139 SkString filename;
mtkleinf6139f72014-12-12 16:41:12 -0800140 for (size_t i = 0; i < num_suffixes; i++) {
141 SkOSFile::Iter it(dir, suffixes[i]);
142 while (it.next(&filename)) {
143 if (!SkCommandLineFlags::ShouldSkip(FLAGS_match, filename.c_str())) {
144 files->push_back(SkOSPath::Join(dir, filename.c_str()));
145 }
commit-bot@chromium.org90b5a2a2014-05-14 17:55:32 +0000146 }
halcanarya98683b2014-07-28 07:21:24 -0700147 }
148}
commit-bot@chromium.org90b5a2a2014-05-14 17:55:32 +0000149
halcanarya98683b2014-07-28 07:21:24 -0700150static void kick_off_skps(const SkTArray<SkString>& skps,
mtklein23c94f02014-09-08 09:12:28 -0700151 DM::Reporter* reporter,
152 DM::TaskRunner* tasks) {
halcanarya98683b2014-07-28 07:21:24 -0700153 for (int i = 0; i < skps.count(); ++i) {
154 SkAutoTUnref<SkStream> stream(SkStream::NewFromFile(skps[i].c_str()));
commit-bot@chromium.org90b5a2a2014-05-14 17:55:32 +0000155 if (stream.get() == NULL) {
halcanarya98683b2014-07-28 07:21:24 -0700156 SkDebugf("Could not read %s.\n", skps[i].c_str());
commit-bot@chromium.org90b5a2a2014-05-14 17:55:32 +0000157 exit(1);
158 }
mtklein72ebb9f2014-08-07 14:27:03 -0700159 SkAutoTUnref<SkPicture> pic(
160 SkPicture::CreateFromStream(stream.get(), &sk_tools::LazyDecodeBitmap));
commit-bot@chromium.org90b5a2a2014-05-14 17:55:32 +0000161 if (pic.get() == NULL) {
halcanarya98683b2014-07-28 07:21:24 -0700162 SkDebugf("Could not read %s as an SkPicture.\n", skps[i].c_str());
commit-bot@chromium.org90b5a2a2014-05-14 17:55:32 +0000163 exit(1);
164 }
165
tfarinaa8e2e152014-07-28 19:26:58 -0700166 SkString filename = SkOSPath::Basename(skps[i].c_str());
mtklein197ceda2014-09-09 07:36:57 -0700167 tasks->add(SkNEW_ARGS(DM::SKPTask, (reporter, tasks, pic, filename)));
halcanary9e398f72015-01-10 11:18:04 -0800168 tasks->add(SkNEW_ARGS(DM::PDFTask, (reporter, tasks, pic, filename,
169 get_pdf_rasterizer_proc())));
commit-bot@chromium.org90b5a2a2014-05-14 17:55:32 +0000170 }
171}
172
mtkleinf6139f72014-12-12 16:41:12 -0800173static void kick_off_images(const SkTArray<SkString>& images,
174 DM::Reporter* reporter,
175 DM::TaskRunner* tasks) {
176 for (int i = 0; i < images.count(); i++) {
177 SkAutoTUnref<SkData> image(SkData::NewFromFileName(images[i].c_str()));
178 if (!image) {
179 SkDebugf("Could not read %s.\n", images[i].c_str());
180 exit(1);
181 }
182 SkString filename = SkOSPath::Basename(images[i].c_str());
183 tasks->add(SkNEW_ARGS(DM::ImageTask, (reporter, tasks, image, filename)));
184 tasks->add(SkNEW_ARGS(DM::ImageTask, (reporter, tasks, image, filename, 5/*subsets*/)));
185 }
186}
187
188
commit-bot@chromium.org39e8d932014-05-29 20:14:48 +0000189static void report_failures(const SkTArray<SkString>& failures) {
mtklein@google.comd36522d2013-10-16 13:02:15 +0000190 if (failures.count() == 0) {
191 return;
192 }
193
194 SkDebugf("Failures:\n");
195 for (int i = 0; i < failures.count(); i++) {
196 SkDebugf(" %s\n", failures[i].c_str());
197 }
commit-bot@chromium.org39e8d932014-05-29 20:14:48 +0000198 SkDebugf("%d failures.\n", failures.count());
mtklein@google.comd36522d2013-10-16 13:02:15 +0000199}
200
kkinnunen80549fc2014-06-30 06:36:31 -0700201static GrGLStandard get_gl_standard() {
202 if (FLAGS_gpuAPI.contains(kGpuAPINameGL)) {
203 return kGL_GrGLStandard;
204 }
205 if (FLAGS_gpuAPI.contains(kGpuAPINameGLES)) {
206 return kGLES_GrGLStandard;
207 }
208 return kNone_GrGLStandard;
209}
210
commit-bot@chromium.org38aeb0f2014-02-26 23:01:57 +0000211template <typename T, typename Registry>
212static void append_matching_factories(Registry* head, SkTDArray<typename Registry::Factory>* out) {
213 for (const Registry* reg = head; reg != NULL; reg = reg->next()) {
214 SkAutoTDelete<T> forName(reg->factory()(NULL));
215 if (!SkCommandLineFlags::ShouldSkip(FLAGS_match, forName->getName())) {
216 *out->append() = reg->factory();
217 }
218 }
219}
220
jcgregorio3b27ade2014-11-13 08:06:40 -0800221int dm_main();
caryclark17f0b6d2014-07-22 10:15:34 -0700222int dm_main() {
jcgregorio3b27ade2014-11-13 08:06:40 -0800223 SetupCrashHandler();
commit-bot@chromium.org39e8d932014-05-29 20:14:48 +0000224 SkAutoGraphics ag;
mtklein406654b2014-09-03 15:34:37 -0700225 SkTaskGroup::Enabler enabled(FLAGS_threads);
commit-bot@chromium.orga65e2fd2014-05-30 17:23:31 +0000226
bsalomon06cddec2014-10-24 10:40:50 -0700227 if (FLAGS_dryRun || FLAGS_veryVerbose) {
commit-bot@chromium.orga65e2fd2014-05-30 17:23:31 +0000228 FLAGS_verbose = true;
229 }
commit-bot@chromium.org0dc5bd12014-02-26 16:31:22 +0000230#if SK_ENABLE_INST_COUNT
231 gPrintInstCount = FLAGS_leaks;
232#endif
commit-bot@chromium.org0dc5bd12014-02-26 16:31:22 +0000233
mtklein@google.comd36522d2013-10-16 13:02:15 +0000234 SkTArray<SkString> configs;
commit-bot@chromium.org38aeb0f2014-02-26 23:01:57 +0000235 for (int i = 0; i < FLAGS_config.count(); i++) {
236 SkStrSplit(FLAGS_config[i], ", ", &configs);
237 }
238
kkinnunen80549fc2014-06-30 06:36:31 -0700239 GrGLStandard gpuAPI = get_gl_standard();
240
mtklein@google.comd36522d2013-10-16 13:02:15 +0000241 SkTDArray<GMRegistry::Factory> gms;
commit-bot@chromium.org0dc5bd12014-02-26 16:31:22 +0000242 if (FLAGS_gms) {
commit-bot@chromium.org38aeb0f2014-02-26 23:01:57 +0000243 append_matching_factories<GM>(GMRegistry::Head(), &gms);
mtklein@google.comd36522d2013-10-16 13:02:15 +0000244 }
245
commit-bot@chromium.org38aeb0f2014-02-26 23:01:57 +0000246 SkTDArray<TestRegistry::Factory> tests;
247 if (FLAGS_tests) {
248 append_matching_factories<Test>(TestRegistry::Head(), &tests);
249 }
250
halcanarya98683b2014-07-28 07:21:24 -0700251
mtkleinf6139f72014-12-12 16:41:12 -0800252 SkTArray<SkString> skps;
253 if (!FLAGS_skps.isEmpty()) {
254 const char* suffixes[] = { "skp" };
255 find_files(FLAGS_skps[0], suffixes, SK_ARRAY_COUNT(suffixes), &skps);
256 }
257
258 SkTArray<SkString> images;
259 if (!FLAGS_images.isEmpty()) {
scroggo1d09ee72014-12-15 08:53:35 -0800260 const char* suffixes[] = {
261 "bmp", "gif", "jpg", "jpeg", "png", "webp", "ktx", "astc", "wbmp", "ico",
262 "BMP", "GIF", "JPG", "JPEG", "PNG", "WEBP", "KTX", "ASTC", "WBMP", "ICO",
263 };
mtkleinf6139f72014-12-12 16:41:12 -0800264 find_files(FLAGS_images[0], suffixes, SK_ARRAY_COUNT(suffixes), &images);
265 }
266
267 SkDebugf("%d GMs x %d configs, %d tests, %d pictures, %d images\n",
268 gms.count(), configs.count(), tests.count(), skps.count(), images.count());
mtklein@google.comd36522d2013-10-16 13:02:15 +0000269 DM::Reporter reporter;
mtklein406654b2014-09-03 15:34:37 -0700270
271 DM::TaskRunner tasks;
commit-bot@chromium.org0dc5bd12014-02-26 16:31:22 +0000272 kick_off_tests(tests, &reporter, &tasks);
mtklein197ceda2014-09-09 07:36:57 -0700273 kick_off_gms(gms, configs, gpuAPI, &reporter, &tasks);
274 kick_off_skps(skps, &reporter, &tasks);
mtkleinf6139f72014-12-12 16:41:12 -0800275 kick_off_images(images, &reporter, &tasks);
mtklein@google.comd36522d2013-10-16 13:02:15 +0000276 tasks.wait();
277
scroggo7a10fb62014-11-04 07:21:10 -0800278 DM::JsonWriter::DumpJson();
mtklein1d0f1642014-09-08 08:05:18 -0700279
mtklein@google.comd36522d2013-10-16 13:02:15 +0000280 SkDebugf("\n");
Cary Clark992c7b02014-07-31 08:58:44 -0400281#ifdef SK_DEBUG
282 if (FLAGS_portableFonts && FLAGS_reportUsedChars) {
283 sk_tool_utils::report_used_chars();
284 }
285#endif
mtklein92007582014-08-01 07:46:52 -0700286
commit-bot@chromium.org39e8d932014-05-29 20:14:48 +0000287 SkTArray<SkString> failures;
288 reporter.getFailures(&failures);
289 report_failures(failures);
290 return failures.count() > 0;
mtklein@google.comd36522d2013-10-16 13:02:15 +0000291}
jcgregorio3b27ade2014-11-13 08:06:40 -0800292
293#if !defined(SK_BUILD_FOR_IOS) && !defined(SK_BUILD_FOR_NACL)
294int main(int argc, char** argv) {
295 SkCommandLineFlags::Parse(argc, argv);
296 return dm_main();
297}
298#endif