StapleGL
Header-only C++20 OpenGL wrapper
Loading...
Searching...
No Matches
framebuffer.hpp
Go to the documentation of this file.
1
16#pragma once
17
18#include "gl_functions.hpp"
19#include "renderbuffer.hpp"
20#include "texture.hpp"
21#include "utility.hpp"
22
23#include <array>
24#include <cstdint>
25#include <exception>
26#include <functional>
27#include <optional>
28#include <span>
29
30#ifdef STAPLEGL_DEBUG
31#include <cstdio>
32#endif // STAPLEGL_DEBUG
33
34namespace staplegl {
35
40enum class fbo_attachment : std::uint8_t {
41 NONE = 0x00,
45};
46
52
53public:
54 framebuffer() noexcept;
56
57 framebuffer(const framebuffer&) = delete;
58 auto operator=(const framebuffer&) -> framebuffer& = delete;
59
60 framebuffer(framebuffer&& other) noexcept;
61 auto operator=(framebuffer&& other) noexcept -> framebuffer&;
62
75
88 void set_texture(texture_2d const& tex, size_t index = 0) const;
89
97 static void set_viewport(resolution res);
98
107 static void bind_default()
108 {
109 glBindFramebuffer(GL_FRAMEBUFFER, 0);
110 }
111
112 void bind() const;
113 static void unbind();
114
129 static auto assert_completeness() -> bool
130 {
131 auto status = glCheckFramebufferStatus(GL_FRAMEBUFFER);
132
133 return status == GL_FRAMEBUFFER_COMPLETE;
134 }
135
143 static void transfer_data(framebuffer const& src, framebuffer const& dst, resolution res)
144 {
145 glBindFramebuffer(GL_READ_FRAMEBUFFER, src.id());
146 glBindFramebuffer(GL_DRAW_FRAMEBUFFER, dst.id());
147
148 glBlitFramebuffer(0, 0, res.width, res.height, 0, 0, res.width, res.height, GL_COLOR_BUFFER_BIT, GL_NEAREST);
149
150 glBindFramebuffer(GL_FRAMEBUFFER, 0);
151 }
152
158 [[nodiscard]] constexpr auto id() const -> std::uint32_t
159 {
160 return m_id;
161 }
162
163 [[nodiscard]] constexpr auto get_renderbuffer() const -> const std::optional<renderbuffer>&;
164
165 [[nodiscard]] constexpr auto attachment() const -> fbo_attachment
166 {
167 return m_attachment;
168 }
169
170private:
171 std::uint32_t m_id {};
173 std::optional<renderbuffer> m_renderbuffer {};
174
175}; // class framebuffer
176
183{
184 glGenFramebuffers(1, &m_id);
185}
186
188{
189 if (m_id != 0) {
190 glDeleteFramebuffers(1, &m_id);
191 }
192}
193
199inline framebuffer::framebuffer(framebuffer&& other) noexcept
200 : m_id(other.m_id)
201 , m_attachment(other.m_attachment)
202 , m_renderbuffer(std::move(other.m_renderbuffer))
203{
204 other.m_id = 0;
205}
206
213inline auto framebuffer::operator=(framebuffer&& other) noexcept -> framebuffer&
214{
215 if (this != &other) {
216 m_id = other.m_id;
217 m_attachment = other.m_attachment;
218 m_renderbuffer = std::move(other.m_renderbuffer);
219 other.m_id = 0;
220 }
221 return *this;
222}
223
233{
235
237
238 switch (attachment) {
241 break;
244 break;
247 break;
248 default:
249#ifdef STAPLEGL_DEBUG
250 std::fprintf(stderr, STAPLEGL_LINEINFO "invalid attachment enum %d for renderbuffer\n", static_cast<std::uint32_t>(attachment));
251#endif
252 std::terminate(); // crash and burn, this is unrecoverable
253 }
254
255 m_attachment = attachment; // in some cases this is a redundant operation, fair enough.
256 m_renderbuffer.emplace(res, type, samples);
257
258 glFramebufferRenderbuffer(GL_FRAMEBUFFER, static_cast<std::uint32_t>(type), GL_RENDERBUFFER, m_renderbuffer->id());
259 } else {
260 // unbind the old framebuffer, if present
261
263 return; // nothing to unbind
264 }
265
267
268 switch (m_attachment) {
271 break;
274 break;
277 break;
278 default:
279 break;
280 }
281
283
284 glFramebufferRenderbuffer(GL_FRAMEBUFFER,
285 static_cast<std::uint32_t>(type),
286 GL_RENDERBUFFER, 0);
287 }
288}
289
290inline void framebuffer::set_texture(staplegl::texture_2d const& tex, size_t index) const
291{
292
293 glFramebufferTexture2D(GL_FRAMEBUFFER, GL_COLOR_ATTACHMENT0 + index, tex.antialias().type, tex.id(), 0);
294}
295
297{
298 glViewport(0, 0, res.width, res.height);
299}
300
301constexpr auto framebuffer::get_renderbuffer() const -> const std::optional<renderbuffer>&
302{
303 return m_renderbuffer;
304}
305
306inline void framebuffer::bind() const
307{
308 glBindFramebuffer(GL_FRAMEBUFFER, m_id);
309}
310
312{
313 glBindFramebuffer(GL_FRAMEBUFFER, 0);
314}
315
316} // namespace staplegl
Framebuffer Object (FBO) wrapper.
static void set_viewport(resolution res)
Resize the OpenGL viewport.
framebuffer() noexcept
Construct a new framebuffer::framebuffer object.
static void bind_default()
Bind the default framebuffer.
void set_renderbuffer(resolution res, fbo_attachment attachment=fbo_attachment::ATTACH_DEPTH_STENCIL_BUFFER, tex_samples samples=tex_samples::MSAA_X1)
Set the renderbuffer object.
auto operator=(const framebuffer &) -> framebuffer &=delete
constexpr auto get_renderbuffer() const -> const std::optional< renderbuffer > &
constexpr auto id() const -> std::uint32_t
Get the id of the framebuffer.
framebuffer(const framebuffer &)=delete
void set_texture(texture_2d const &tex, size_t index=0) const
Set a texture as the color attachment of the framebuffer.
static void transfer_data(framebuffer const &src, framebuffer const &dst, resolution res)
Transfer the contents of a framebuffer to another.
fbo_attachment m_attachment
static auto assert_completeness() -> bool
Check if the framebuffer is complete.
constexpr auto attachment() const -> fbo_attachment
std::optional< renderbuffer > m_renderbuffer
Render Buffer Object (RBO) wrapper.
attachment_type
Renderbuffer attachment type.
2D texture wrapper.
Definition texture.hpp:97
constexpr auto id() const -> std::uint32_t
Get the id of the texture object.
Definition texture.hpp:250
constexpr auto antialias() const -> texture_antialias
Get the texture antialias struct.
Definition texture.hpp:280
Loads OpenGL functions.
tex_samples
An enum that represents the number of samples for a texture.
Definition utility.hpp:38
fbo_attachment
enum class for framebuffer attachments.
Render Buffer Object (RBO) wrapper.
A struct that represents an image's dimensions.
Definition utility.hpp:28
std::int32_t height
Definition utility.hpp:31
std::int32_t width
Definition utility.hpp:30
Texture abstraction.
Utility functions for parsing files.