blob: 353f35bfdeb5ab47991621998f51a2723448e221 [file] [log] [blame]
// Copyright 2015 The Chromium Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include "sky/shell/gpu/direct/rasterizer_direct.h"
#include "base/trace_event/trace_event.h"
#include "mojo/public/cpp/system/data_pipe.h"
#include "sky/compositor/container_layer.h"
#include "sky/compositor/layer.h"
#include "sky/compositor/paint_context.h"
#include "sky/compositor/picture_layer.h"
#include "sky/shell/shell.h"
#include "third_party/skia/include/core/SkCanvas.h"
#include "third_party/skia/include/core/SkPicture.h"
#include "ui/gl/gl_bindings.h"
#include "ui/gl/gl_bindings_skia_in_process.h"
#include "ui/gl/gl_context.h"
#include "ui/gl/gl_share_group.h"
#include "ui/gl/gl_surface.h"
namespace sky {
namespace shell {
static const double kOneFrameDuration = 1e3 / 60.0;
scoped_ptr<Rasterizer> Rasterizer::Create() {
return make_scoped_ptr(new RasterizerDirect());
}
RasterizerDirect::RasterizerDirect()
: share_group_(new gfx::GLShareGroup()), binding_(this),
weak_factory_(this) {
}
RasterizerDirect::~RasterizerDirect() {
}
base::WeakPtr<RasterizerDirect> RasterizerDirect::GetWeakPtr() {
return weak_factory_.GetWeakPtr();
}
base::WeakPtr<Rasterizer> RasterizerDirect::GetWeakRasterizerPtr() {
return GetWeakPtr();
}
void RasterizerDirect::ConnectToRasterizer(
mojo::InterfaceRequest<rasterizer::Rasterizer> request) {
binding_.Bind(request.Pass());
}
void RasterizerDirect::OnAcceleratedWidgetAvailable(gfx::AcceleratedWidget widget) {
surface_ =
gfx::GLSurface::CreateViewGLSurface(widget, gfx::SurfaceConfiguration());
CHECK(surface_) << "GLSurface required.";
}
void RasterizerDirect::Draw(uint64_t layer_tree_ptr,
const DrawCallback& callback) {
TRACE_EVENT0("flutter", "RasterizerDirect::Draw");
if (!surface_) {
callback.Run();
return;
}
scoped_ptr<compositor::LayerTree> layer_tree(
reinterpret_cast<compositor::LayerTree*>(layer_tree_ptr));
gfx::Size size(layer_tree->frame_size().width(),
layer_tree->frame_size().height());
if (surface_->GetSize() != size)
surface_->Resize(size);
// There is no way for the compositor to know how long the layer tree
// construction took. Fortunately, the layer tree does. Grab that time
// for instrumentation.
paint_context_.engine_time().setLapTime(layer_tree->construction_time());
{
EnsureGLContext();
CHECK(context_->MakeCurrent(surface_.get()));
SkCanvas* canvas = ganesh_canvas_.GetCanvas(
surface_->GetBackingFrameBufferObject(), layer_tree->frame_size());
sky::compositor::PaintContext::ScopedFrame frame =
paint_context_.AcquireFrame(*canvas);
canvas->clear(SK_ColorBLACK);
layer_tree->root_layer()->Paint(frame);
canvas->flush();
surface_->SwapBuffers();
}
// Trace to a file if necessary
bool frameExceededThreshold = false;
uint32_t thresholdInterval = layer_tree->rasterizer_tracing_threshold();
if (thresholdInterval != 0 &&
paint_context_.frame_time().lastLap().InMillisecondsF() >
thresholdInterval * kOneFrameDuration) {
// While rendering the last frame, if we exceeded the tracing threshold
// specified in the layer tree, we force a trace to disk.
frameExceededThreshold = true;
}
const auto& tracingController = Shell::Shared().tracing_controller();
if (frameExceededThreshold || tracingController.picture_tracing_enabled()) {
base::FilePath path = tracingController.PictureTracingPathForCurrentTime();
sky::compositor::PaintContext::ScopedFrame to_file_frame =
paint_context_.AcquireFrame(path.AsUTF8Unsafe(), size);
layer_tree->root_layer()->Paint(to_file_frame);
}
callback.Run();
}
void RasterizerDirect::OnOutputSurfaceDestroyed() {
if (context_) {
CHECK(context_->MakeCurrent(surface_.get()));
ganesh_canvas_.SetGrGLInterface(nullptr);
context_ = nullptr;
}
CHECK(!ganesh_canvas_.IsValid());
CHECK(!context_);
surface_ = nullptr;
}
void RasterizerDirect::EnsureGLContext() {
if (context_)
return;
context_ = gfx::GLContext::CreateGLContext(share_group_.get(), surface_.get(),
gfx::PreferIntegratedGpu);
CHECK(context_) << "GLContext required.";
CHECK(context_->MakeCurrent(surface_.get()));
gr_gl_interface_ = skia::AdoptRef(gfx::CreateInProcessSkiaGLBinding());
ganesh_canvas_.SetGrGLInterface(gr_gl_interface_.get());
}
} // namespace shell
} // namespace sky