base: Change DCHECK_IS_ON to a macro DCHECK_IS_ON().
[chromium-blink-merge.git] / ui / gl / gl_surface_ozone.cc
blobeecd06413876bdbceb9b0d7a50c26c5f40eb6559
1 // Copyright 2014 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.
5 #include "ui/gl/gl_surface.h"
7 #include "base/logging.h"
8 #include "base/memory/ref_counted.h"
9 #include "ui/gfx/native_widget_types.h"
10 #include "ui/gl/gl_context.h"
11 #include "ui/gl/gl_image.h"
12 #include "ui/gl/gl_implementation.h"
13 #include "ui/gl/gl_surface_egl.h"
14 #include "ui/gl/gl_surface_osmesa.h"
15 #include "ui/gl/gl_surface_stub.h"
16 #include "ui/gl/scoped_make_current.h"
17 #include "ui/ozone/public/surface_factory_ozone.h"
18 #include "ui/ozone/public/surface_ozone_egl.h"
20 namespace gfx {
22 namespace {
24 // A thin wrapper around GLSurfaceEGL that owns the EGLNativeWindow
25 class GL_EXPORT GLSurfaceOzoneEGL : public NativeViewGLSurfaceEGL {
26 public:
27 GLSurfaceOzoneEGL(scoped_ptr<ui::SurfaceOzoneEGL> ozone_surface,
28 AcceleratedWidget widget)
29 : NativeViewGLSurfaceEGL(ozone_surface->GetNativeWindow()),
30 ozone_surface_(ozone_surface.Pass()),
31 widget_(widget) {}
33 bool Initialize() override {
34 return Initialize(ozone_surface_->CreateVSyncProvider());
36 bool Resize(const gfx::Size& size) override {
37 if (!ozone_surface_->ResizeNativeWindow(size)) {
38 if (!ReinitializeNativeSurface() ||
39 !ozone_surface_->ResizeNativeWindow(size))
40 return false;
43 return NativeViewGLSurfaceEGL::Resize(size);
45 bool SwapBuffers() override {
46 if (!NativeViewGLSurfaceEGL::SwapBuffers())
47 return false;
49 return ozone_surface_->OnSwapBuffers();
51 bool ScheduleOverlayPlane(int z_order,
52 OverlayTransform transform,
53 GLImage* image,
54 const Rect& bounds_rect,
55 const RectF& crop_rect) override {
56 return image->ScheduleOverlayPlane(
57 widget_, z_order, transform, bounds_rect, crop_rect);
60 private:
61 using NativeViewGLSurfaceEGL::Initialize;
63 ~GLSurfaceOzoneEGL() override {
64 Destroy(); // EGL surface must be destroyed before SurfaceOzone
67 bool ReinitializeNativeSurface() {
68 scoped_ptr<ui::ScopedMakeCurrent> scoped_make_current;
69 GLContext* current_context = GLContext::GetCurrent();
70 bool was_current =
71 current_context && current_context->IsCurrent(this);
72 if (was_current) {
73 scoped_make_current.reset(
74 new ui::ScopedMakeCurrent(current_context, this));
77 Destroy();
78 ozone_surface_ =
79 ui::SurfaceFactoryOzone::GetInstance()->CreateEGLSurfaceForWidget(
80 widget_).Pass();
81 if (!ozone_surface_) {
82 LOG(ERROR) << "Failed to create native surface.";
83 return false;
86 window_ = ozone_surface_->GetNativeWindow();
87 if (!Initialize()) {
88 LOG(ERROR) << "Failed to initialize.";
89 return false;
92 return true;
95 // The native surface. Deleting this is allowed to free the EGLNativeWindow.
96 scoped_ptr<ui::SurfaceOzoneEGL> ozone_surface_;
97 AcceleratedWidget widget_;
99 DISALLOW_COPY_AND_ASSIGN(GLSurfaceOzoneEGL);
102 class GL_EXPORT GLSurfaceOzoneSurfaceless : public SurfacelessEGL {
103 public:
104 GLSurfaceOzoneSurfaceless(scoped_ptr<ui::SurfaceOzoneEGL> ozone_surface,
105 AcceleratedWidget widget)
106 : SurfacelessEGL(gfx::Size()),
107 ozone_surface_(ozone_surface.Pass()),
108 widget_(widget) {}
110 bool Initialize() override {
111 if (!SurfacelessEGL::Initialize())
112 return false;
113 vsync_provider_ = ozone_surface_->CreateVSyncProvider();
114 if (!vsync_provider_)
115 return false;
116 return true;
118 bool Resize(const gfx::Size& size) override {
119 if (!ozone_surface_->ResizeNativeWindow(size))
120 return false;
122 return SurfacelessEGL::Resize(size);
124 bool SwapBuffers() override {
125 // TODO: this should be replaced by a fence when supported by the driver.
126 glFlush();
127 return ozone_surface_->OnSwapBuffers();
129 bool ScheduleOverlayPlane(int z_order,
130 OverlayTransform transform,
131 GLImage* image,
132 const Rect& bounds_rect,
133 const RectF& crop_rect) override {
134 return image->ScheduleOverlayPlane(
135 widget_, z_order, transform, bounds_rect, crop_rect);
137 bool IsOffscreen() override { return false; }
138 VSyncProvider* GetVSyncProvider() override { return vsync_provider_.get(); }
139 bool SupportsPostSubBuffer() override { return true; }
140 bool PostSubBuffer(int x, int y, int width, int height) override {
141 // The actual sub buffer handling is handled at higher layers.
142 SwapBuffers();
143 return true;
145 bool SwapBuffersAsync(const SwapCompletionCallback& callback) override {
146 // TODO: this should be replaced by a fence when supported by the driver.
147 glFlush();
148 return ozone_surface_->OnSwapBuffersAsync(callback);
150 bool PostSubBufferAsync(int x,
151 int y,
152 int width,
153 int height,
154 const SwapCompletionCallback& callback) override {
155 return SwapBuffersAsync(callback);
158 private:
159 ~GLSurfaceOzoneSurfaceless() override {
160 Destroy(); // EGL surface must be destroyed before SurfaceOzone
163 // The native surface. Deleting this is allowed to free the EGLNativeWindow.
164 scoped_ptr<ui::SurfaceOzoneEGL> ozone_surface_;
165 AcceleratedWidget widget_;
166 scoped_ptr<VSyncProvider> vsync_provider_;
168 DISALLOW_COPY_AND_ASSIGN(GLSurfaceOzoneSurfaceless);
171 } // namespace
173 // static
174 bool GLSurface::InitializeOneOffInternal() {
175 switch (GetGLImplementation()) {
176 case kGLImplementationEGLGLES2:
177 if (!GLSurfaceEGL::InitializeOneOff()) {
178 LOG(ERROR) << "GLSurfaceEGL::InitializeOneOff failed.";
179 return false;
182 return true;
183 case kGLImplementationOSMesaGL:
184 case kGLImplementationMockGL:
185 return true;
186 default:
187 return false;
191 // static
192 scoped_refptr<GLSurface> GLSurface::CreateViewGLSurface(
193 gfx::AcceleratedWidget window) {
194 if (GetGLImplementation() == kGLImplementationOSMesaGL) {
195 scoped_refptr<GLSurface> surface(new GLSurfaceOSMesaHeadless());
196 if (!surface->Initialize())
197 return NULL;
198 return surface;
200 DCHECK(GetGLImplementation() == kGLImplementationEGLGLES2);
201 if (window != kNullAcceleratedWidget) {
202 scoped_refptr<GLSurface> surface;
203 if (GLSurfaceEGL::IsEGLSurfacelessContextSupported() &&
204 ui::SurfaceFactoryOzone::GetInstance()
205 ->CanShowPrimaryPlaneAsOverlay()) {
206 scoped_ptr<ui::SurfaceOzoneEGL> surface_ozone =
207 ui::SurfaceFactoryOzone::GetInstance()
208 ->CreateSurfacelessEGLSurfaceForWidget(window);
209 if (!surface_ozone)
210 return NULL;
211 surface = new GLSurfaceOzoneSurfaceless(surface_ozone.Pass(), window);
212 } else {
213 scoped_ptr<ui::SurfaceOzoneEGL> surface_ozone =
214 ui::SurfaceFactoryOzone::GetInstance()->CreateEGLSurfaceForWidget(
215 window);
216 if (!surface_ozone)
217 return NULL;
219 surface = new GLSurfaceOzoneEGL(surface_ozone.Pass(), window);
221 if (!surface->Initialize())
222 return NULL;
223 return surface;
224 } else {
225 scoped_refptr<GLSurface> surface = new GLSurfaceStub();
226 if (surface->Initialize())
227 return surface;
229 return NULL;
232 // static
233 scoped_refptr<GLSurface> GLSurface::CreateOffscreenGLSurface(
234 const gfx::Size& size) {
235 switch (GetGLImplementation()) {
236 case kGLImplementationOSMesaGL: {
237 scoped_refptr<GLSurface> surface(
238 new GLSurfaceOSMesa(OSMesaSurfaceFormatBGRA, size));
239 if (!surface->Initialize())
240 return NULL;
242 return surface;
244 case kGLImplementationEGLGLES2: {
245 scoped_refptr<GLSurface> surface;
246 if (GLSurfaceEGL::IsEGLSurfacelessContextSupported() &&
247 (size.width() == 0 && size.height() == 0)) {
248 surface = new SurfacelessEGL(size);
249 } else
250 surface = new PbufferGLSurfaceEGL(size);
252 if (!surface->Initialize())
253 return NULL;
254 return surface;
256 default:
257 NOTREACHED();
258 return NULL;
262 EGLNativeDisplayType GetPlatformDefaultEGLNativeDisplay() {
263 return ui::SurfaceFactoryOzone::GetInstance()->GetNativeDisplay();
266 } // namespace gfx