1 /*
2 * Copyright 2016 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 #include "Request.h"
9
10 #include "SkPictureRecorder.h"
11 #include "SkPixelSerializer.h"
12 #include "SkPM4fPriv.h"
13 #include "picture_utils.h"
14 #include "sk_tool_utils.h"
15
16 using namespace sk_gpu_test;
17
18 static int kDefaultWidth = 1920;
19 static int kDefaultHeight = 1080;
20 static int kMaxWidth = 8192;
21 static int kMaxHeight = 8192;
22
23
Request(SkString rootUrl)24 Request::Request(SkString rootUrl)
25 : fUploadContext(nullptr)
26 , fUrlDataManager(rootUrl)
27 , fGPUEnabled(false)
28 , fOverdraw(false)
29 , fColorMode(0) {
30 // create surface
31 #if SK_SUPPORT_GPU
32 GrContextOptions grContextOpts;
33 fContextFactory = new GrContextFactory(grContextOpts);
34 #else
35 fContextFactory = nullptr;
36 #endif
37 }
38
~Request()39 Request::~Request() {
40 #if SK_SUPPORT_GPU
41 if (fContextFactory) {
42 delete fContextFactory;
43 }
44 #endif
45 }
46
getBitmapFromCanvas(SkCanvas * canvas)47 SkBitmap* Request::getBitmapFromCanvas(SkCanvas* canvas) {
48 SkBitmap* bmp = new SkBitmap();
49 bmp->setInfo(canvas->imageInfo());
50 if (!canvas->readPixels(bmp, 0, 0)) {
51 fprintf(stderr, "Can't read pixels\n");
52 return nullptr;
53 }
54 return bmp;
55 }
56
writeCanvasToPng(SkCanvas * canvas)57 sk_sp<SkData> Request::writeCanvasToPng(SkCanvas* canvas) {
58 // capture pixels
59 std::unique_ptr<SkBitmap> bmp(this->getBitmapFromCanvas(canvas));
60 SkASSERT(bmp);
61
62 // Convert to format suitable for PNG output
63 sk_sp<SkData> encodedBitmap = sk_tools::encode_bitmap_for_png(*bmp);
64 SkASSERT(encodedBitmap.get());
65
66 // write to an opaque png (black background)
67 SkDynamicMemoryWStream buffer;
68 SkDrawCommand::WritePNG(encodedBitmap->bytes(), bmp->width(), bmp->height(),
69 buffer, true);
70 return buffer.detachAsData();
71 }
72
getCanvas()73 SkCanvas* Request::getCanvas() {
74 #if SK_SUPPORT_GPU
75 GrContextFactory* factory = fContextFactory;
76 GLTestContext* gl = factory->getContextInfo(GrContextFactory::kGL_ContextType,
77 GrContextFactory::ContextOverrides::kNone).glContext();
78 if (!gl) {
79 gl = factory->getContextInfo(GrContextFactory::kGLES_ContextType,
80 GrContextFactory::ContextOverrides::kNone).glContext();
81 }
82 if (!gl) {
83 gl = factory->getContextInfo(GrContextFactory::kMESA_ContextType,
84 GrContextFactory::ContextOverrides::kNone).glContext();
85 }
86 if (gl) {
87 gl->makeCurrent();
88 }
89 #endif
90 SkASSERT(fDebugCanvas);
91
92 // create the appropriate surface if necessary
93 if (!fSurface) {
94 this->enableGPU(fGPUEnabled);
95 }
96 SkCanvas* target = fSurface->getCanvas();
97 return target;
98 }
99
drawToCanvas(int n,int m)100 void Request::drawToCanvas(int n, int m) {
101 SkCanvas* target = this->getCanvas();
102 fDebugCanvas->drawTo(target, n, m);
103 }
104
drawToPng(int n,int m)105 sk_sp<SkData> Request::drawToPng(int n, int m) {
106 //fDebugCanvas->setOverdrawViz(true);
107 this->drawToCanvas(n, m);
108 //fDebugCanvas->setOverdrawViz(false);
109 return writeCanvasToPng(this->getCanvas());
110 }
111
writeOutSkp()112 sk_sp<SkData> Request::writeOutSkp() {
113 // Playback into picture recorder
114 SkIRect bounds = this->getBounds();
115 SkPictureRecorder recorder;
116 SkCanvas* canvas = recorder.beginRecording(SkIntToScalar(bounds.width()),
117 SkIntToScalar(bounds.height()));
118
119 fDebugCanvas->draw(canvas);
120
121 sk_sp<SkPicture> picture(recorder.finishRecordingAsPicture());
122
123 SkDynamicMemoryWStream outStream;
124
125 sk_sp<SkPixelSerializer> serializer = sk_tool_utils::MakePixelSerializer();
126 picture->serialize(&outStream, serializer.get());
127
128 return outStream.detachAsData();
129 }
130
getContext()131 GrContext* Request::getContext() {
132 #if SK_SUPPORT_GPU
133 GrContext* result = fContextFactory->get(GrContextFactory::kGL_ContextType,
134 GrContextFactory::ContextOverrides::kNone);
135 if (!result) {
136 result = fContextFactory->get(GrContextFactory::kGLES_ContextType,
137 GrContextFactory::ContextOverrides::kNone);
138 }
139 if (!result) {
140 result = fContextFactory->get(GrContextFactory::kMESA_ContextType,
141 GrContextFactory::ContextOverrides::kNone);
142 }
143 return result;
144 #else
145 return nullptr;
146 #endif
147 }
148
getBounds()149 SkIRect Request::getBounds() {
150 SkIRect bounds;
151 if (fPicture) {
152 bounds = fPicture->cullRect().roundOut();
153 if (fGPUEnabled) {
154 #if SK_SUPPORT_GPU
155 int maxRTSize = this->getContext()->caps()->maxRenderTargetSize();
156 bounds = SkIRect::MakeWH(SkTMin(bounds.width(), maxRTSize),
157 SkTMin(bounds.height(), maxRTSize));
158 #endif
159 }
160 } else {
161 bounds = SkIRect::MakeWH(kDefaultWidth, kDefaultHeight);
162 }
163
164 // We clip to kMaxWidth / kMaxHeight for performance reasons.
165 // TODO make this configurable
166 bounds = SkIRect::MakeWH(SkTMin(bounds.width(), kMaxWidth),
167 SkTMin(bounds.height(), kMaxHeight));
168 return bounds;
169 }
170
171 namespace {
172
173 struct ColorAndProfile {
174 SkColorType fColorType;
175 bool fSRGB;
176 };
177
178 ColorAndProfile ColorModes[] = {
179 { kN32_SkColorType, false },
180 { kN32_SkColorType, true },
181 { kRGBA_F16_SkColorType, true },
182 };
183
184 }
185
createCPUSurface()186 SkSurface* Request::createCPUSurface() {
187 SkIRect bounds = this->getBounds();
188 ColorAndProfile cap = ColorModes[fColorMode];
189 auto colorSpace = kRGBA_F16_SkColorType == cap.fColorType
190 ? SkColorSpace::MakeSRGBLinear()
191 : SkColorSpace::MakeSRGB();
192 SkImageInfo info = SkImageInfo::Make(bounds.width(), bounds.height(), cap.fColorType,
193 kPremul_SkAlphaType, cap.fSRGB ? colorSpace : nullptr);
194 return SkSurface::MakeRaster(info).release();
195 }
196
createGPUSurface()197 SkSurface* Request::createGPUSurface() {
198 GrContext* context = this->getContext();
199 SkIRect bounds = this->getBounds();
200 ColorAndProfile cap = ColorModes[fColorMode];
201 auto colorSpace = kRGBA_F16_SkColorType == cap.fColorType
202 ? SkColorSpace::MakeSRGBLinear()
203 : SkColorSpace::MakeSRGB();
204 SkImageInfo info = SkImageInfo::Make(bounds.width(), bounds.height(), cap.fColorType,
205 kPremul_SkAlphaType, cap.fSRGB ? colorSpace: nullptr);
206 SkSurface* surface = SkSurface::MakeRenderTarget(context, SkBudgeted::kNo, info).release();
207 return surface;
208 }
209
setOverdraw(bool enable)210 bool Request::setOverdraw(bool enable) {
211 fOverdraw = enable;
212 return true;
213 }
214
setColorMode(int mode)215 bool Request::setColorMode(int mode) {
216 fColorMode = mode;
217 return enableGPU(fGPUEnabled);
218 }
219
enableGPU(bool enable)220 bool Request::enableGPU(bool enable) {
221 if (enable) {
222 SkSurface* surface = this->createGPUSurface();
223 if (surface) {
224 fSurface.reset(surface);
225 fGPUEnabled = true;
226
227 // When we switch to GPU, there seems to be some mystery draws in the canvas. So we
228 // draw once to flush the pipe
229 // TODO understand what is actually happening here
230 if (fDebugCanvas) {
231 fDebugCanvas->drawTo(this->getCanvas(), this->getLastOp());
232 this->getCanvas()->flush();
233 }
234
235 return true;
236 }
237 return false;
238 }
239 fSurface.reset(this->createCPUSurface());
240 fGPUEnabled = false;
241 return true;
242 }
243
initPictureFromStream(SkStream * stream)244 bool Request::initPictureFromStream(SkStream* stream) {
245 // parse picture from stream
246 fPicture = SkPicture::MakeFromStream(stream);
247 if (!fPicture) {
248 fprintf(stderr, "Could not create picture from stream.\n");
249 return false;
250 }
251
252 // reinitialize canvas with the new picture dimensions
253 this->enableGPU(fGPUEnabled);
254
255 // pour picture into debug canvas
256 SkIRect bounds = this->getBounds();
257 fDebugCanvas.reset(new SkDebugCanvas(bounds.width(), bounds.height()));
258 fDebugCanvas->drawPicture(fPicture);
259
260 // for some reason we need to 'flush' the debug canvas by drawing all of the ops
261 fDebugCanvas->drawTo(this->getCanvas(), this->getLastOp());
262 this->getCanvas()->flush();
263 return true;
264 }
265
getJsonOps(int n)266 sk_sp<SkData> Request::getJsonOps(int n) {
267 SkCanvas* canvas = this->getCanvas();
268 Json::Value root = fDebugCanvas->toJSON(fUrlDataManager, n, canvas);
269 root["mode"] = Json::Value(fGPUEnabled ? "gpu" : "cpu");
270 root["drawGpuOpBounds"] = Json::Value(fDebugCanvas->getDrawGpuOpBounds());
271 root["colorMode"] = Json::Value(fColorMode);
272 SkDynamicMemoryWStream stream;
273 stream.writeText(Json::FastWriter().write(root).c_str());
274
275 return stream.detachAsData();
276 }
277
getJsonOpList(int n)278 sk_sp<SkData> Request::getJsonOpList(int n) {
279 SkCanvas* canvas = this->getCanvas();
280 SkASSERT(fGPUEnabled);
281
282 Json::Value result = fDebugCanvas->toJSONOpList(n, canvas);
283
284 SkDynamicMemoryWStream stream;
285 stream.writeText(Json::FastWriter().write(result).c_str());
286
287 return stream.detachAsData();
288 }
289
getJsonInfo(int n)290 sk_sp<SkData> Request::getJsonInfo(int n) {
291 // drawTo
292 sk_sp<SkSurface> surface(this->createCPUSurface());
293 SkCanvas* canvas = surface->getCanvas();
294
295 // TODO this is really slow and we should cache the matrix and clip
296 fDebugCanvas->drawTo(canvas, n);
297
298 // make some json
299 SkMatrix vm = fDebugCanvas->getCurrentMatrix();
300 SkIRect clip = fDebugCanvas->getCurrentClip();
301 Json::Value info(Json::objectValue);
302 info["ViewMatrix"] = SkDrawCommand::MakeJsonMatrix(vm);
303 info["ClipRect"] = SkDrawCommand::MakeJsonIRect(clip);
304
305 std::string json = Json::FastWriter().write(info);
306
307 // We don't want the null terminator so strlen is correct
308 return SkData::MakeWithCopy(json.c_str(), strlen(json.c_str()));
309 }
310
getPixel(int x,int y)311 SkColor Request::getPixel(int x, int y) {
312 SkCanvas* canvas = this->getCanvas();
313 canvas->flush();
314 std::unique_ptr<SkBitmap> bitmap(this->getBitmapFromCanvas(canvas));
315 SkASSERT(bitmap);
316
317 // Convert to format suitable for inspection
318 sk_sp<SkData> encodedBitmap = sk_tools::encode_bitmap_for_png(*bitmap);
319 SkASSERT(encodedBitmap);
320
321 const uint8_t* start = encodedBitmap->bytes() + ((y * bitmap->width() + x) * 4);
322 SkColor result = SkColorSetARGB(start[3], start[0], start[1], start[2]);
323 return result;
324 }
325