rvargas@google.com | dd25504 | 2012-03-05 20:14:17 | [diff] [blame] | 1 | // Copyright (c) 2012 The Chromium Authors. All rights reserved. |
gman@chromium.org | a25fa87 | 2010-03-25 02:57:58 | [diff] [blame] | 2 | // Use of this source code is governed by a BSD-style license that can be |
| 3 | // found in the LICENSE file. |
| 4 | |
| 5 | #ifndef GPU_COMMAND_BUFFER_SERVICE_FRAMEBUFFER_MANAGER_H_ |
| 6 | #define GPU_COMMAND_BUFFER_SERVICE_FRAMEBUFFER_MANAGER_H_ |
| 7 | |
gman@chromium.org | a25fa87 | 2010-03-25 02:57:58 | [diff] [blame] | 8 | #include "base/basictypes.h" |
brettw@chromium.org | 14c1c23 | 2013-06-11 17:52:44 | [diff] [blame] | 9 | #include "base/containers/hash_tables.h" |
levin@chromium.org | 3b63f8f4 | 2011-03-28 01:54:15 | [diff] [blame] | 10 | #include "base/memory/ref_counted.h" |
| 11 | #include "base/memory/scoped_ptr.h" |
gman@chromium.org | a25fa87 | 2010-03-25 02:57:58 | [diff] [blame] | 12 | #include "gpu/command_buffer/service/gl_utils.h" |
rvargas@google.com | dd25504 | 2012-03-05 20:14:17 | [diff] [blame] | 13 | #include "gpu/gpu_export.h" |
gman@chromium.org | a25fa87 | 2010-03-25 02:57:58 | [diff] [blame] | 14 | |
| 15 | namespace gpu { |
| 16 | namespace gles2 { |
| 17 | |
gman@chromium.org | ed9f9cd | 2013-02-27 21:12:35 | [diff] [blame] | 18 | class FramebufferManager; |
gman@chromium.org | 31494b8 | 2013-02-28 10:10:26 | [diff] [blame] | 19 | class Renderbuffer; |
| 20 | class RenderbufferManager; |
| 21 | class Texture; |
piman@chromium.org | 370eaf1 | 2013-05-18 09:19:49 | [diff] [blame] | 22 | class TextureRef; |
gman@chromium.org | 31494b8 | 2013-02-28 10:10:26 | [diff] [blame] | 23 | class TextureManager; |
gman@chromium.org | ed9f9cd | 2013-02-27 21:12:35 | [diff] [blame] | 24 | |
| 25 | // Info about a particular Framebuffer. |
| 26 | class GPU_EXPORT Framebuffer : public base::RefCounted<Framebuffer> { |
| 27 | public: |
| 28 | class Attachment : public base::RefCounted<Attachment> { |
| 29 | public: |
| 30 | virtual GLsizei width() const = 0; |
| 31 | virtual GLsizei height() const = 0; |
| 32 | virtual GLenum internal_format() const = 0; |
| 33 | virtual GLsizei samples() const = 0; |
piman@chromium.org | 62e65f0 | 2013-05-29 22:28:10 | [diff] [blame] | 34 | virtual GLuint object_name() const = 0; |
gman@chromium.org | ed9f9cd | 2013-02-27 21:12:35 | [diff] [blame] | 35 | virtual bool cleared() const = 0; |
| 36 | virtual void SetCleared( |
| 37 | RenderbufferManager* renderbuffer_manager, |
| 38 | TextureManager* texture_manager, |
| 39 | bool cleared) = 0; |
piman@chromium.org | 370eaf1 | 2013-05-18 09:19:49 | [diff] [blame] | 40 | virtual bool IsTexture(TextureRef* texture) const = 0; |
gman@chromium.org | ed9f9cd | 2013-02-27 21:12:35 | [diff] [blame] | 41 | virtual bool IsRenderbuffer( |
| 42 | Renderbuffer* renderbuffer) const = 0; |
| 43 | virtual bool CanRenderTo() const = 0; |
| 44 | virtual void DetachFromFramebuffer() const = 0; |
zmo@chromium.org | 2f143d48 | 2013-03-14 18:04:49 | [diff] [blame] | 45 | virtual bool ValidForAttachmentType( |
| 46 | GLenum attachment_type, uint32 max_color_attachments) = 0; |
gman@chromium.org | ed9f9cd | 2013-02-27 21:12:35 | [diff] [blame] | 47 | virtual void AddToSignature( |
| 48 | TextureManager* texture_manager, std::string* signature) const = 0; |
| 49 | |
| 50 | protected: |
| 51 | friend class base::RefCounted<Attachment>; |
| 52 | virtual ~Attachment() {} |
| 53 | }; |
| 54 | |
| 55 | Framebuffer(FramebufferManager* manager, GLuint service_id); |
| 56 | |
| 57 | GLuint service_id() const { |
| 58 | return service_id_; |
| 59 | } |
| 60 | |
| 61 | bool HasUnclearedAttachment(GLenum attachment) const; |
| 62 | |
| 63 | void MarkAttachmentAsCleared( |
| 64 | RenderbufferManager* renderbuffer_manager, |
| 65 | TextureManager* texture_manager, |
| 66 | GLenum attachment, |
| 67 | bool cleared); |
| 68 | |
| 69 | // Attaches a renderbuffer to a particlar attachment. |
| 70 | // Pass null to detach. |
| 71 | void AttachRenderbuffer( |
| 72 | GLenum attachment, Renderbuffer* renderbuffer); |
| 73 | |
| 74 | // Attaches a texture to a particlar attachment. Pass null to detach. |
| 75 | void AttachTexture( |
piman@chromium.org | 370eaf1 | 2013-05-18 09:19:49 | [diff] [blame] | 76 | GLenum attachment, TextureRef* texture_ref, GLenum target, |
gman@chromium.org | ed9f9cd | 2013-02-27 21:12:35 | [diff] [blame] | 77 | GLint level); |
| 78 | |
| 79 | // Unbinds the given renderbuffer if it is bound. |
| 80 | void UnbindRenderbuffer( |
| 81 | GLenum target, Renderbuffer* renderbuffer); |
| 82 | |
| 83 | // Unbinds the given texture if it is bound. |
| 84 | void UnbindTexture( |
piman@chromium.org | 370eaf1 | 2013-05-18 09:19:49 | [diff] [blame] | 85 | GLenum target, TextureRef* texture_ref); |
gman@chromium.org | ed9f9cd | 2013-02-27 21:12:35 | [diff] [blame] | 86 | |
| 87 | const Attachment* GetAttachment(GLenum attachment) const; |
| 88 | |
| 89 | bool IsDeleted() const { |
| 90 | return deleted_; |
| 91 | } |
| 92 | |
| 93 | void MarkAsValid() { |
| 94 | has_been_bound_ = true; |
| 95 | } |
| 96 | |
| 97 | bool IsValid() const { |
| 98 | return has_been_bound_ && !IsDeleted(); |
| 99 | } |
| 100 | |
| 101 | bool HasDepthAttachment() const; |
| 102 | bool HasStencilAttachment() const; |
| 103 | GLenum GetColorAttachmentFormat() const; |
| 104 | |
| 105 | // Verify all the rules in OpenGL ES 2.0.25 4.4.5 are followed. |
| 106 | // Returns GL_FRAMEBUFFER_COMPLETE if there are no reasons we know we can't |
| 107 | // use this combination of attachments. Otherwise returns the value |
| 108 | // that glCheckFramebufferStatus should return for this set of attachments. |
| 109 | // Note that receiving GL_FRAMEBUFFER_COMPLETE from this function does |
| 110 | // not mean the real OpenGL will consider it framebuffer complete. It just |
| 111 | // means it passed our tests. |
| 112 | GLenum IsPossiblyComplete() const; |
| 113 | |
| 114 | // Implements optimized glGetFramebufferStatus. |
| 115 | GLenum GetStatus(TextureManager* texture_manager, GLenum target) const; |
| 116 | |
| 117 | // Check all attachments are cleared |
| 118 | bool IsCleared() const; |
| 119 | |
zmo@chromium.org | 2f143d48 | 2013-03-14 18:04:49 | [diff] [blame] | 120 | GLenum GetDrawBuffer(GLenum draw_buffer) const; |
| 121 | |
| 122 | void SetDrawBuffers(GLsizei n, const GLenum* bufs); |
| 123 | |
zmo@chromium.org | f3b191b | 2013-06-19 03:43:54 | [diff] [blame] | 124 | // Return true if any draw buffers has an alpha channel. |
| 125 | bool HasAlphaMRT() const; |
| 126 | |
gman@chromium.org | ed9f9cd | 2013-02-27 21:12:35 | [diff] [blame] | 127 | static void ClearFramebufferCompleteComboMap(); |
| 128 | |
ccameron@chromium.org | 97419c0 | 2013-04-10 02:52:38 | [diff] [blame] | 129 | static bool AllowFramebufferComboCompleteMapForTesting() { |
| 130 | return allow_framebuffer_combo_complete_map_; |
| 131 | } |
| 132 | |
gman@chromium.org | ed9f9cd | 2013-02-27 21:12:35 | [diff] [blame] | 133 | private: |
| 134 | friend class FramebufferManager; |
| 135 | friend class base::RefCounted<Framebuffer>; |
| 136 | |
| 137 | ~Framebuffer(); |
| 138 | |
| 139 | void MarkAsDeleted(); |
| 140 | |
| 141 | void MarkAttachmentsAsCleared( |
| 142 | RenderbufferManager* renderbuffer_manager, |
| 143 | TextureManager* texture_manager, |
| 144 | bool cleared); |
| 145 | |
| 146 | void MarkAsComplete(unsigned state_id) { |
| 147 | framebuffer_complete_state_count_id_ = state_id; |
| 148 | } |
| 149 | |
| 150 | unsigned framebuffer_complete_state_count_id() const { |
| 151 | return framebuffer_complete_state_count_id_; |
| 152 | } |
| 153 | |
| 154 | // The managers that owns this. |
| 155 | FramebufferManager* manager_; |
| 156 | |
| 157 | bool deleted_; |
| 158 | |
| 159 | // Service side framebuffer id. |
| 160 | GLuint service_id_; |
| 161 | |
| 162 | // Whether this framebuffer has ever been bound. |
| 163 | bool has_been_bound_; |
| 164 | |
| 165 | // state count when this framebuffer was last checked for completeness. |
| 166 | unsigned framebuffer_complete_state_count_id_; |
| 167 | |
| 168 | // A map of attachments. |
| 169 | typedef base::hash_map<GLenum, scoped_refptr<Attachment> > AttachmentMap; |
| 170 | AttachmentMap attachments_; |
| 171 | |
| 172 | // A map of successful frame buffer combos. If it's in the map |
| 173 | // it should be FRAMEBUFFER_COMPLETE. |
| 174 | typedef base::hash_map<std::string, bool> FramebufferComboCompleteMap; |
| 175 | static FramebufferComboCompleteMap* framebuffer_combo_complete_map_; |
ccameron@chromium.org | 97419c0 | 2013-04-10 02:52:38 | [diff] [blame] | 176 | static bool allow_framebuffer_combo_complete_map_; |
gman@chromium.org | ed9f9cd | 2013-02-27 21:12:35 | [diff] [blame] | 177 | |
dcheng@chromium.org | 40d90a2 | 2013-04-09 03:39:55 | [diff] [blame] | 178 | scoped_ptr<GLenum[]> draw_buffers_; |
zmo@chromium.org | 2f143d48 | 2013-03-14 18:04:49 | [diff] [blame] | 179 | |
gman@chromium.org | ed9f9cd | 2013-02-27 21:12:35 | [diff] [blame] | 180 | DISALLOW_COPY_AND_ASSIGN(Framebuffer); |
| 181 | }; |
| 182 | |
gman@chromium.org | a25fa87 | 2010-03-25 02:57:58 | [diff] [blame] | 183 | // This class keeps track of the frambebuffers and their attached renderbuffers |
| 184 | // so we can correctly clear them. |
rvargas@google.com | dd25504 | 2012-03-05 20:14:17 | [diff] [blame] | 185 | class GPU_EXPORT FramebufferManager { |
gman@chromium.org | a25fa87 | 2010-03-25 02:57:58 | [diff] [blame] | 186 | public: |
zmo@chromium.org | 2f143d48 | 2013-03-14 18:04:49 | [diff] [blame] | 187 | FramebufferManager(uint32 max_draw_buffers, uint32 max_color_attachments); |
gman@chromium.org | d304cbd | 2010-07-01 22:41:16 | [diff] [blame] | 188 | ~FramebufferManager(); |
| 189 | |
| 190 | // Must call before destruction. |
| 191 | void Destroy(bool have_context); |
gman@chromium.org | a25fa87 | 2010-03-25 02:57:58 | [diff] [blame] | 192 | |
gman@chromium.org | ed9f9cd | 2013-02-27 21:12:35 | [diff] [blame] | 193 | // Creates a Framebuffer for the given framebuffer. |
| 194 | void CreateFramebuffer(GLuint client_id, GLuint service_id); |
gman@chromium.org | a25fa87 | 2010-03-25 02:57:58 | [diff] [blame] | 195 | |
| 196 | // Gets the framebuffer info for the given framebuffer. |
gman@chromium.org | ed9f9cd | 2013-02-27 21:12:35 | [diff] [blame] | 197 | Framebuffer* GetFramebuffer(GLuint client_id); |
gman@chromium.org | a25fa87 | 2010-03-25 02:57:58 | [diff] [blame] | 198 | |
| 199 | // Removes a framebuffer info for the given framebuffer. |
gman@chromium.org | ed9f9cd | 2013-02-27 21:12:35 | [diff] [blame] | 200 | void RemoveFramebuffer(GLuint client_id); |
gman@chromium.org | a25fa87 | 2010-03-25 02:57:58 | [diff] [blame] | 201 | |
gman@chromium.org | 6b8cf1a | 2010-05-06 16:13:58 | [diff] [blame] | 202 | // Gets a client id for a given service id. |
| 203 | bool GetClientId(GLuint service_id, GLuint* client_id) const; |
| 204 | |
gman@chromium.org | 968351b | 2011-12-20 08:26:51 | [diff] [blame] | 205 | void MarkAttachmentsAsCleared( |
gman@chromium.org | ed9f9cd | 2013-02-27 21:12:35 | [diff] [blame] | 206 | Framebuffer* framebuffer, |
gman@chromium.org | 968351b | 2011-12-20 08:26:51 | [diff] [blame] | 207 | RenderbufferManager* renderbuffer_manager, |
| 208 | TextureManager* texture_manager); |
| 209 | |
gman@chromium.org | ed9f9cd | 2013-02-27 21:12:35 | [diff] [blame] | 210 | void MarkAsComplete(Framebuffer* framebuffer); |
gman@chromium.org | 968351b | 2011-12-20 08:26:51 | [diff] [blame] | 211 | |
gman@chromium.org | ed9f9cd | 2013-02-27 21:12:35 | [diff] [blame] | 212 | bool IsComplete(Framebuffer* framebuffer); |
gman@chromium.org | 968351b | 2011-12-20 08:26:51 | [diff] [blame] | 213 | |
| 214 | void IncFramebufferStateChangeCount() { |
| 215 | // make sure this is never 0. |
| 216 | framebuffer_state_change_count_ = |
| 217 | (framebuffer_state_change_count_ + 1) | 0x80000000U; |
| 218 | } |
| 219 | |
gman@chromium.org | a25fa87 | 2010-03-25 02:57:58 | [diff] [blame] | 220 | private: |
gman@chromium.org | ed9f9cd | 2013-02-27 21:12:35 | [diff] [blame] | 221 | friend class Framebuffer; |
| 222 | |
gman@chromium.org | 4d8f0dd | 2013-03-09 14:37:06 | [diff] [blame] | 223 | void StartTracking(Framebuffer* framebuffer); |
| 224 | void StopTracking(Framebuffer* framebuffer); |
gman@chromium.org | 7c5e8b1 | 2012-04-06 00:35:24 | [diff] [blame] | 225 | |
gman@chromium.org | a25fa87 | 2010-03-25 02:57:58 | [diff] [blame] | 226 | // Info for each framebuffer in the system. |
gman@chromium.org | ed9f9cd | 2013-02-27 21:12:35 | [diff] [blame] | 227 | typedef base::hash_map<GLuint, scoped_refptr<Framebuffer> > |
gman@chromium.org | 4d8f0dd | 2013-03-09 14:37:06 | [diff] [blame] | 228 | FramebufferMap; |
| 229 | FramebufferMap framebuffers_; |
gman@chromium.org | a25fa87 | 2010-03-25 02:57:58 | [diff] [blame] | 230 | |
gman@chromium.org | 968351b | 2011-12-20 08:26:51 | [diff] [blame] | 231 | // Incremented anytime anything changes that might effect framebuffer |
| 232 | // state. |
| 233 | unsigned framebuffer_state_change_count_; |
| 234 | |
gman@chromium.org | ed9f9cd | 2013-02-27 21:12:35 | [diff] [blame] | 235 | // Counts the number of Framebuffer allocated with 'this' as its manager. |
| 236 | // Allows to check no Framebuffer will outlive this. |
gman@chromium.org | 4d8f0dd | 2013-03-09 14:37:06 | [diff] [blame] | 237 | unsigned int framebuffer_count_; |
gman@chromium.org | 7c5e8b1 | 2012-04-06 00:35:24 | [diff] [blame] | 238 | |
| 239 | bool have_context_; |
| 240 | |
zmo@chromium.org | 2f143d48 | 2013-03-14 18:04:49 | [diff] [blame] | 241 | uint32 max_draw_buffers_; |
| 242 | uint32 max_color_attachments_; |
| 243 | |
gman@chromium.org | a25fa87 | 2010-03-25 02:57:58 | [diff] [blame] | 244 | DISALLOW_COPY_AND_ASSIGN(FramebufferManager); |
| 245 | }; |
| 246 | |
| 247 | } // namespace gles2 |
| 248 | } // namespace gpu |
| 249 | |
| 250 | #endif // GPU_COMMAND_BUFFER_SERVICE_FRAMEBUFFER_MANAGER_H_ |