blob: 5572b286d3c70f3bca02e53531aab292e165036d [file] [log] [blame]
junov@chromium.org777442d2012-06-12 14:56:36 +00001/*
2 * Copyright 2012 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
9#include "SkBitmap.h"
keyar@chromium.orga2333d92012-07-16 17:29:16 +000010#include "SamplePipeControllers.h"
junov@chromium.org777442d2012-06-12 14:56:36 +000011#include "SkCanvas.h"
12#include "SkColorPriv.h"
keyar@chromium.org472b3792012-07-20 22:34:27 +000013#include "SkDevice.h"
keyar@chromium.orga2333d92012-07-16 17:29:16 +000014#include "SkGPipe.h"
junov@chromium.org777442d2012-06-12 14:56:36 +000015#include "SkImageEncoder.h"
16#include "SkOSFile.h"
17#include "SkPicture.h"
18#include "SkStream.h"
19#include "SkString.h"
senorblanco@chromium.org3cbbb542012-07-13 18:55:53 +000020#include "SkTArray.h"
keyar@chromium.org472b3792012-07-20 22:34:27 +000021#include "SkTDArray.h"
twiz@google.coma31b8bb2012-06-22 18:24:56 +000022#include "picture_utils.h"
junov@chromium.org777442d2012-06-12 14:56:36 +000023
keyar@chromium.org472b3792012-07-20 22:34:27 +000024enum {
25 kDefaultTileWidth = 256,
26 kDefaultTileHeight = 256
27};
junov@chromium.org777442d2012-06-12 14:56:36 +000028
29static void usage(const char* argv0) {
30 SkDebugf("SkPicture rendering tool\n");
31 SkDebugf("\n"
32"Usage: \n"
keyar@chromium.org472b3792012-07-20 22:34:27 +000033" %s <input>... <outputDir> \n"
34" [--pipe | --tile]"
junov@chromium.org777442d2012-06-12 14:56:36 +000035, argv0);
keyar@chromium.org472b3792012-07-20 22:34:27 +000036 SkDebugf("\n\n");
junov@chromium.org777442d2012-06-12 14:56:36 +000037 SkDebugf(
keyar@chromium.orgd1dc9202012-07-09 18:32:08 +000038" input: A list of directories and files to use as input.\n"
39" Files are expected to have the .skp extension.\n");
junov@chromium.org777442d2012-06-12 14:56:36 +000040 SkDebugf(
41" outputDir: directory to write the rendered images.\n");
keyar@chromium.org472b3792012-07-20 22:34:27 +000042 SkDebugf(
43" --pipe : Render using a SkGPipe\n");
44 SkDebugf(
45" --tile : Render using tiles.\n");
junov@chromium.org777442d2012-06-12 14:56:36 +000046}
47
keyar@chromium.org472b3792012-07-20 22:34:27 +000048static void simple_render(SkPicture* pict, SkBitmap* bitmap);
49typedef void (*RenderFunc) (SkPicture*, SkBitmap*);
50
keyar@chromium.org1cbd47c2012-07-13 18:22:59 +000051static void make_output_filepath(SkString* path, const SkString& dir,
junov@chromium.org777442d2012-06-12 14:56:36 +000052 const SkString& name) {
twiz@google.coma31b8bb2012-06-22 18:24:56 +000053 sk_tools::make_filepath(path, dir, name);
junov@chromium.org777442d2012-06-12 14:56:36 +000054 path->remove(path->size() - 3, 3);
55 path->append("png");
56}
57
keyar@chromium.org1cbd47c2012-07-13 18:22:59 +000058static void simple_render(SkPicture* pict, SkBitmap* bitmap) {
59 sk_tools::setup_bitmap(bitmap, pict->width(), pict->height());
junov@chromium.org777442d2012-06-12 14:56:36 +000060 SkCanvas canvas(*bitmap);
keyar@chromium.org1cbd47c2012-07-13 18:22:59 +000061 canvas.drawPicture(*pict);
junov@chromium.org777442d2012-06-12 14:56:36 +000062}
63
64/* since PNG insists on unpremultiplying our alpha, we take no precision chances
65 and force all pixels to be 100% opaque, otherwise on compare we may not get
66 a perfect match.
67 */
68static void force_all_opaque(const SkBitmap& bitmap) {
69 SkAutoLockPixels lock(bitmap);
70 for (int y = 0; y < bitmap.height(); y++) {
71 for (int x = 0; x < bitmap.width(); x++) {
72 *bitmap.getAddr32(x, y) |= (SK_A32_MASK << SK_A32_SHIFT);
73 }
74 }
75}
76
77static bool write_bitmap(const SkString& path, const SkBitmap& bitmap) {
78 SkBitmap copy;
79 bitmap.copyTo(&copy, SkBitmap::kARGB_8888_Config);
80 force_all_opaque(copy);
81 return SkImageEncoder::EncodeFile(path.c_str(), copy,
82 SkImageEncoder::kPNG_Type, 100);
83}
84
keyar@chromium.org1cbd47c2012-07-13 18:22:59 +000085static void write_output(const SkString& outputDir, const SkString& inputFilename,
junov@chromium.org777442d2012-06-12 14:56:36 +000086 const SkBitmap& bitmap) {
87 SkString outputPath;
88 make_output_filepath(&outputPath, outputDir, inputFilename);
89 bool isWritten = write_bitmap(outputPath, bitmap);
90 if (!isWritten) {
91 SkDebugf("Could not write to file %s\n", outputPath.c_str());
92 }
93}
94
keyar@chromium.orga2333d92012-07-16 17:29:16 +000095static void pipe_run(SkPicture* picture, SkCanvas* canvas) {
96 PipeController pipeController(canvas);
97 SkGPipeWriter writer;
98 SkCanvas* pipeCanvas = writer.startRecording(&pipeController);
99 pipeCanvas->drawPicture(*picture);
100 writer.endRecording();
101}
102
103static void pipe_render(SkPicture* picture, SkBitmap* bitmap) {
104 sk_tools::setup_bitmap(bitmap, picture->width(), picture->height());
105
106 SkCanvas canvas(*bitmap);
107
108 pipe_run(picture, &canvas);
109}
110
keyar@chromium.org472b3792012-07-20 22:34:27 +0000111struct TileInfo {
112 SkBitmap* fBitmap;
113 SkCanvas* fCanvas;
114};
115
116// Clips the tile to an area that is completely in what the SkPicture says is the
117// drawn-to area. This is mostly important for tiles on the right and bottom edges
118// as they may go over this area and the picture may have some commands that
119// draw outside of this area and so should not actually be written.
120static void clip_tile(SkPicture* picture, const TileInfo& tile) {
121 SkRect clip = SkRect::MakeWH(SkIntToScalar(picture->width()),
122 SkIntToScalar(picture->height()));
123 tile.fCanvas->clipRect(clip);
124}
125
126static void add_tile(SkPicture* picture, SkTDArray<TileInfo>* tiles,
127 int tile_x_start, int tile_y_start) {
128 TileInfo* tile = tiles->push();
129
130 tile->fBitmap = new SkBitmap();
131 sk_tools::setup_bitmap(tile->fBitmap, kDefaultTileWidth, kDefaultTileHeight);
132
133 tile->fCanvas = new SkCanvas(*(tile->fBitmap));
134 tile->fCanvas->translate(SkIntToScalar(-tile_x_start), SkIntToScalar(-tile_y_start));
135 clip_tile(picture, *tile);
136}
137
138static void setup_tiles(SkPicture* picture, SkTDArray<TileInfo>* tiles) {
139 for (int tile_y_start = 0; tile_y_start < picture->height();
140 tile_y_start += kDefaultTileWidth) {
141 for (int tile_x_start = 0; tile_x_start < picture->width();
142 tile_x_start += kDefaultTileHeight) {
143 add_tile(picture, tiles, tile_x_start, tile_y_start);
144 }
145 }
146}
147
148static void copy_tiles_to_bitmap(const SkTDArray<TileInfo>& tiles, SkBitmap* bitmap) {
149 SkCanvas destination(*bitmap);
150
151 int tile_index = 0;
152 for (int tile_y_start = 0; tile_y_start < bitmap->height();
153 tile_y_start += kDefaultTileWidth) {
154 for (int tile_x_start = 0; tile_x_start < bitmap->width();
155 tile_x_start += kDefaultTileHeight) {
156 SkBitmap source = tiles[tile_index].fCanvas->getDevice()->accessBitmap(false);
157 destination.drawBitmap(source, tile_x_start, tile_y_start);
158 ++tile_index;
159 }
160 }
161}
162
163static void tile_render(SkPicture* picture, SkBitmap* bitmap) {
164 sk_tools::setup_bitmap(bitmap, picture->width(), picture->height());
165
166 SkTDArray<TileInfo> tiles;
167 setup_tiles(picture, &tiles);
168
169 for (int i = 0; i < tiles.count(); ++i) {
170 tiles[i].fCanvas->drawPicture(*picture);
171 }
172
173 copy_tiles_to_bitmap(tiles, bitmap);
174
175 for (int i = 0; i < tiles.count(); ++i) {
176 delete tiles[i].fCanvas;
177 delete tiles[i].fBitmap;
178 }
179}
180
keyar@chromium.org1cbd47c2012-07-13 18:22:59 +0000181static void render_picture(const SkString& inputPath, const SkString& outputDir,
182 RenderFunc renderFunc) {
keyar@chromium.orgd1dc9202012-07-09 18:32:08 +0000183 SkString inputFilename;
184 sk_tools::get_basename(&inputFilename, inputPath);
twiz@google.coma31b8bb2012-06-22 18:24:56 +0000185
keyar@chromium.orgd1dc9202012-07-09 18:32:08 +0000186 SkFILEStream inputStream;
twiz@google.coma31b8bb2012-06-22 18:24:56 +0000187 inputStream.setPath(inputPath.c_str());
188 if (!inputStream.isValid()) {
189 SkDebugf("Could not open file %s\n", inputPath.c_str());
190 return;
191 }
192
junov@chromium.org777442d2012-06-12 14:56:36 +0000193 SkPicture picture(&inputStream);
194 SkBitmap bitmap;
keyar@chromium.org1cbd47c2012-07-13 18:22:59 +0000195 renderFunc(&picture, &bitmap);
junov@chromium.org777442d2012-06-12 14:56:36 +0000196 write_output(outputDir, inputFilename, bitmap);
197}
198
keyar@chromium.org1cbd47c2012-07-13 18:22:59 +0000199static void process_input(const SkString& input, const SkString& outputDir,
200 RenderFunc renderFunc) {
201 SkOSFile::Iter iter(input.c_str(), "skp");
junov@chromium.org777442d2012-06-12 14:56:36 +0000202 SkString inputFilename;
203
keyar@chromium.orgd1dc9202012-07-09 18:32:08 +0000204 if (iter.next(&inputFilename)) {
205 do {
206 SkString inputPath;
207 sk_tools::make_filepath(&inputPath, input, inputFilename);
keyar@chromium.org1cbd47c2012-07-13 18:22:59 +0000208 render_picture(inputPath, outputDir, renderFunc);
keyar@chromium.orgd1dc9202012-07-09 18:32:08 +0000209 } while(iter.next(&inputFilename));
210 } else {
211 SkString inputPath(input);
keyar@chromium.org1cbd47c2012-07-13 18:22:59 +0000212 render_picture(inputPath, outputDir, renderFunc);
213 }
214}
215
216static void parse_commandline(int argc, char* const argv[], SkTArray<SkString>* inputs,
217 RenderFunc* renderFunc){
218 const char* argv0 = argv[0];
219 char* const* stop = argv + argc;
220
221 for (++argv; argv < stop; ++argv) {
keyar@chromium.orga2333d92012-07-16 17:29:16 +0000222 if (0 == strcmp(*argv, "--pipe")) {
223 *renderFunc = pipe_render;
keyar@chromium.org472b3792012-07-20 22:34:27 +0000224 } else if (0 == strcmp(*argv, "--tile")) {
225 *renderFunc = tile_render;
226 } else if ((0 == strcmp(*argv, "-h")) || (0 == strcmp(*argv, "--help"))) {
227 usage(argv0);
228 exit(-1);
keyar@chromium.orga2333d92012-07-16 17:29:16 +0000229 } else {
230 inputs->push_back(SkString(*argv));
231 }
keyar@chromium.orgd1dc9202012-07-09 18:32:08 +0000232 }
keyar@chromium.org472b3792012-07-20 22:34:27 +0000233
234 if (inputs->count() < 2) {
235 usage(argv0);
236 exit(-1);
237 }
keyar@chromium.orgd1dc9202012-07-09 18:32:08 +0000238}
239
240int main(int argc, char* const argv[]) {
keyar@chromium.org1cbd47c2012-07-13 18:22:59 +0000241 SkTArray<SkString> inputs;
242 RenderFunc renderFunc = simple_render;
keyar@chromium.org472b3792012-07-20 22:34:27 +0000243
keyar@chromium.org1cbd47c2012-07-13 18:22:59 +0000244 parse_commandline(argc, argv, &inputs, &renderFunc);
245 SkString outputDir = inputs[inputs.count() - 1];
keyar@chromium.orgd1dc9202012-07-09 18:32:08 +0000246
keyar@chromium.org1cbd47c2012-07-13 18:22:59 +0000247 for (int i = 0; i < inputs.count() - 1; i ++) {
248 process_input(inputs[i], outputDir, renderFunc);
junov@chromium.org777442d2012-06-12 14:56:36 +0000249 }
250}