Gamp v0.0.7-36-g24b1eb6
Gamp: Graphics, Audio, Multimedia and Processing
Loading...
Searching...
No Matches
Surface.hpp
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#ifndef GAMP_WTSURFACE_HPP_
12#define GAMP_WTSURFACE_HPP_
13
14#include <jau/fraction_type.hpp>
15#include <jau/int_types.hpp>
16#include <jau/locks.hpp>
17
18#include <gamp/GampTypes.hpp>
21
22namespace gamp::wt {
23 /** \addtogroup Gamp_WT
24 *
25 * @{
26 */
27 class Window;
28 class Surface;
29 typedef std::shared_ptr<Surface> SurfaceRef;
30
31 using namespace jau::fractions_i64_literals;
32 using namespace jau::math;
33
34 //
35 // TODO: GraphicsConfiguration: Screen + Device?
36 //
37
38 class Surface : public std::enable_shared_from_this<Surface> {
39 public:
40 static constexpr const jau::fraction_i64 TIMEOUT = 5_s;
41
46 static constexpr bool is_locked(const lock_status_t ls) noexcept {
47 return static_cast<uint16_t>(lock_status_t::not_ready) < static_cast<uint16_t>(ls);
48 }
49
50 private:
51 int m_swapInterval = -1;
52 handle_t m_surface_handle;
53 /// Surface client-area size in pixel units
54 Vec2i m_surface_size;
55 CapabilitiesPtr m_capsptr;
56 gamp::render::RenderContextPtr m_renderContext;
57
58 jau::RecursiveLock m_surface_lock;
59
60 void disposeImpl(handle_t) noexcept {}
61 bool setSwapIntervalImpl(int v) noexcept;
62 static CapabilitiesPtr retrieveCaps(const wt::SurfaceRef& surface) noexcept;
63
64 protected:
65 struct Private{ explicit Private() = default; };
66
67 template <typename ChildT>
68 std::shared_ptr<ChildT> shared_from_base() {
69 return std::static_pointer_cast<ChildT>(shared_from_this());
70 }
71
72 void setSurfaceSize(const Vec2i& sz) noexcept { m_surface_size = sz; }
73
75 virtual void nativeSurfaceUnlock() noexcept { }
76
77 public:
78 /** Private ctor for single Surface::create() method w/o public ctor. */
79 Surface(Private, handle_t surface_handle, const Vec2i& surface_size)
80 : m_surface_handle(surface_handle), m_surface_size(surface_size) { }
81
82 static SurfaceRef create(handle_t surface_handle, const Vec2i& surface_size) {
83 return std::make_shared<Surface>(Private(), surface_handle, surface_size);
84 }
85
86 Surface(const Surface&) = delete;
87 void operator=(const Surface&) = delete;
88
89 /**
90 * Releases this instance.
91 */
92 virtual ~Surface() noexcept {
93 if( m_surface_lock.isOwner() ) {
95 }
96 }
97
98 virtual void disposedNotify(const jau::fraction_timespec&) noexcept {
99 m_surface_handle = 0;
100 m_renderContext = nullptr;
101 }
102 virtual void dispose(const jau::fraction_timespec&) noexcept {
103 if( m_renderContext ) {
104 m_renderContext->dispose();
105 m_renderContext = nullptr;
106 }
107 if( m_surface_handle ) {
108 disposeImpl(m_surface_handle);
109 m_surface_handle = 0;
110 }
111 }
112
113 const SurfaceRef shared() { return shared_from_this(); }
114
116 gamp::render::RenderContextPtr rc = createContext(shared(), profile, contextFlags, nullptr);
117 if( rc && rc->isValid() ) {
118 m_renderContext = std::move(rc);
119 m_capsptr = retrieveCaps(shared());
120 return true;
121 }
122 return false;
123 }
125 const gamp::render::RenderProfile& profile,
126 const gamp::render::RenderContextFlags& contextFlags,
127 gamp::render::RenderContext* shareWith) noexcept;
128
130 return m_renderContext ? m_renderContext.get() : nullptr;
131 }
133 return m_renderContext ? m_renderContext.get() : nullptr;
134 }
135
136 const Capabilities* capabilities() const noexcept { return m_capsptr ? m_capsptr.get() : nullptr; }
137
138 /**
139 * Returns desired or determined swap interval. Defaults to -1, i.e. adaptive swap interval.
140 *
141 * Use setSwapInterval() to set the swap interval.
142 *
143 * * [1..n] denotes the swap interval
144 * * 0 indicates no swap interval, e.g. by failed setSwapInterval()
145 * * -1 indicates adaptive swap interval
146 */
147 constexpr int swapInterval() const noexcept { return m_swapInterval; }
148
149 /** Returns true if swap interval could be set with the native toolkit post createContext(). See swapInterval(). */
150 bool setSwapInterval(int v) noexcept {
151 if( m_renderContext && m_renderContext->isValid() ) {
152 return setSwapIntervalImpl(v);
153 }
154 m_swapInterval=v;
155 return false;
156 }
157
158 /**
159 * Returns <code>true</code> if this surface is rendered in
160 * OpenGL's coordinate system, <i>origin at bottom left</i>.
161 * Otherwise returns <code>false</code>, i.e. <i>origin at top left</i>.
162 *
163 * Default impl. is <code>true</code>, i.e. OpenGL bottom-left coordinate system.
164 *
165 * Currently only MS-Windows bitmap offscreen drawable uses a non OpenGL orientation and hence returns <code>false</code>.<br/>
166 * This removes the need of a vertical flip when used in AWT or Windows applications.
167 */
168 constexpr bool isBLOriented() const noexcept { return true; }
169
170 /** Returns the surface size of the client area excluding insets (window decorations) in pixel units. */
171 constexpr const Vec2i& surfaceSize() const noexcept { return m_surface_size; }
172
173 /**
174 * Returns the handle to the surface for this NativeSurface. <P>
175 *
176 * The surface handle should be set/update by {@link #lockSurface()},
177 * where {@link #unlockSurface()} is not allowed to modify it.
178 * After {@link #unlockSurface()} it is no more guaranteed
179 * that the surface handle is still valid.
180 *
181 * The surface handle shall reflect the platform one
182 * for all drawable surface operations, e.g. opengl, swap-buffer. <P>
183 *
184 * On X11 this returns an entity of type Window,
185 * since there is no differentiation of surface and window there. <BR>
186 * On Microsoft Windows this returns an entity of type HDC.
187 */
188 constexpr handle_t surfaceHandle() const noexcept { return m_surface_handle; }
189
190 virtual bool isValid() const noexcept { return 0 != m_surface_handle; }
191
192 /**
193 * Provide a mechanism to utilize custom (pre-) swap surface
194 * code. This method is called before the render toolkit (e.g. JOGL)
195 * swaps the buffer/surface if double buffering is enabled.
196 *
197 * The implementation may itself apply the swapping,
198 * in which case true shall be returned.
199 *
200 * @return true if this method completed swapping the surface,
201 * otherwise false, in which case eg the GLDrawable
202 * implementation has to swap the code.
203 */
204 virtual bool surfaceSwap() noexcept { return true; }
205
206 /**
207 * Lock the surface of this native window.
208 *
209 * The surface handle shall be valid after a successfull call,
210 * i.e. return a value other than lock_status_t::unlocked and lock_status_t::not_ready.
211 *
212 * The caller may need to take care of the result lock_status_t::locked_changed,
213 * where the surface handle is valid but has changed.
214 *
215 * This call is blocking until the surface has been locked
216 * or a timeout is reached. The latter will throw a runtime exception.
217 *
218 * This call allows recursion from the same thread.
219 *
220 * The implementation may want to aquire the
221 * application level {@link com.jogamp.common.util.locks.RecursiveLock}
222 * first before proceeding with a native surface lock.
223 *
224 * The implementation shall also invoke {@link AbstractGraphicsDevice#lock()}
225 * for the initial lock (recursive count zero).
226 *
227 * @return Surface::lock_status_t
228 *
229 * @throws RuntimeException after timeout when waiting for the surface lock
230 * @throws NativeWindowException if native locking failed, maybe platform related
231 */
233 if( !m_surface_lock.tryLock(TIMEOUT) ) {
234 throw jau::RuntimeException("Waited "+TIMEOUT.to_string()+"s for: "+toString()+" - "+jau::threadName(std::this_thread::get_id()), E_FILE_LINE);
235 }
236 if (1 == m_surface_lock.holdCount()) {
237 const lock_status_t res = nativeSurfaceLock();
238 if( !is_locked(res) ) {
239 m_surface_lock.unlock();
240 }
241 return res;
242 } else {
244 }
245 }
246
247 /**
248 * Unlock the surface of this native window
249 *
250 * Shall not modify the surface handle, see lockSurface().
251 *
252 * The implementation shall also invoke GraphicsDevice::unlock()
253 * for the final unlock (recursive count zero).
254 *
255 * The implementation shall be fail safe, i.e. tolerant in case the native resources
256 * are already released / unlocked. In this case the implementation shall simply ignore the call.
257 *
258 * @see lockSurface()
259 */
261 m_surface_lock.validateLocked();
262 if (1 == m_surface_lock.holdCount()) {
264 }
265 m_surface_lock.unlock();
266 }
267
268 std::string toString() const noexcept {
269 std::string res = "Surface[";
270 res.append("handle ").append(jau::to_hexstring(m_surface_handle))
271 .append(", ").append(m_renderContext ? m_renderContext->toString() : "nil-ctx")
272 .append(", size ").append(m_surface_size.toString())
273 .append(", ").append(m_capsptr?m_capsptr->toString():"nocaps")
274 .append("]");
275 return res;
276 }
277 };
278 /**@}*/
279} // namespace gamp::wt
280
281#endif /* GAMP_WTSURFACE_HPP_ */
#define E_FILE_LINE
Specifies the render profile.
Specifies a set of capabilities that a window's rendering context must support, such as color depth p...
void operator=(const Surface &)=delete
gamp::render::RenderContext * renderContext() noexcept
Definition Surface.hpp:132
static constexpr bool is_locked(const lock_status_t ls) noexcept
Definition Surface.hpp:46
constexpr handle_t surfaceHandle() const noexcept
Returns the handle to the surface for this NativeSurface.
Definition Surface.hpp:188
lock_status_t lockSurface()
Lock the surface of this native window.
Definition Surface.hpp:232
const SurfaceRef shared()
Definition Surface.hpp:113
const gamp::render::RenderContext * renderContext() const noexcept
Definition Surface.hpp:129
Surface(const Surface &)=delete
void setSurfaceSize(const Vec2i &sz) noexcept
Definition Surface.hpp:72
Surface(Private, handle_t surface_handle, const Vec2i &surface_size)
Private ctor for single Surface::create() method w/o public ctor.
Definition Surface.hpp:79
void unlockSurface()
Unlock the surface of this native window.
Definition Surface.hpp:260
static SurfaceRef create(handle_t surface_handle, const Vec2i &surface_size)
Definition Surface.hpp:82
std::string toString() const noexcept
Definition Surface.hpp:268
virtual ~Surface() noexcept
Releases this instance.
Definition Surface.hpp:92
virtual bool isValid() const noexcept
Definition Surface.hpp:190
virtual void nativeSurfaceUnlock() noexcept
Definition Surface.hpp:75
constexpr int swapInterval() const noexcept
Returns desired or determined swap interval.
Definition Surface.hpp:147
std::shared_ptr< ChildT > shared_from_base()
Definition Surface.hpp:68
constexpr bool isBLOriented() const noexcept
Returns true if this surface is rendered in OpenGL's coordinate system, origin at bottom left.
Definition Surface.hpp:168
virtual bool surfaceSwap() noexcept
Provide a mechanism to utilize custom (pre-) swap surface code.
Definition Surface.hpp:204
const Capabilities * capabilities() const noexcept
Definition Surface.hpp:136
virtual void dispose(const jau::fraction_timespec &) noexcept
Definition Surface.hpp:102
static constexpr const jau::fraction_i64 TIMEOUT
Definition Surface.hpp:40
virtual void disposedNotify(const jau::fraction_timespec &) noexcept
Definition Surface.hpp:98
constexpr const Vec2i & surfaceSize() const noexcept
Returns the surface size of the client area excluding insets (window decorations) in pixel units.
Definition Surface.hpp:171
bool setSwapInterval(int v) noexcept
Returns true if swap interval could be set with the native toolkit post createContext().
Definition Surface.hpp:150
virtual lock_status_t nativeSurfaceLock() noexcept
Definition Surface.hpp:74
bool createContext(const gamp::render::RenderProfile &profile, const gamp::render::RenderContextFlags &contextFlags)
Definition Surface.hpp:115
fraction< int64_t > fraction_i64
fraction using int64_t as integral type
std::unique_ptr< RenderContext > RenderContextPtr
RenderContextFlags
OpenGL context flags.
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
Vector2I< int > Vec2i
Definition vec2i.hpp:328
std::string to_hexstring(value_type const &v, const bool skipLeading0x=false) noexcept
Produce a lower-case hexadecimal string representation with leading 0x in MSB of the given pointer.
std::string threadName(const std::thread::id id) noexcept
Timespec structure using int64_t for its components in analogy to struct timespec_t on 64-bit platfor...