• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 // Copyright (c) 2013 The Chromium Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file.
4 
5 #include "content/renderer/skia_benchmarking_extension.h"
6 
7 #include "base/base64.h"
8 #include "base/time/time.h"
9 #include "base/values.h"
10 #include "cc/base/math_util.h"
11 #include "cc/resources/picture.h"
12 #include "content/public/renderer/v8_value_converter.h"
13 #include "skia/ext/benchmarking_canvas.h"
14 #include "third_party/WebKit/public/platform/WebArrayBuffer.h"
15 #include "third_party/WebKit/public/web/WebFrame.h"
16 #include "third_party/skia/include/core/SkBitmapDevice.h"
17 #include "third_party/skia/include/core/SkCanvas.h"
18 #include "third_party/skia/include/core/SkColorPriv.h"
19 #include "third_party/skia/include/core/SkGraphics.h"
20 #include "third_party/skia/include/core/SkStream.h"
21 #include "third_party/skia/src/utils/debugger/SkDebugCanvas.h"
22 #include "third_party/skia/src/utils/debugger/SkDrawCommand.h"
23 #include "ui/gfx/rect_conversions.h"
24 #include "ui/gfx/skia_util.h"
25 #include "v8/include/v8.h"
26 
27 using blink::WebFrame;
28 
29 namespace {
30 
31 const char kSkiaBenchmarkingExtensionName[] = "v8/SkiaBenchmarking";
32 
ParsePictureArg(v8::Isolate * isolate,v8::Handle<v8::Value> arg)33 static scoped_ptr<base::Value> ParsePictureArg(v8::Isolate* isolate,
34                                                v8::Handle<v8::Value> arg) {
35   scoped_ptr<content::V8ValueConverter> converter(
36       content::V8ValueConverter::create());
37   return scoped_ptr<base::Value>(
38       converter->FromV8Value(arg, isolate->GetCurrentContext()));
39 }
40 
ParsePictureStr(v8::Isolate * isolate,v8::Handle<v8::Value> arg)41 static scoped_refptr<cc::Picture> ParsePictureStr(v8::Isolate* isolate,
42                                                   v8::Handle<v8::Value> arg) {
43   scoped_ptr<base::Value> picture_value = ParsePictureArg(isolate, arg);
44   if (!picture_value)
45     return NULL;
46   return cc::Picture::CreateFromSkpValue(picture_value.get());
47 }
48 
ParsePictureHash(v8::Isolate * isolate,v8::Handle<v8::Value> arg)49 static scoped_refptr<cc::Picture> ParsePictureHash(v8::Isolate* isolate,
50                                                    v8::Handle<v8::Value> arg) {
51   scoped_ptr<base::Value> picture_value = ParsePictureArg(isolate, arg);
52   if (!picture_value)
53     return NULL;
54   return cc::Picture::CreateFromValue(picture_value.get());
55 }
56 
57 class SkiaBenchmarkingWrapper : public v8::Extension {
58  public:
SkiaBenchmarkingWrapper()59   SkiaBenchmarkingWrapper() :
60       v8::Extension(kSkiaBenchmarkingExtensionName,
61         "if (typeof(chrome) == 'undefined') {"
62         "  chrome = {};"
63         "};"
64         "if (typeof(chrome.skiaBenchmarking) == 'undefined') {"
65         "  chrome.skiaBenchmarking = {};"
66         "};"
67         "chrome.skiaBenchmarking.rasterize = function(picture, params) {"
68         "  /* "
69         "     Rasterizes a Picture JSON-encoded by cc::Picture::AsValue()."
70         "     @param {Object} picture A json-encoded cc::Picture."
71         "     @param {"
72         "                 'scale':    {Number},"
73         "                 'stop':     {Number},"
74         "                 'overdraw': {Boolean},"
75         "                 'clip':     [Number, Number, Number, Number]"
76         "     } (optional) Rasterization parameters."
77         "     @returns {"
78         "                 'width':    {Number},"
79         "                 'height':   {Number},"
80         "                 'data':     {ArrayBuffer}"
81         "     }"
82         "     @returns undefined if the arguments are invalid or the picture"
83         "                        version is not supported."
84         "   */"
85         "  native function Rasterize();"
86         "  return Rasterize(picture, params);"
87         "};"
88         "chrome.skiaBenchmarking.getOps = function(picture) {"
89         "  /* "
90         "     Extracts the Skia draw commands from a JSON-encoded cc::Picture"
91         "     @param {Object} picture A json-encoded cc::Picture."
92         "     @returns [{ 'cmd': {String}, 'info': [String, ...] }, ...]"
93         "     @returns undefined if the arguments are invalid or the picture"
94         "                        version is not supported."
95         "   */"
96         "  native function GetOps();"
97         "  return GetOps(picture);"
98         "};"
99         "chrome.skiaBenchmarking.getOpTimings = function(picture) {"
100         "  /* "
101         "     Returns timing information for the given picture."
102         "     @param {Object} picture A json-encoded cc::Picture."
103         "     @returns { 'total_time': {Number}, 'cmd_times': [Number, ...] }"
104         "     @returns undefined if the arguments are invalid or the picture"
105         "                        version is not supported."
106         "   */"
107         "  native function GetOpTimings();"
108         "  return GetOpTimings(picture);"
109         "};"
110         "chrome.skiaBenchmarking.getInfo = function(picture) {"
111         "  /* "
112         "     Returns meta information for the given picture."
113         "     @param {Object} picture A base64 encoded SKP."
114         "     @returns { 'width': {Number}, 'height': {Number} }"
115         "     @returns undefined if the picture version is not supported."
116         "   */"
117         "  native function GetInfo();"
118         "  return GetInfo(picture);"
119         "};"
120         ) {
121       content::SkiaBenchmarkingExtension::InitSkGraphics();
122   }
123 
GetNativeFunctionTemplate(v8::Isolate * isolate,v8::Handle<v8::String> name)124   virtual v8::Handle<v8::FunctionTemplate> GetNativeFunctionTemplate(
125       v8::Isolate* isolate,
126       v8::Handle<v8::String> name) OVERRIDE {
127     if (name->Equals(v8::String::NewFromUtf8(isolate, "Rasterize")))
128       return v8::FunctionTemplate::New(isolate, Rasterize);
129     if (name->Equals(v8::String::NewFromUtf8(isolate, "GetOps")))
130       return v8::FunctionTemplate::New(isolate, GetOps);
131     if (name->Equals(v8::String::NewFromUtf8(isolate, "GetOpTimings")))
132       return v8::FunctionTemplate::New(isolate, GetOpTimings);
133     if (name->Equals(v8::String::NewFromUtf8(isolate, "GetInfo")))
134       return v8::FunctionTemplate::New(isolate, GetInfo);
135 
136     return v8::Handle<v8::FunctionTemplate>();
137   }
138 
Rasterize(const v8::FunctionCallbackInfo<v8::Value> & args)139   static void Rasterize(const v8::FunctionCallbackInfo<v8::Value>& args) {
140     if (args.Length() < 1)
141       return;
142 
143     v8::Isolate* isolate = args.GetIsolate();
144     scoped_refptr<cc::Picture> picture = ParsePictureHash(isolate, args[0]);
145     if (!picture.get())
146       return;
147 
148     double scale = 1.0;
149     gfx::Rect clip_rect(picture->LayerRect());
150     int stop_index = -1;
151     bool overdraw = false;
152 
153     if (args.Length() > 1) {
154       scoped_ptr<content::V8ValueConverter> converter(
155           content::V8ValueConverter::create());
156       scoped_ptr<base::Value> params_value(
157           converter->FromV8Value(args[1], isolate->GetCurrentContext()));
158 
159       const base::DictionaryValue* params_dict = NULL;
160       if (params_value.get() && params_value->GetAsDictionary(&params_dict)) {
161         params_dict->GetDouble("scale", &scale);
162         params_dict->GetInteger("stop", &stop_index);
163         params_dict->GetBoolean("overdraw", &overdraw);
164 
165         const base::Value* clip_value = NULL;
166         if (params_dict->Get("clip", &clip_value))
167           cc::MathUtil::FromValue(clip_value, &clip_rect);
168       }
169     }
170 
171     gfx::RectF clip(clip_rect);
172     clip.Intersect(picture->LayerRect());
173     clip.Scale(scale);
174     gfx::Rect snapped_clip = gfx::ToEnclosingRect(clip);
175 
176     const int kMaxBitmapSize = 4096;
177     if (snapped_clip.width() > kMaxBitmapSize
178         || snapped_clip.height() > kMaxBitmapSize)
179       return;
180 
181     SkBitmap bitmap;
182     bitmap.setConfig(SkBitmap::kARGB_8888_Config, snapped_clip.width(),
183                      snapped_clip.height());
184     if (!bitmap.allocPixels())
185       return;
186     bitmap.eraseARGB(0, 0, 0, 0);
187 
188     SkCanvas canvas(bitmap);
189     canvas.translate(SkFloatToScalar(-clip.x()),
190                      SkFloatToScalar(-clip.y()));
191     canvas.clipRect(gfx::RectToSkRect(snapped_clip));
192     canvas.scale(scale, scale);
193     canvas.translate(picture->LayerRect().x(),
194                      picture->LayerRect().y());
195 
196     // First, build a debug canvas for the given picture.
197     SkDebugCanvas debug_canvas(picture->LayerRect().width(),
198                                picture->LayerRect().height());
199     picture->Replay(&debug_canvas);
200 
201     // Raster the requested command subset into the bitmap-backed canvas.
202     int last_index = debug_canvas.getSize() - 1;
203     if (last_index >= 0) {
204         debug_canvas.setOverdrawViz(overdraw);
205         debug_canvas.drawTo(&canvas, stop_index < 0
206                             ? last_index
207                             : std::min(last_index, stop_index));
208     }
209 
210     blink::WebArrayBuffer buffer =
211         blink::WebArrayBuffer::create(bitmap.getSize(), 1);
212     uint32* packed_pixels = reinterpret_cast<uint32*>(bitmap.getPixels());
213     uint8* buffer_pixels = reinterpret_cast<uint8*>(buffer.data());
214     // Swizzle from native Skia format to RGBA as we copy out.
215     for (size_t i = 0; i < bitmap.getSize(); i += 4) {
216         uint32 c = packed_pixels[i >> 2];
217         buffer_pixels[i]     = SkGetPackedR32(c);
218         buffer_pixels[i + 1] = SkGetPackedG32(c);
219         buffer_pixels[i + 2] = SkGetPackedB32(c);
220         buffer_pixels[i + 3] = SkGetPackedA32(c);
221     }
222 
223     v8::Handle<v8::Object> result = v8::Object::New();
224     result->Set(v8::String::NewFromUtf8(isolate, "width"),
225                 v8::Number::New(snapped_clip.width()));
226     result->Set(v8::String::NewFromUtf8(isolate, "height"),
227                 v8::Number::New(snapped_clip.height()));
228     result->Set(v8::String::NewFromUtf8(isolate, "data"), buffer.toV8Value());
229 
230     args.GetReturnValue().Set(result);
231   }
232 
GetOps(const v8::FunctionCallbackInfo<v8::Value> & args)233   static void GetOps(const v8::FunctionCallbackInfo<v8::Value>& args) {
234     if (args.Length() != 1)
235       return;
236 
237     v8::Isolate* isolate = args.GetIsolate();
238     scoped_refptr<cc::Picture> picture = ParsePictureHash(isolate, args[0]);
239     if (!picture.get())
240       return;
241 
242     gfx::Rect bounds = picture->LayerRect();
243     SkDebugCanvas canvas(bounds.width(), bounds.height());
244     picture->Replay(&canvas);
245 
246     v8::Local<v8::Array> result = v8::Array::New(isolate, canvas.getSize());
247     for (int i = 0; i < canvas.getSize(); ++i) {
248       DrawType cmd_type = canvas.getDrawCommandAt(i)->getType();
249       v8::Handle<v8::Object> cmd = v8::Object::New();
250       cmd->Set(v8::String::NewFromUtf8(isolate, "cmd_type"),
251                v8::Integer::New(isolate, cmd_type));
252       cmd->Set(v8::String::NewFromUtf8(isolate, "cmd_string"),
253                v8::String::NewFromUtf8(
254                    isolate, SkDrawCommand::GetCommandString(cmd_type)));
255 
256       SkTDArray<SkString*>* info = canvas.getCommandInfo(i);
257       DCHECK(info);
258 
259       v8::Local<v8::Array> v8_info = v8::Array::New(isolate, info->count());
260       for (int j = 0; j < info->count(); ++j) {
261         const SkString* info_str = (*info)[j];
262         DCHECK(info_str);
263         v8_info->Set(j, v8::String::NewFromUtf8(isolate, info_str->c_str()));
264       }
265 
266       cmd->Set(v8::String::NewFromUtf8(isolate, "info"), v8_info);
267 
268       result->Set(i, cmd);
269     }
270 
271     args.GetReturnValue().Set(result);
272   }
273 
GetOpTimings(const v8::FunctionCallbackInfo<v8::Value> & args)274   static void GetOpTimings(const v8::FunctionCallbackInfo<v8::Value>& args) {
275     if (args.Length() != 1)
276       return;
277 
278     v8::Isolate* isolate = args.GetIsolate();
279     scoped_refptr<cc::Picture> picture = ParsePictureHash(isolate, args[0]);
280     if (!picture.get())
281       return;
282 
283     gfx::Rect bounds = picture->LayerRect();
284 
285     // Measure the total time by drawing straight into a bitmap-backed canvas.
286     skia::RefPtr<SkBaseDevice> device = skia::AdoptRef(
287         SkNEW_ARGS(SkBitmapDevice,
288              (SkBitmap::kARGB_8888_Config, bounds.width(), bounds.height())));
289     SkCanvas bitmap_canvas(device.get());
290     bitmap_canvas.clear(SK_ColorTRANSPARENT);
291     base::TimeTicks t0 = base::TimeTicks::HighResNow();
292     picture->Replay(&bitmap_canvas);
293     base::TimeDelta total_time = base::TimeTicks::HighResNow() - t0;
294 
295     // Gather per-op timing info by drawing into a BenchmarkingCanvas.
296     skia::BenchmarkingCanvas benchmarking_canvas(bounds.width(),
297                                                  bounds.height());
298     picture->Replay(&benchmarking_canvas);
299 
300     v8::Local<v8::Array> op_times =
301         v8::Array::New(isolate, benchmarking_canvas.CommandCount());
302     for (size_t i = 0; i < benchmarking_canvas.CommandCount(); ++i)
303         op_times->Set(i, v8::Number::New(benchmarking_canvas.GetTime(i)));
304 
305     v8::Handle<v8::Object> result = v8::Object::New(isolate);
306     result->Set(v8::String::NewFromUtf8(isolate, "total_time"),
307                 v8::Number::New(total_time.InMillisecondsF()));
308     result->Set(v8::String::NewFromUtf8(isolate, "cmd_times"), op_times);
309 
310     args.GetReturnValue().Set(result);
311   }
312 
GetInfo(const v8::FunctionCallbackInfo<v8::Value> & args)313   static void GetInfo(const v8::FunctionCallbackInfo<v8::Value>& args) {
314     if (args.Length() != 1)
315       return;
316 
317     v8::Isolate* isolate = args.GetIsolate();
318     scoped_refptr<cc::Picture> picture = ParsePictureStr(isolate, args[0]);
319     if (!picture.get())
320       return;
321 
322     v8::Handle<v8::Object> result = v8::Object::New(isolate);
323     result->Set(v8::String::NewFromUtf8(isolate, "width"),
324                 v8::Number::New(picture->LayerRect().width()));
325     result->Set(v8::String::NewFromUtf8(isolate, "height"),
326                 v8::Number::New(picture->LayerRect().height()));
327 
328     args.GetReturnValue().Set(result);
329   }
330 };
331 
332 } // namespace
333 
334 namespace content {
335 
Get()336 v8::Extension* SkiaBenchmarkingExtension::Get() {
337   return new SkiaBenchmarkingWrapper();
338 }
339 
InitSkGraphics()340 void SkiaBenchmarkingExtension::InitSkGraphics() {
341     // Always call on the main render thread.
342     // Does not need to be thread-safe, as long as the above holds.
343     // FIXME: remove this after Skia updates SkGraphics::Init() to be
344     //        thread-safe and idempotent.
345     static bool skia_initialized = false;
346     if (!skia_initialized) {
347       SkGraphics::Init();
348       skia_initialized = true;
349     }
350 }
351 
352 } // namespace content
353