| junov@chromium.org | 777442d | 2012-06-12 14:56:36 +0000 | [diff] [blame] | 1 | /* |
| 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.org | a2333d9 | 2012-07-16 17:29:16 +0000 | [diff] [blame] | 10 | #include "SamplePipeControllers.h" |
| junov@chromium.org | 777442d | 2012-06-12 14:56:36 +0000 | [diff] [blame] | 11 | #include "SkCanvas.h" |
| 12 | #include "SkColorPriv.h" |
| keyar@chromium.org | 472b379 | 2012-07-20 22:34:27 +0000 | [diff] [blame^] | 13 | #include "SkDevice.h" |
| keyar@chromium.org | a2333d9 | 2012-07-16 17:29:16 +0000 | [diff] [blame] | 14 | #include "SkGPipe.h" |
| junov@chromium.org | 777442d | 2012-06-12 14:56:36 +0000 | [diff] [blame] | 15 | #include "SkImageEncoder.h" |
| 16 | #include "SkOSFile.h" |
| 17 | #include "SkPicture.h" |
| 18 | #include "SkStream.h" |
| 19 | #include "SkString.h" |
| senorblanco@chromium.org | 3cbbb54 | 2012-07-13 18:55:53 +0000 | [diff] [blame] | 20 | #include "SkTArray.h" |
| keyar@chromium.org | 472b379 | 2012-07-20 22:34:27 +0000 | [diff] [blame^] | 21 | #include "SkTDArray.h" |
| twiz@google.com | a31b8bb | 2012-06-22 18:24:56 +0000 | [diff] [blame] | 22 | #include "picture_utils.h" |
| junov@chromium.org | 777442d | 2012-06-12 14:56:36 +0000 | [diff] [blame] | 23 | |
| keyar@chromium.org | 472b379 | 2012-07-20 22:34:27 +0000 | [diff] [blame^] | 24 | enum { |
| 25 | kDefaultTileWidth = 256, |
| 26 | kDefaultTileHeight = 256 |
| 27 | }; |
| junov@chromium.org | 777442d | 2012-06-12 14:56:36 +0000 | [diff] [blame] | 28 | |
| 29 | static void usage(const char* argv0) { |
| 30 | SkDebugf("SkPicture rendering tool\n"); |
| 31 | SkDebugf("\n" |
| 32 | "Usage: \n" |
| keyar@chromium.org | 472b379 | 2012-07-20 22:34:27 +0000 | [diff] [blame^] | 33 | " %s <input>... <outputDir> \n" |
| 34 | " [--pipe | --tile]" |
| junov@chromium.org | 777442d | 2012-06-12 14:56:36 +0000 | [diff] [blame] | 35 | , argv0); |
| keyar@chromium.org | 472b379 | 2012-07-20 22:34:27 +0000 | [diff] [blame^] | 36 | SkDebugf("\n\n"); |
| junov@chromium.org | 777442d | 2012-06-12 14:56:36 +0000 | [diff] [blame] | 37 | SkDebugf( |
| keyar@chromium.org | d1dc920 | 2012-07-09 18:32:08 +0000 | [diff] [blame] | 38 | " input: A list of directories and files to use as input.\n" |
| 39 | " Files are expected to have the .skp extension.\n"); |
| junov@chromium.org | 777442d | 2012-06-12 14:56:36 +0000 | [diff] [blame] | 40 | SkDebugf( |
| 41 | " outputDir: directory to write the rendered images.\n"); |
| keyar@chromium.org | 472b379 | 2012-07-20 22:34:27 +0000 | [diff] [blame^] | 42 | SkDebugf( |
| 43 | " --pipe : Render using a SkGPipe\n"); |
| 44 | SkDebugf( |
| 45 | " --tile : Render using tiles.\n"); |
| junov@chromium.org | 777442d | 2012-06-12 14:56:36 +0000 | [diff] [blame] | 46 | } |
| 47 | |
| keyar@chromium.org | 472b379 | 2012-07-20 22:34:27 +0000 | [diff] [blame^] | 48 | static void simple_render(SkPicture* pict, SkBitmap* bitmap); |
| 49 | typedef void (*RenderFunc) (SkPicture*, SkBitmap*); |
| 50 | |
| keyar@chromium.org | 1cbd47c | 2012-07-13 18:22:59 +0000 | [diff] [blame] | 51 | static void make_output_filepath(SkString* path, const SkString& dir, |
| junov@chromium.org | 777442d | 2012-06-12 14:56:36 +0000 | [diff] [blame] | 52 | const SkString& name) { |
| twiz@google.com | a31b8bb | 2012-06-22 18:24:56 +0000 | [diff] [blame] | 53 | sk_tools::make_filepath(path, dir, name); |
| junov@chromium.org | 777442d | 2012-06-12 14:56:36 +0000 | [diff] [blame] | 54 | path->remove(path->size() - 3, 3); |
| 55 | path->append("png"); |
| 56 | } |
| 57 | |
| keyar@chromium.org | 1cbd47c | 2012-07-13 18:22:59 +0000 | [diff] [blame] | 58 | static void simple_render(SkPicture* pict, SkBitmap* bitmap) { |
| 59 | sk_tools::setup_bitmap(bitmap, pict->width(), pict->height()); |
| junov@chromium.org | 777442d | 2012-06-12 14:56:36 +0000 | [diff] [blame] | 60 | SkCanvas canvas(*bitmap); |
| keyar@chromium.org | 1cbd47c | 2012-07-13 18:22:59 +0000 | [diff] [blame] | 61 | canvas.drawPicture(*pict); |
| junov@chromium.org | 777442d | 2012-06-12 14:56:36 +0000 | [diff] [blame] | 62 | } |
| 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 | */ |
| 68 | static 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 | |
| 77 | static bool write_bitmap(const SkString& path, const SkBitmap& bitmap) { |
| 78 | SkBitmap copy; |
| 79 | bitmap.copyTo(©, 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.org | 1cbd47c | 2012-07-13 18:22:59 +0000 | [diff] [blame] | 85 | static void write_output(const SkString& outputDir, const SkString& inputFilename, |
| junov@chromium.org | 777442d | 2012-06-12 14:56:36 +0000 | [diff] [blame] | 86 | 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.org | a2333d9 | 2012-07-16 17:29:16 +0000 | [diff] [blame] | 95 | static 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 | |
| 103 | static 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.org | 472b379 | 2012-07-20 22:34:27 +0000 | [diff] [blame^] | 111 | struct 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. |
| 120 | static 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 | |
| 126 | static 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 | |
| 138 | static 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 | |
| 148 | static 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 | |
| 163 | static 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.org | 1cbd47c | 2012-07-13 18:22:59 +0000 | [diff] [blame] | 181 | static void render_picture(const SkString& inputPath, const SkString& outputDir, |
| 182 | RenderFunc renderFunc) { |
| keyar@chromium.org | d1dc920 | 2012-07-09 18:32:08 +0000 | [diff] [blame] | 183 | SkString inputFilename; |
| 184 | sk_tools::get_basename(&inputFilename, inputPath); |
| twiz@google.com | a31b8bb | 2012-06-22 18:24:56 +0000 | [diff] [blame] | 185 | |
| keyar@chromium.org | d1dc920 | 2012-07-09 18:32:08 +0000 | [diff] [blame] | 186 | SkFILEStream inputStream; |
| twiz@google.com | a31b8bb | 2012-06-22 18:24:56 +0000 | [diff] [blame] | 187 | 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.org | 777442d | 2012-06-12 14:56:36 +0000 | [diff] [blame] | 193 | SkPicture picture(&inputStream); |
| 194 | SkBitmap bitmap; |
| keyar@chromium.org | 1cbd47c | 2012-07-13 18:22:59 +0000 | [diff] [blame] | 195 | renderFunc(&picture, &bitmap); |
| junov@chromium.org | 777442d | 2012-06-12 14:56:36 +0000 | [diff] [blame] | 196 | write_output(outputDir, inputFilename, bitmap); |
| 197 | } |
| 198 | |
| keyar@chromium.org | 1cbd47c | 2012-07-13 18:22:59 +0000 | [diff] [blame] | 199 | static void process_input(const SkString& input, const SkString& outputDir, |
| 200 | RenderFunc renderFunc) { |
| 201 | SkOSFile::Iter iter(input.c_str(), "skp"); |
| junov@chromium.org | 777442d | 2012-06-12 14:56:36 +0000 | [diff] [blame] | 202 | SkString inputFilename; |
| 203 | |
| keyar@chromium.org | d1dc920 | 2012-07-09 18:32:08 +0000 | [diff] [blame] | 204 | if (iter.next(&inputFilename)) { |
| 205 | do { |
| 206 | SkString inputPath; |
| 207 | sk_tools::make_filepath(&inputPath, input, inputFilename); |
| keyar@chromium.org | 1cbd47c | 2012-07-13 18:22:59 +0000 | [diff] [blame] | 208 | render_picture(inputPath, outputDir, renderFunc); |
| keyar@chromium.org | d1dc920 | 2012-07-09 18:32:08 +0000 | [diff] [blame] | 209 | } while(iter.next(&inputFilename)); |
| 210 | } else { |
| 211 | SkString inputPath(input); |
| keyar@chromium.org | 1cbd47c | 2012-07-13 18:22:59 +0000 | [diff] [blame] | 212 | render_picture(inputPath, outputDir, renderFunc); |
| 213 | } |
| 214 | } |
| 215 | |
| 216 | static 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.org | a2333d9 | 2012-07-16 17:29:16 +0000 | [diff] [blame] | 222 | if (0 == strcmp(*argv, "--pipe")) { |
| 223 | *renderFunc = pipe_render; |
| keyar@chromium.org | 472b379 | 2012-07-20 22:34:27 +0000 | [diff] [blame^] | 224 | } 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.org | a2333d9 | 2012-07-16 17:29:16 +0000 | [diff] [blame] | 229 | } else { |
| 230 | inputs->push_back(SkString(*argv)); |
| 231 | } |
| keyar@chromium.org | d1dc920 | 2012-07-09 18:32:08 +0000 | [diff] [blame] | 232 | } |
| keyar@chromium.org | 472b379 | 2012-07-20 22:34:27 +0000 | [diff] [blame^] | 233 | |
| 234 | if (inputs->count() < 2) { |
| 235 | usage(argv0); |
| 236 | exit(-1); |
| 237 | } |
| keyar@chromium.org | d1dc920 | 2012-07-09 18:32:08 +0000 | [diff] [blame] | 238 | } |
| 239 | |
| 240 | int main(int argc, char* const argv[]) { |
| keyar@chromium.org | 1cbd47c | 2012-07-13 18:22:59 +0000 | [diff] [blame] | 241 | SkTArray<SkString> inputs; |
| 242 | RenderFunc renderFunc = simple_render; |
| keyar@chromium.org | 472b379 | 2012-07-20 22:34:27 +0000 | [diff] [blame^] | 243 | |
| keyar@chromium.org | 1cbd47c | 2012-07-13 18:22:59 +0000 | [diff] [blame] | 244 | parse_commandline(argc, argv, &inputs, &renderFunc); |
| 245 | SkString outputDir = inputs[inputs.count() - 1]; |
| keyar@chromium.org | d1dc920 | 2012-07-09 18:32:08 +0000 | [diff] [blame] | 246 | |
| keyar@chromium.org | 1cbd47c | 2012-07-13 18:22:59 +0000 | [diff] [blame] | 247 | for (int i = 0; i < inputs.count() - 1; i ++) { |
| 248 | process_input(inputs[i], outputDir, renderFunc); |
| junov@chromium.org | 777442d | 2012-06-12 14:56:36 +0000 | [diff] [blame] | 249 | } |
| 250 | } |