OLD | NEW |
| (Empty) |
1 // Copyright 2015 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 "services/surfaces/display_impl.h" | |
6 | |
7 #include "cc/output/compositor_frame.h" | |
8 #include "cc/surfaces/display.h" | |
9 #include "mojo/converters/geometry/geometry_type_converters.h" | |
10 #include "mojo/converters/surfaces/surfaces_type_converters.h" | |
11 #include "services/surfaces/context_provider_mojo.h" | |
12 #include "services/surfaces/surfaces_output_surface.h" | |
13 #include "services/surfaces/surfaces_scheduler.h" | |
14 | |
15 namespace surfaces { | |
16 namespace { | |
17 void CallCallback(const mojo::Closure& callback, cc::SurfaceDrawStatus status) { | |
18 callback.Run(); | |
19 } | |
20 } | |
21 | |
22 DisplayImpl::DisplayImpl(cc::SurfaceManager* manager, | |
23 cc::SurfaceId cc_id, | |
24 SurfacesScheduler* scheduler, | |
25 mojo::ContextProviderPtr context_provider, | |
26 mojo::ResourceReturnerPtr returner, | |
27 mojo::InterfaceRequest<mojo::Display> display_request) | |
28 : manager_(manager), | |
29 factory_(manager, this), | |
30 cc_id_(cc_id), | |
31 scheduler_(scheduler), | |
32 context_provider_(context_provider.Pass()), | |
33 returner_(returner.Pass()), | |
34 viewport_param_binding_(this), | |
35 display_binding_(this, display_request.Pass()) { | |
36 mojo::ViewportParameterListenerPtr viewport_parameter_listener; | |
37 viewport_param_binding_.Bind(GetProxy(&viewport_parameter_listener)); | |
38 context_provider_->Create( | |
39 viewport_parameter_listener.Pass(), | |
40 base::Bind(&DisplayImpl::OnContextCreated, base::Unretained(this))); | |
41 factory_.Create(cc_id_); | |
42 } | |
43 | |
44 void DisplayImpl::OnContextCreated(mojo::CommandBufferPtr gles2_client) { | |
45 DCHECK(!display_); | |
46 | |
47 cc::RendererSettings settings; | |
48 display_.reset(new cc::Display(this, manager_, nullptr, nullptr, settings)); | |
49 scheduler_->AddDisplay(display_.get()); | |
50 display_->Initialize(make_scoped_ptr(new mojo::DirectOutputSurface( | |
51 new mojo::ContextProviderMojo( | |
52 gles2_client.PassInterface().PassHandle())))); | |
53 display_->Resize(last_submitted_frame_size_); | |
54 | |
55 display_->SetSurfaceId(cc_id_, 1.f); | |
56 if (pending_frame_) | |
57 Draw(); | |
58 } | |
59 | |
60 DisplayImpl::~DisplayImpl() { | |
61 factory_.Destroy(cc_id_); | |
62 if (display_) { | |
63 scheduler_->RemoveDisplay(display_.get()); | |
64 } | |
65 } | |
66 | |
67 void DisplayImpl::SubmitFrame(mojo::FramePtr frame, | |
68 const SubmitFrameCallback& callback) { | |
69 DCHECK(pending_callback_.is_null()); | |
70 pending_frame_ = frame.Pass(); | |
71 pending_callback_ = callback; | |
72 if (display_) | |
73 Draw(); | |
74 } | |
75 | |
76 void DisplayImpl::Draw() { | |
77 gfx::Size frame_size = | |
78 pending_frame_->passes[0]->output_rect.To<gfx::Rect>().size(); | |
79 last_submitted_frame_size_ = frame_size; | |
80 display_->Resize(frame_size); | |
81 factory_.SubmitFrame(cc_id_, | |
82 pending_frame_.To<scoped_ptr<cc::CompositorFrame>>(), | |
83 base::Bind(&CallCallback, pending_callback_)); | |
84 scheduler_->SetNeedsDraw(); | |
85 pending_frame_.reset(); | |
86 pending_callback_.reset(); | |
87 } | |
88 | |
89 void DisplayImpl::DisplayDamaged() { | |
90 } | |
91 | |
92 void DisplayImpl::DidSwapBuffers() { | |
93 } | |
94 | |
95 void DisplayImpl::DidSwapBuffersComplete() { | |
96 } | |
97 | |
98 void DisplayImpl::CommitVSyncParameters(base::TimeTicks timebase, | |
99 base::TimeDelta interval) { | |
100 } | |
101 | |
102 void DisplayImpl::OutputSurfaceLost() { | |
103 // If our OutputSurface is lost we can't draw until we get a new one. For now, | |
104 // destroy the display and create a new one when our ContextProvider provides | |
105 // a new one. | |
106 // TODO: This is more violent than necessary - we could simply remove this | |
107 // display from the scheduler's set and pass a new context in to the | |
108 // OutputSurface. It should be able to reinitialize properly. | |
109 scheduler_->RemoveDisplay(display_.get()); | |
110 display_.reset(); | |
111 viewport_param_binding_.Close(); | |
112 mojo::ViewportParameterListenerPtr viewport_parameter_listener; | |
113 viewport_param_binding_.Bind(GetProxy(&viewport_parameter_listener)); | |
114 context_provider_->Create( | |
115 viewport_parameter_listener.Pass(), | |
116 base::Bind(&DisplayImpl::OnContextCreated, base::Unretained(this))); | |
117 } | |
118 | |
119 void DisplayImpl::OnVSyncParametersUpdated(int64_t timebase, int64_t interval) { | |
120 scheduler_->OnVSyncParametersUpdated( | |
121 base::TimeTicks::FromInternalValue(timebase), | |
122 base::TimeDelta::FromInternalValue(interval)); | |
123 } | |
124 | |
125 void DisplayImpl::ReturnResources(const cc::ReturnedResourceArray& resources) { | |
126 if (resources.empty()) | |
127 return; | |
128 DCHECK(returner_); | |
129 | |
130 auto ret = mojo::Array<mojo::ReturnedResourcePtr>::New(resources.size()); | |
131 for (size_t i = 0; i < resources.size(); ++i) { | |
132 ret[i] = mojo::ReturnedResource::From(resources[i]); | |
133 } | |
134 returner_->ReturnResources(ret.Pass()); | |
135 } | |
136 | |
137 } // namespace surfaces | |
OLD | NEW |