| OLD | NEW |
| (Empty) |
| 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. | |
| 4 | |
| 5 #include "ui/gl/gl_surface.h" | |
| 6 | |
| 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" | |
| 19 | |
| 20 namespace gfx { | |
| 21 | |
| 22 namespace { | |
| 23 | |
| 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) {} | |
| 32 | |
| 33 bool Initialize() override { | |
| 34 return Initialize(ozone_surface_->CreateVSyncProvider()); | |
| 35 } | |
| 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; | |
| 41 } | |
| 42 | |
| 43 return NativeViewGLSurfaceEGL::Resize(size); | |
| 44 } | |
| 45 bool SwapBuffers() override { | |
| 46 if (!NativeViewGLSurfaceEGL::SwapBuffers()) | |
| 47 return false; | |
| 48 | |
| 49 return ozone_surface_->OnSwapBuffers(); | |
| 50 } | |
| 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); | |
| 58 } | |
| 59 | |
| 60 private: | |
| 61 using NativeViewGLSurfaceEGL::Initialize; | |
| 62 | |
| 63 ~GLSurfaceOzoneEGL() override { | |
| 64 Destroy(); // EGL surface must be destroyed before SurfaceOzone | |
| 65 } | |
| 66 | |
| 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)); | |
| 75 } | |
| 76 | |
| 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; | |
| 84 } | |
| 85 | |
| 86 window_ = ozone_surface_->GetNativeWindow(); | |
| 87 if (!Initialize()) { | |
| 88 LOG(ERROR) << "Failed to initialize."; | |
| 89 return false; | |
| 90 } | |
| 91 | |
| 92 return true; | |
| 93 } | |
| 94 | |
| 95 // The native surface. Deleting this is allowed to free the EGLNativeWindow. | |
| 96 scoped_ptr<ui::SurfaceOzoneEGL> ozone_surface_; | |
| 97 AcceleratedWidget widget_; | |
| 98 | |
| 99 DISALLOW_COPY_AND_ASSIGN(GLSurfaceOzoneEGL); | |
| 100 }; | |
| 101 | |
| 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) {} | |
| 109 | |
| 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; | |
| 117 } | |
| 118 bool Resize(const gfx::Size& size) override { | |
| 119 if (!ozone_surface_->ResizeNativeWindow(size)) | |
| 120 return false; | |
| 121 | |
| 122 return SurfacelessEGL::Resize(size); | |
| 123 } | |
| 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(); | |
| 128 } | |
| 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); | |
| 136 } | |
| 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; | |
| 144 } | |
| 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); | |
| 149 } | |
| 150 bool PostSubBufferAsync(int x, | |
| 151 int y, | |
| 152 int width, | |
| 153 int height, | |
| 154 const SwapCompletionCallback& callback) override { | |
| 155 return SwapBuffersAsync(callback); | |
| 156 } | |
| 157 | |
| 158 private: | |
| 159 ~GLSurfaceOzoneSurfaceless() override { | |
| 160 Destroy(); // EGL surface must be destroyed before SurfaceOzone | |
| 161 } | |
| 162 | |
| 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_; | |
| 167 | |
| 168 DISALLOW_COPY_AND_ASSIGN(GLSurfaceOzoneSurfaceless); | |
| 169 }; | |
| 170 | |
| 171 } // namespace | |
| 172 | |
| 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; | |
| 180 } | |
| 181 | |
| 182 return true; | |
| 183 case kGLImplementationOSMesaGL: | |
| 184 case kGLImplementationMockGL: | |
| 185 return true; | |
| 186 default: | |
| 187 return false; | |
| 188 } | |
| 189 } | |
| 190 | |
| 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; | |
| 199 } | |
| 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; | |
| 218 | |
| 219 surface = new GLSurfaceOzoneEGL(surface_ozone.Pass(), window); | |
| 220 } | |
| 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; | |
| 228 } | |
| 229 return NULL; | |
| 230 } | |
| 231 | |
| 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; | |
| 241 | |
| 242 return surface; | |
| 243 } | |
| 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); | |
| 251 | |
| 252 if (!surface->Initialize()) | |
| 253 return NULL; | |
| 254 return surface; | |
| 255 } | |
| 256 default: | |
| 257 NOTREACHED(); | |
| 258 return NULL; | |
| 259 } | |
| 260 } | |
| 261 | |
| 262 EGLNativeDisplayType GetPlatformDefaultEGLNativeDisplay() { | |
| 263 return ui::SurfaceFactoryOzone::GetInstance()->GetNativeDisplay(); | |
| 264 } | |
| 265 | |
| 266 } // namespace gfx | |
| OLD | NEW |