| // Copyright 2015 The Cobalt Authors. All Rights Reserved. |
| // |
| // Licensed under the Apache License, Version 2.0 (the "License"); |
| // you may not use this file except in compliance with the License. |
| // You may obtain a copy of the License at |
| // |
| // http://www.apache.org/licenses/LICENSE-2.0 |
| // |
| // Unless required by applicable law or agreed to in writing, software |
| // distributed under the License is distributed on an "AS IS" BASIS, |
| // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| // See the License for the specific language governing permissions and |
| // limitations under the License. |
| |
| #ifndef COBALT_RENDERER_BACKEND_EGL_GRAPHICS_CONTEXT_H_ |
| #define COBALT_RENDERER_BACKEND_EGL_GRAPHICS_CONTEXT_H_ |
| |
| #include <memory> |
| |
| #include "base/memory/ref_counted.h" |
| #include "base/optional.h" |
| #include "cobalt/base/polymorphic_downcast.h" |
| #include "cobalt/renderer/backend/egl/render_target.h" |
| #include "cobalt/renderer/backend/egl/resource_context.h" |
| #include "cobalt/renderer/backend/egl/texture.h" |
| #include "cobalt/renderer/backend/egl/texture_data.h" |
| #include "cobalt/renderer/backend/graphics_context.h" |
| #include "cobalt/renderer/egl_and_gles.h" |
| |
| namespace cobalt { |
| namespace renderer { |
| namespace backend { |
| |
| class GraphicsSystemEGL; |
| |
| // Encapsulates a context created via EGL, which implies that it makes |
| // accessible the OpenGL ES API. |
| class GraphicsContextEGL : public GraphicsContext { |
| public: |
| GraphicsContextEGL(GraphicsSystem* parent_system, EGLDisplay display, |
| EGLConfig config, ResourceContext* resource_context); |
| ~GraphicsContextEGL() override; |
| |
| GraphicsSystemEGL* system_egl(); |
| |
| EGLContext GetContext() { return context_; } |
| |
| std::unique_ptr<TextureEGL> CreateTexture( |
| std::unique_ptr<TextureDataEGL> texture_data); |
| |
| std::unique_ptr<TextureEGL> CreateTextureFromRawMemory( |
| const scoped_refptr<ConstRawTextureMemoryEGL>& raw_texture_memory, |
| intptr_t offset, const math::Size& size, GLenum format, |
| int pitch_in_bytes); |
| |
| scoped_refptr<RenderTarget> CreateOffscreenRenderTarget( |
| const math::Size& dimensions) override; |
| |
| scoped_refptr<RenderTarget> CreateDownloadableOffscreenRenderTarget( |
| const math::Size& dimensions) override; |
| |
| void InitializeDebugContext() override; |
| |
| std::unique_ptr<uint8_t[]> DownloadPixelDataAsRGBA( |
| const scoped_refptr<RenderTarget>& render_target) override; |
| |
| void Finish() override; |
| |
| // Helper class to allow one to create a RAII object that will acquire the |
| // current context upon construction and release it upon destruction. |
| class ScopedMakeCurrent { |
| public: |
| explicit ScopedMakeCurrent(GraphicsContextEGL* graphics_context) |
| : graphics_context_(graphics_context), |
| was_current_(graphics_context_->is_current_), |
| previous_current_surface_(graphics_context_->current_surface_.get()) { |
| graphics_context_->MakeCurrent(); |
| } |
| ScopedMakeCurrent(GraphicsContextEGL* graphics_context, |
| RenderTargetEGL* surface) |
| : graphics_context_(graphics_context), |
| was_current_(graphics_context_->is_current_), |
| previous_current_surface_(graphics_context_->current_surface_.get()) { |
| graphics_context_->MakeCurrentWithSurface(surface); |
| } |
| ~ScopedMakeCurrent() { |
| if (was_current_) { |
| graphics_context_->MakeCurrentWithSurface(previous_current_surface_); |
| } else { |
| graphics_context_->ReleaseCurrentContext(); |
| } |
| } |
| |
| private: |
| GraphicsContextEGL* graphics_context_; |
| bool was_current_; |
| RenderTargetEGL* previous_current_surface_; |
| }; |
| |
| // Helper methods to make this context current and associate it with |
| // a surface. In EGL, a context is either not current or it is current and |
| // associated with a surface. For all functionality (e.g. texture creation) |
| // that is provided but does not require a surface binding, null_surface_ is |
| // specified as a surface. |
| void MakeCurrentWithSurface(RenderTargetEGL* surface); |
| |
| // If this context is current, then forcefully rebind its current surface. |
| void ResetCurrentSurface(); |
| |
| // Alternatively, this call can be made to make the context current along |
| // with a null surface. You would be interested in this method if you don't |
| // plan to be making any draw calls, such as if you're setting up a texture. |
| void MakeCurrent(); |
| void ReleaseCurrentContext(); |
| |
| void SwapBuffers(RenderTargetEGL* surface); |
| |
| void Blit(GLuint texture, int x, int y, int width, int height); |
| |
| // Returns if this graphics context is current on the calling thread or not. |
| bool IsCurrent() const { return is_current_; } |
| |
| private: |
| // Make the given surface the current surface. Do error detection in case |
| // the surface has become invalid (which can happen during shutdown). |
| void SafeEglMakeCurrent(RenderTargetEGL* surface); |
| |
| // Performs a test to determine if the pixel data returned by glReadPixels |
| // needs to be vertically flipped or not. This test is expensive, so it |
| // caches the results the first time it is computed and simply returns the |
| // cached value on subsequent calls. |
| bool ComputeReadPixelsNeedVerticalFlip(); |
| |
| // Sets up all structures (like Shaders and vertex buffers) required to |
| // support the Frame::BlitToRenderTarget() functionality. |
| void SetupBlitObjects(); |
| |
| // Clear render target. For security reasons, we clear the display buffer |
| // before rendering to it for the first 3 swaps. This way if the application |
| // does not fill the frame (presumably by accident), this clear color will |
| // appear and not data that was left in the buffer from a previous process |
| // that could be a security risk. This is meant to act as a layer of |
| // security, other layers are 1) The graphics driver should not let our |
| // process access data from other processes in the first place and 2) The |
| // application should completely cover the screen when it is rendered. |
| void SecurityClear(); |
| |
| EGLDisplay display_; |
| |
| // The EGL/OpenGL ES context hosted by this GraphicsContextEGL object. |
| EGLContext context_; |
| EGLConfig config_; |
| |
| // Keep track of whether this context has been set to current or not, and |
| // if so what surface was associated with it. |
| bool is_current_; |
| scoped_refptr<RenderTargetEGL> current_surface_; |
| |
| // A dummy surface that is made current when we wish to execute OpenGL ES |
| // commands that don't actually involve a surface in any way (e.g. texture |
| // creation). |
| scoped_refptr<RenderTargetEGL> null_surface_; |
| |
| // Cache whether or not BGRA texture format is supported by the underlying |
| // OpenGL ES implementation. |
| bool bgra_format_supported_; |
| |
| // Data required to provide BlitToRenderTarget() functionality via OpenGL ES. |
| GLuint blit_vertex_shader_; |
| GLuint blit_fragment_shader_; |
| static const int kBlitPositionAttribute = 0; |
| static const int kBlitTexcoordAttribute = 1; |
| GLuint blit_program_; |
| GLuint blit_vertex_buffer_; |
| |
| // Lazily evaluate whether we need to do a vertical flip when calling |
| // glReadPixels(), and cache the result here when that question is answered. |
| base::Optional<bool> read_pixels_needs_vertical_flip_; |
| |
| // When creating and destroying textures, OpenGL calls need to be made with |
| // a GL context current. By making TextureEGL a friend class of |
| // GraphicsContextEGL, the tight relationship and association of a TextureEGL |
| // with a GraphicsContextEGL is made more explicit, and it also allows |
| // TextureEGL to access MakeCurrent()/ReleaseCurrentContext() methods |
| // of GraphicsContextEGL. |
| friend class TextureEGL; |
| }; |
| |
| } // namespace backend |
| } // namespace renderer |
| } // namespace cobalt |
| |
| #endif // COBALT_RENDERER_BACKEND_EGL_GRAPHICS_CONTEXT_H_ |