Gamp v0.0.7-36-g24b1eb6
Gamp: Graphics, Audio, Multimedia and Processing
Loading...
Searching...
No Matches
gamp_sdl2_gl.cpp
Go to the documentation of this file.
1/*
2 * Author: Sven Gothel <sgothel@jausoft.com>
3 * Copyright Gothel Software e.K.
4 *
5 * SPDX-License-Identifier: MIT
6 *
7 * This Source Code Form is subject to the terms of the MIT License
8 * If a copy of the MIT was not distributed with this file,
9 * you can obtain one at https://opensource.org/license/mit/.
10 */
11
12#include <gamp/Gamp.hpp>
13#include <gamp/GampTypes.hpp>
20#include <gamp/wt/Window.hpp>
21#include <memory>
22
23#include <SDL2/SDL.h>
24#include <SDL2/SDL_video.h>
25
26using namespace jau::int_literals;
27using namespace jau::fractions_i64_literals;
28using namespace jau::enums;
29
30using namespace gamp;
31using namespace gamp::wt;
32using namespace gamp::render;
33using namespace gamp::render::gl;
34
35static bool GetGLAttribute(SDL_GLattr attr, int& value) noexcept {
36 return 0 == SDL_GL_GetAttribute(attr, &value);
37}
38
39bool Surface::setSwapIntervalImpl(int v) noexcept {
40 if( 0 == SDL_GL_SetSwapInterval( v ) ) {
41 m_swapInterval = v;
42 return true;
43 }
44 if( -1 == v && 0 == SDL_GL_SetSwapInterval( 1 ) ) {
45 m_swapInterval = 1;
46 return true;
47 }
48 m_swapInterval = 0;
49 return false;
50}
51
52CapabilitiesPtr Surface::retrieveCaps(const wt::SurfaceRef& surface) noexcept {
53 if( !surface->isValid() ) {
54 return nullptr;
55 }
56 // SDL_Window* sdl_win = reinterpret_cast<SDL_Window*>(surface->surfaceHandle()); // NOLINT
57 GLCapabilitiesPtr caps = std::make_unique<GLCapabilities>();
58 bool ok;
59 int v;
60
61 ok = GetGLAttribute(SDL_GL_RED_SIZE, caps->redBits());
62 ok = ok && GetGLAttribute(SDL_GL_GREEN_SIZE, caps->greenBits());
63 ok = ok && GetGLAttribute(SDL_GL_BLUE_SIZE, caps->blueBits());
64 ok = ok && GetGLAttribute(SDL_GL_ALPHA_SIZE, caps->alphaBits());
65 if( !ok ) {
66 return nullptr;
67 }
68
69 ok = GetGLAttribute(SDL_GL_DOUBLEBUFFER, v);
70 if( ok ) { caps->setDoubleBuffered(v); }
71
72 GetGLAttribute(SDL_GL_DEPTH_SIZE, caps->depthBits());
73 GetGLAttribute(SDL_GL_STENCIL_SIZE, caps->stencilBits());
74 GetGLAttribute(SDL_GL_ACCUM_RED_SIZE, caps->accumRedBits());
75 GetGLAttribute(SDL_GL_ACCUM_GREEN_SIZE, caps->accumGreenBits());
76 GetGLAttribute(SDL_GL_ACCUM_BLUE_SIZE, caps->accumBlueBits());
77 GetGLAttribute(SDL_GL_ACCUM_ALPHA_SIZE, caps->accumAlphaBits());
78 ok = GetGLAttribute(SDL_GL_STEREO, v);
79 if( ok ) { caps->setStereo(v); }
80 return caps;
81}
82
84 const gamp::render::RenderProfile& profile,
85 const gamp::render::RenderContextFlags& contextFlags,
86 gamp::render::RenderContext* shareWith) noexcept {
87 // const wt::SurfaceRef& surface, render::gl::GLProfileMask profile, render::gl::GLContextFlags contextFlags, render::gl::GL* shareWith
88 if( !surface->isValid() ) {
89 printf("SDL: Error creating GL context: Invalid surface: %s\n", surface->toString().c_str());
90 return nullptr;
91 }
93 if( verbose ) {
94 printf("Surface::createContext: shareWith %p, profile %s, ctx %s, surface %s\n",
95 (void*)shareWith, profile.toString().c_str(), to_string(contextFlags).c_str(), surface->toString().c_str());
96 }
97 SDL_Window* sdl_win = reinterpret_cast<SDL_Window*>(surface->surfaceHandle()); // NOLINT
98 // Create OpenGL context on SDL window
99 surface->setSwapIntervalImpl( surface->swapInterval() );
100 {
101 int ctxFlags = 0;
103 ctxFlags |= SDL_GL_CONTEXT_DEBUG_FLAG;
104 }
106 ctxFlags |= SDL_GL_CONTEXT_ROBUST_ACCESS_FLAG;
107 }
108 SDL_GL_SetAttribute(SDL_GL_CONTEXT_FLAGS, ctxFlags);
109 }
110 if( shareWith ) {
111 // FIXME: shareWith should be current!
112 SDL_GL_SetAttribute(SDL_GL_SHARE_WITH_CURRENT_CONTEXT, 1);
113 }
114 if( verbose ) {
115 printf("Surface::createContext.1: surface %s\n", surface->toString().c_str());
116 }
117
118 GLProfile glp_in;
119 bool use_glp_core = false;
120 if( profile.signature() == GLProfile::GLSignature() ) {
121 glp_in = GLProfile::downcast(profile);
122 if( !glp_in.isGLES() ) {
123 use_glp_core = true;
124 SDL_GL_SetAttribute(SDL_GL_CONTEXT_PROFILE_MASK, SDL_GL_CONTEXT_PROFILE_CORE);
125 if( glp_in.version().major() >= 4 ) {
126 SDL_GL_SetAttribute(SDL_GL_CONTEXT_MAJOR_VERSION, 4);
127 } else if( glp_in.version().major() >= 3 ) {
128 SDL_GL_SetAttribute(SDL_GL_CONTEXT_MAJOR_VERSION, 3);
129 }
130 }
131 }
132 if( !use_glp_core ) {
133 SDL_GL_SetAttribute(SDL_GL_CONTEXT_PROFILE_MASK, SDL_GL_CONTEXT_PROFILE_ES);
134 SDL_GL_SetAttribute(SDL_GL_CONTEXT_MAJOR_VERSION, 3);
135 SDL_GL_SetAttribute(SDL_GL_CONTEXT_MINOR_VERSION, 0);
136 }
137 SDL_GLContext sdl_glc = SDL_GL_CreateContext(sdl_win);
138 if (nullptr == sdl_glc) {
139 if( use_glp_core ) {
140 printf("SDL: Error creating %s context: %s\n", glp_in.toString().c_str(), SDL_GetError());
141 return nullptr;
142 }
143 printf("SDL: Error creating GL ES 3 context: %s\n", SDL_GetError());
144 SDL_GL_SetAttribute(SDL_GL_CONTEXT_PROFILE_MASK, SDL_GL_CONTEXT_PROFILE_ES);
145 SDL_GL_SetAttribute(SDL_GL_CONTEXT_MAJOR_VERSION, 2);
146 SDL_GL_SetAttribute(SDL_GL_CONTEXT_MINOR_VERSION, 0);
147 sdl_glc = SDL_GL_CreateContext(sdl_win);
148 if (nullptr == sdl_glc) {
149 printf("SDL: Error creating GL ES 2 context: %s\n", SDL_GetError());
150 return nullptr;
151 }
152 }
153 if (0 != SDL_GL_MakeCurrent(sdl_win, sdl_glc)) {
154 printf("SDL: Error making GL context current: %s\n", SDL_GetError());
155 SDL_GL_DeleteContext(sdl_glc);
156 return nullptr;
157 }
158 const char* gl_version_cstr = reinterpret_cast<const char*>( glGetString(GL_VERSION) );
159 if (nullptr == gl_version_cstr) {
160 printf("SDL: Error retrieving GL version: %s\n", SDL_GetError());
161 SDL_GL_DeleteContext(sdl_glc);
162 return nullptr;
163 }
164 bool ok;
165 int major, minor, nContextFlags, nProfileMask;
166
167 ok = GetGLAttribute(SDL_GL_CONTEXT_MAJOR_VERSION, major);
168 ok = ok && GetGLAttribute(SDL_GL_CONTEXT_MINOR_VERSION, minor);
169 ok = ok && GetGLAttribute(SDL_GL_CONTEXT_FLAGS, nContextFlags);
170 ok = ok && GetGLAttribute(SDL_GL_CONTEXT_PROFILE_MASK, nProfileMask);
171 if( !ok ) {
172 printf("SDL: Error retrieving GL context version information: %s\n", SDL_GetError());
173 SDL_GL_DeleteContext(sdl_glc);
174 return nullptr;
175 }
177 if( 0 != ( nProfileMask & SDL_GL_CONTEXT_PROFILE_CORE ) ) {
178 profileMask |= render::gl::GLProfileMask::core;
179 } else if( 0 != ( nProfileMask & SDL_GL_CONTEXT_PROFILE_COMPATIBILITY ) ) {
181 } else if( 0 != ( nProfileMask & SDL_GL_CONTEXT_PROFILE_ES ) ) {
182 profileMask |= render::gl::GLProfileMask::es;
183 }
185 if( 0 != ( nContextFlags & SDL_GL_CONTEXT_FORWARD_COMPATIBLE_FLAG ) ) {
187 }
188 if( 0 != ( nContextFlags & SDL_GL_CONTEXT_DEBUG_FLAG ) ) {
190 }
191 if( 0 != ( nContextFlags & SDL_GL_CONTEXT_ROBUST_ACCESS_FLAG ) ) {
193 }
194 GLVersionNumber glv = GLVersionNumber::create(gl_version_cstr);
195 render::gl::GLProfile glp_out(glv, profileMask);
196 if( verbose ) {
197 render::gl::GLProfile glp_out0(jau::util::VersionNumber(major, minor, 0), profileMask);
198 printf("Surface::createContext.2: GLProfile (query) %s\n", glp_out0.toString().c_str());
199 printf("Surface::createContext.2: GLProfile (version) %s\n", glp_out.toString().c_str());
200 }
201 return render::gl::GLContext::create(surface, (handle_t)sdl_glc, std::move(glp_out), ctxFlags, gl_version_cstr); // current!
202}
203
204
205bool Window::surfaceSwap() noexcept {
206 if( isValid() ) {
207 SDL_GL_SwapWindow(reinterpret_cast<SDL_Window*>(windowHandle())); // NOLINT
208 return true;
209 }
210 return false;
211}
212
213//
214//
215//
216
217thread_local render::gl::GLContext* render::gl::GLContext::m_current = nullptr;
218
220 SDL_GLContext sdl_glc = reinterpret_cast<SDL_GLContext>(context()); // NOLINT
221 if( sdl_glc ) {
222 if( isCurrent() ) {
224 }
225 SDL_GL_DeleteContext(sdl_glc);
226 }
228}
229
230bool gamp::render::gl::GLContext::makeCurrentImpl(const gamp::wt::SurfaceRef& surface, gamp::handle_t context) noexcept {
231 if( !surface || !surface->isValid() || !context) {
232 printf("SDL: Error GLContext::makeCurrent: Invalid surface/context: surface %s, context %p\n",
233 surface ? surface->toString().c_str() : "nil", (void*)context); // NOLINT
234 return false;
235 }
236 SDL_Window* sdl_win = reinterpret_cast<SDL_Window*>(surface->surfaceHandle()); // NOLINT
237 SDL_GLContext sdl_glc = reinterpret_cast<SDL_GLContext>(context); // NOLINT
238 if (0 != SDL_GL_MakeCurrent(sdl_win, sdl_glc)) {
239 return false;
240 }
241 return true;
242}
243
244void gamp::render::gl::GLContext::releaseContextImpl(const gamp::wt::SurfaceRef& surface) noexcept {
245 if( !surface || !surface->isValid() ) {
246 printf("SDL: Error GLContext::release: Invalid surface: surface %s\n",
247 surface ? surface->toString().c_str() : "nil");
248 return;
249 }
250 SDL_Window* sdl_win = reinterpret_cast<SDL_Window*>(surface->surfaceHandle()); // NOLINT
251 SDL_GL_MakeCurrent(sdl_win, nullptr);
252}
constexpr gamp::handle_t context() const noexcept
Specifies the render profile.
constexpr const jau::util::VersionNumber & version() const noexcept
constexpr int stencilBits() const noexcept
Returns the number of stencil buffer bits.
constexpr int accumGreenBits() const noexcept
Returns the number of bits for the accumulation buffer's green component.
constexpr int accumBlueBits() const noexcept
Returns the number of bits for the accumulation buffer's blue component.
constexpr int depthBits() const noexcept
Returns the number of depth buffer bits.
constexpr void setDoubleBuffered(bool enable) noexcept
Enables or disables double buffering.
constexpr int accumAlphaBits() const noexcept
Returns the number of bits for the accumulation buffer's alpha component.
constexpr int accumRedBits() const noexcept
Returns the number of bits for the accumulation buffer's red component.
constexpr void setStereo(bool enable) noexcept
Enables or disables stereo viewing.
OpenGL Rendering Context.
bool isCurrent() const noexcept
void dispose() noexcept override
void disposedNotify() override
void releaseContext() noexcept override
Release this context (used for OpenGL, but a NOP on Vulkan)
static RenderContextPtr create(gamp::handle_t context, GLProfile profile, RenderContextFlags contextFlags, const char *gl_version_cstr) noexcept
Create a new instance of a non-current context.
Specifies the OpenGL profile.
Definition GLContext.hpp:41
std::string toString() const override
constexpr bool isGLES() const noexcept
Indicates whether this profile is capable of GLES.
static const GLProfile & downcast(const RenderProfile &rp)
Downcast dereferenced given const RenderProfile& to const GLProfile&, throws exception if signature d...
static const jau::type_info & GLSignature() noexcept
A class for storing and comparing OpenGL version numbers.
static GLVersionNumber create(const std::string &versionString) noexcept
constexpr int redBits() const noexcept
Returns the number of bits for the color buffer's red component.
constexpr int greenBits() const noexcept
Returns the number of bits for the color buffer's green component.
constexpr int blueBits() const noexcept
Returns the number of bits for the color buffer's blue component.
constexpr int alphaBits() const noexcept
Returns the number of bits for the color buffer's alpha component.
bool createContext(const gamp::render::RenderProfile &profile, const gamp::render::RenderContextFlags &contextFlags)
Definition Surface.hpp:115
constexpr handle_t windowHandle() const noexcept
Returns the handle to the surface for this NativeSurface.
Definition Window.hpp:264
bool isValid() const noexcept override
Definition Window.hpp:265
bool surfaceSwap() noexcept override
Provide a mechanism to utilize custom (pre-) swap surface code.
Simple version number class containing a version number either being defined explicit or derived from...
constexpr int major() const noexcept
static bool GetGLAttribute(SDL_GLattr attr, int &value) noexcept
constexpr bool value(const Bool rhs) noexcept
constexpr bool is_set(const E mask, const E bits) noexcept
GLProfileMask
OpenGL profile-mask bits.
std::unique_ptr< GLCapabilities > GLCapabilitiesPtr
@ compat
Desktop compatibility profile.
@ core
Desktop core profile.
std::unique_ptr< RenderContext > RenderContextPtr
RenderContextFlags
OpenGL context flags.
@ verbose
Verbose operations (debugging).
std::unique_ptr< Capabilities > CapabilitiesPtr
std::shared_ptr< Surface > SurfaceRef
uintptr_t handle_t
A native handle type, big enough to store a pointer.
Definition GampTypes.hpp:47
std::string to_string(const math_error_t v) noexcept
Returns std::string representation of math_error_t.
Gamp: Graphics, Audio, Multimedia and Processing Framework (Native C++, WebAssembly,...
Definition Gamp.hpp:29
Author: Sven Gothel sgothel@jausoft.com Copyright Gothel Software e.K.
Definition enum_util.hpp:65
int printf(const char *format,...)
Operating Systems predefined macros.