Gamp v0.0.7-36-g24b1eb6
Gamp: Graphics, Audio, Multimedia and Processing
Loading...
Searching...
No Matches
FBObject.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
12#ifndef GAMP_RENDER_GL_FBOJECT_HPP_
13#define GAMP_RENDER_GL_FBOJECT_HPP_
14
15#include <GL/gl.h>
20
21#include <jau/basic_types.hpp>
22#include <jau/cpp_lang_util.hpp>
23#include <jau/enum_util.hpp>
24#include <jau/string_util.hpp>
25
26namespace gamp::render::gl {
27 using namespace jau::enums;
28
29 /** \addtogroup Gamp_GL
30 *
31 * @{
32 */
33
34 /**
35 * Core utility class simplifying usage of framebuffer objects (FBO)
36 * with all {@link GLProfile}s.
37 * <p>
38 * Supports on-the-fly reconfiguration of dimension and multisample buffers via {@link #reset(GL, int, int, int, boolean)}
39 * while preserving the {@link Attachment} references.
40 * </p>
41 * <p>
42 * Integrates default read/write framebuffers via {@link GLContext#getDefaultReadFramebuffer()} and {@link GLContext#getDefaultReadFramebuffer()},
43 * which is being hooked at {@link GL#glBindFramebuffer(int, int)} when the default (<code>zero</code>) framebuffer is selected.
44 * </p>
45 *
46 * <p>FIXME: Implement support for {@link Type#DEPTH_TEXTURE}, {@link Type#STENCIL_TEXTURE} .</p>
47 */
48 class FBObject {
49 private:
50 constexpr static bool DEBUG_FBO = true;
51 constexpr static int USER_MAX_TEXTURE_SIZE = 0;
52 constexpr static bool FBOResizeQuirk = false;
53
54 enum class DetachAction { NONE, DISPOSE, RECREATE };
55
57 class ColorAttachment;
58
59 /**
60 * Generic color buffer FBO attachment, either of type {@link ColorAttachment} or {@link TextureAttachment}.
61 * <p>Always an instance of {@link Attachment}.</p>
62 */
63 class Colorbuffer {
64 public:
65 virtual ~Colorbuffer() = default;
66 /**
67 * Initializes the color buffer and set it's parameter, if uninitialized, i.e. name is <code>zero</code>.
68 * @return <code>true</code> if newly initialized, otherwise <code>false</code>.
69 * @throws GLException if buffer generation or setup fails. The just created buffer name will be deleted in this case.
70 */
71 virtual bool initialize(const GL& gl) = 0;
72
73 /**
74 * Releases the color buffer if initialized, i.e. name is not <code>zero</code>.
75 * @throws GLException if buffer release fails.
76 */
77 virtual void free(const GL& gl) = 0;
78
79 /**
80 * Writes the internal format to the given GLCapabilities object.
81 * @param caps the destination for format bits
82 * @param rgba8Avail whether rgba8 is available
83 */
84 virtual void formatToGLCapabilities(GLCapabilities& caps, bool rgba8Avail) = 0;
85
86 /**
87 * Returns <code>true</code> if instance is of type {@link TextureAttachment}
88 * and <code>false</code> if instance is of type {@link ColorAttachment}.
89 */
90 virtual bool isTextureAttachment() = 0;
91
92 /**
93 * Casts this object to a {@link TextureAttachment} reference, see {@link #isTextureAttachment()}.
94 * @throws GLException if this object is not of type {@link TextureAttachment}
95 * @see #isTextureAttachment()
96 */
98
99 /**
100 * Casts this object to a {@link ColorAttachment} reference, see {@link #isTextureAttachment()}.
101 * @throws GLException if this object is not of type {@link ColorAttachment}
102 * @see #isTextureAttachment()
103 */
104 virtual ColorAttachment& getColorAttachment() = 0;
105
106 /** internal format of colorbuffer */
107 virtual int getFormat() = 0;
108
109 /** width of colorbuffer */
110 virtual int getWidth() = 0;
111
112 /** height of colorbuffer */
113 virtual int getHeight() = 0;
114
115 /** colorbuffer name [1..max] */
116 virtual int getName() = 0;
117 };
118
119 /** Common super class of all FBO attachments */
120 class Attachment {
121 public:
122 enum class Type {
123 NONE, DEPTH, STENCIL, DEPTH_STENCIL, COLOR, COLOR_TEXTURE, DEPTH_TEXTURE, STENCIL_TEXTURE
124 };
125
126 /**
127 * Returns {@link #COLOR}, {@link #DEPTH}, {@link #STENCIL} or {@link #DEPTH_STENCIL}
128 * @throws IllegalArgumentException if <code>format</code> cannot be handled.
129 */
130 static Type determine(GLenum format) {
131 switch(format) {
132 case GL_RGBA4:
133 case GL_RGB5_A1:
134 case GL_RGB565:
135 case GL_RGB8:
136 case GL_RGBA8:
137 return Type.COLOR;
138 case GL_DEPTH_COMPONENT16:
139 case GL_DEPTH_COMPONENT24:
140 case GL_DEPTH_COMPONENT32:
141 return Type.DEPTH;
142 case GL_STENCIL_INDEX1:
143 case GL_STENCIL_INDEX4:
144 case GL_STENCIL_INDEX8:
145 return Type.STENCIL;
146 case GL_DEPTH24_STENCIL8:
147 return Type.DEPTH_STENCIL;
148 default:
149 throw jau::IllegalArgumentError("format invalid: "+jau::to_hexstring(format), E_FILE_LINE);
150 }
151 }
152
153 /**
154 * Interface abstraction to allow custom definitions of {@link Attachment}'s storage.
155 * <p>
156 * Please see {@link #setStorage(GL, Attachment)} for details.
157 * </p>
158 */
160 public:
161 /**
162 * Set or create the {@link Attachment}'s storage after generating its name and binding it to the target.
163 * Typical examples for standard definitions as implemented in {@link Attachment} specializations are:
164 * <pre>
165 * // Renderbuffer (Color, Debt, Stencil, ..) storage definition w/o multisamples
166 * gl.glRenderbufferStorage(GL.GL_RENDERBUFFER, a.format, a.getWidth(), a.getHeight());
167 * // Renderbuffer (Color, Debt, Stencil, ..) storage definition with multisamples
168 * gl.glRenderbufferStorageMultisample(GL.GL_RENDERBUFFER, samples, a.format, a.getWidth(), a.getHeight());
169 * // TextureAttachment
170 * gl.glTexImage2D(GL.GL_TEXTURE_2D, 0, format, getWidth(), getHeight(), 0, dataFormat, dataType, null);
171 * </pre>
172 * The storage is setup within {@link Attachment#initialize(GL)} and hence the {@link Attachment}
173 * instance's {@link Attachment#setStorageDefinition(StorageDefinition)}.
174 *
175 * @param gl current {@link GL} instance
176 * @param a instance of the {@link Attachment} this {@link StorageDefinition} has been assigned to via {@link Attachment#setStorageDefinition(StorageDefinition)}.
177 */
178 virtual void setStorage(const GL& gl, const Attachment& a) = 0;
179 };
180
181 /** immutable type [{@link #COLOR}, {@link #DEPTH}, {@link #STENCIL}, {@link #COLOR_TEXTURE}, {@link #DEPTH_TEXTURE}, {@link #STENCIL_TEXTURE} ] */
182 Type type;
183
184 /** immutable the internal format */
185 GLenum format;
186
187 int width, height;
188
189 private:
190 int name;
191
192 /** Every implementation needs to have set their default instance via {@link #setStorageDefinition(StorageDefinition). */
193 StorageDefinition storageDefinition;
194
195 protected Attachment(final Type type, final int iFormat, final int width, final int height, final int name) {
196 this.type = type;
197 this.format = iFormat;
198 this.width = width;
199 this.height = height;
200 this.name = name;
201 }
202
203 /**
204 * Override implementation default {@link StorageDefinition}
205 * @see {@link StorageDefinition#setStorage(GL, Attachment)}
206 */
207 public void setStorageDefinition(final StorageDefinition sd) { this.storageDefinition = sd; }
208
209 /**
210 * Accessor to call {@link StorageDefinition#setStorage(GL, Attachment)} within {@link #initialize(GL)} for implementations of {@link Attachment}.
211 */
212 protected final void setStorage(final GL gl) { storageDefinition.setStorage(gl, this); }
213
214 /**
215 * Writes the internal format to the given GLCapabilities object.
216 * @param caps the destination for format bits
217 * @param rgba8Avail whether rgba8 is available
218 */
219 public final void formatToGLCapabilities(final GLCapabilities caps, final boolean rgba8Avail) {
220 final int _format;
221 switch(format) {
222 case GL.GL_RGBA:
223 case 4:
224 _format = rgba8Avail ? GL.GL_RGBA8 : GL.GL_RGBA4;
225 break;
226 case GL.GL_RGB:
227 case 3:
228 _format = rgba8Avail ? GL.GL_RGB8 : GL.GL_RGB565;
229 break;
230 default:
231 _format = format;
232 }
233 switch(_format) {
234 case GL.GL_RGBA4:
235 caps.setRedBits(4);
236 caps.setGreenBits(4);
237 caps.setBlueBits(4);
238 caps.setAlphaBits(4);
239 break;
240 case GL.GL_RGB5_A1:
241 caps.setRedBits(5);
242 caps.setGreenBits(5);
243 caps.setBlueBits(5);
244 caps.setAlphaBits(1);
245 break;
246 case GL.GL_RGB565:
247 caps.setRedBits(5);
248 caps.setGreenBits(6);
249 caps.setBlueBits(5);
250 caps.setAlphaBits(0);
251 break;
252 case GL.GL_RGB8:
253 caps.setRedBits(8);
254 caps.setGreenBits(8);
255 caps.setBlueBits(8);
256 caps.setAlphaBits(0);
257 break;
258 case GL.GL_RGBA8:
259 caps.setRedBits(8);
260 caps.setGreenBits(8);
261 caps.setBlueBits(8);
262 caps.setAlphaBits(8);
263 break;
264 case GL.GL_DEPTH_COMPONENT16:
265 caps.setDepthBits(16);
266 break;
267 case GL.GL_DEPTH_COMPONENT24:
268 caps.setDepthBits(24);
269 break;
270 case GL.GL_DEPTH_COMPONENT32:
271 caps.setDepthBits(32);
272 break;
273 case GL.GL_STENCIL_INDEX1:
274 caps.setStencilBits(1);
275 break;
276 case GL.GL_STENCIL_INDEX4:
277 caps.setStencilBits(4);
278 break;
279 case GL.GL_STENCIL_INDEX8:
280 caps.setStencilBits(8);
281 break;
282 case GL.GL_DEPTH24_STENCIL8:
283 caps.setDepthBits(24);
284 caps.setStencilBits(8);
285 break;
286 default:
287 throw IllegalArgumentException("format invalid: "+toHexString(format));
288 }
289 }
290
291 /** immutable internal format of attachment */
292 public final int getFormat() { return format; }
293
294 /** width of attachment */
295 public final int getWidth() { return width; }
296 /** height of attachment */
297 public final int getHeight() { return height; }
298 /* pp */ final void setSize(final int w, final int h) { width = w; height = h; }
299
300 /** buffer name [1..max], maybe a texture or renderbuffer name, depending on type. */
301 public final int getName() { return name; }
302 /* pp */ final void setName(final int n) { name = n; }
303
304 /**
305 * Initializes the attachment and set it's parameter, if uninitialized, i.e. name is <code>zero</code>.
306 * <pre>
307 final boolean init = 0 == name;
308 if( init ) {
309 do init ..
310 }
311 return init;
312 * </pre>
313 * @return <code>true</code> if newly initialized, otherwise <code>false</code>.
314 * @throws GLException if buffer generation or setup fails. The just created buffer name will be deleted in this case.
315 */
316 public abstract boolean initialize(final GL gl) throws GLException;
317
318 /**
319 * Releases the attachment if initialized, i.e. name is not <code>zero</code>.
320 * <pre>
321 if(0 != name) {
322 do free ..
323 name = 0;
324 }
325 * </pre>
326 * @throws GLException if buffer release fails.
327 */
328 public abstract void free(final GL gl) throws GLException;
329
330 /**
331 * <p>
332 * Comparison by {@link #type}, {@link #format}, {@link #width}, {@link #height} and {@link #name}.
333 * </p>
334 * {@inheritDoc}
335 */
336 @Override
337 public boolean equals(final Object o) {
338 if( this == o ) return true;
339 if( ! ( o instanceof Attachment ) ) return false;
340 final Attachment a = (Attachment)o;
341 return type == a.type &&
342 format == a.format &&
343 width == a.width &&
344 height== a.height &&
345 name == a.name ;
346 }
347
348 /**
349 * <p>
350 * Hashed by {@link #type}, {@link #format}, {@link #width}, {@link #height} and {@link #name}.
351 * </p>
352 * {@inheritDoc}
353 */
354 @Override
355 public int hashCode() {
356 // 31 * x == (x << 5) - x
357 int hash = 31 + type.ordinal();
358 hash = ((hash << 5) - hash) + format;
359 hash = ((hash << 5) - hash) + width;
360 hash = ((hash << 5) - hash) + height;
361 hash = ((hash << 5) - hash) + name;
362 return hash;
363 }
364
365 int objectHashCode() { return super.hashCode(); }
366
367 @Override
368 public String toString() {
369 return getClass().getSimpleName()+"[type "+type+", format "+toHexString(format)+", "+width+"x"+height+
370 "; name "+toHexString(name)+", obj "+toHexString(objectHashCode())+"]";
371 }
372
373 public static Type getType(final int attachmentPoint, final int maxColorAttachments) {
374 if( GL.GL_COLOR_ATTACHMENT0 <= attachmentPoint && attachmentPoint < GL.GL_COLOR_ATTACHMENT0+maxColorAttachments ) {
375 return Type.COLOR;
376 }
377 switch(attachmentPoint) {
378 case GL.GL_DEPTH_ATTACHMENT:
379 return Type.DEPTH;
380 case GL.GL_STENCIL_ATTACHMENT:
381 return Type.STENCIL;
382 default:
383 throw IllegalArgumentException("Invalid attachment point "+toHexString(attachmentPoint));
384 }
385 }
386 }
387
388 /** Other renderbuffer attachment which maybe a colorbuffer, depth or stencil. */
389 public static class RenderAttachment extends Attachment {
390 private int samples;
391
392 /**
393 * @param type allowed types are {@link Type#DEPTH_STENCIL} {@link Type#DEPTH}, {@link Type#STENCIL} or {@link Type#COLOR}
394 * @param iFormat
395 * @param samples
396 * @param width
397 * @param height
398 * @param name
399 */
400 public RenderAttachment(final Type type, final int iFormat, final int samples, final int width, final int height, final int name) {
401 super(validateType(type), iFormat, width, height, name);
402 this.setStorageDefinition(defStorageDefinition);
403 this.samples = samples;
404 }
405
406 /** number of samples, or zero for no multisampling */
407 public final int getSamples() { return samples; }
408 /* pp */ final void setSamples(final int s) { samples = s; }
409
410 private static Type validateType(final Type type) {
411 switch(type) {
412 case DEPTH_STENCIL:
413 case DEPTH:
414 case STENCIL:
415 case COLOR:
416 return type;
417 default:
418 throw IllegalArgumentException("Invalid type: "+type);
419 }
420 }
421
422 /**
423 * <p>
424 * Comparison by {@link #type}, {@link #format}, {@link #samples}, {@link #width}, {@link #height} and {@link #name}.
425 * </p>
426 * {@inheritDoc}
427 */
428 @Override
429 public boolean equals(final Object o) {
430 if( this == o ) return true;
431 if( ! ( o instanceof RenderAttachment ) ) return false;
432 return super.equals(o) &&
433 samples == ((RenderAttachment)o).samples;
434 }
435
436 /**
437 * <p>
438 * Hashed by {@link #type}, {@link #format}, {@link #samples}, {@link #width}, {@link #height} and {@link #name}.
439 * </p>
440 * {@inheritDoc}
441 */
442 @Override
443 public int hashCode() {
444 // 31 * x == (x << 5) - x
445 int hash = super.hashCode();
446 hash = ((hash << 5) - hash) + samples;
447 return hash;
448 }
449
450 @Override
451 public boolean initialize(final GL gl) throws GLException {
452 final boolean init = 0 == getName();
453 if( init ) {
454 final boolean checkError = DEBUG_FBO || GLContext.DEBUG_FBO_GL;
455 if( checkError ) {
456 checkPreGLError(gl);
457 }
458 final int[] name = new int[] { -1 };
459 gl.glGenRenderbuffers(1, name, 0);
460 setName(name[0]);
461
462 gl.glBindRenderbuffer(GL.GL_RENDERBUFFER, getName());
463 setStorage(gl);
464 if( checkError ) {
465 final int glerr = gl.glGetError();
466 if(GL.GL_NO_ERROR != glerr) {
467 gl.glDeleteRenderbuffers(1, name, 0);
468 setName(0);
469 throw GLException("GL Error "+toHexString(glerr)+" while creating "+this);
470 }
471 }
472 if(DEBUG_FBO) {
473 System.err.println("Attachment.init.X: "+this);
474 }
475 }
476 return init;
477 }
478 private final StorageDefinition defStorageDefinition = new StorageDefinition() {
479 @Override
480 public void setStorage(final GL gl, final Attachment a) {
481 // a == this.super for this instance
482 if( samples > 0 ) {
483 gl.glRenderbufferStorageMultisample(GL.GL_RENDERBUFFER, samples, format, getWidth(), getHeight());
484 } else {
485 gl.glRenderbufferStorage(GL.GL_RENDERBUFFER, format, getWidth(), getHeight());
486 }
487 } };
488
489 @Override
490 public void free(final GL gl) {
491 final int[] name = new int[] { getName() };
492 if( 0 != name[0] ) {
493 if(DEBUG_FBO) {
494 System.err.println("Attachment.free.0: "+this);
495 }
496 gl.glDeleteRenderbuffers(1, name, 0);
497 setName(0);
498 }
499 }
500
501 @Override
502 public String toString() {
503 return getClass().getSimpleName()+"[type "+type+", format "+toHexString(format)+", samples "+samples+", "+getWidth()+"x"+getHeight()+
504 ", name "+toHexString(getName())+", obj "+toHexString(objectHashCode())+"]";
505 }
506 }
507
508 /** Color render buffer FBO attachment */
509 public static class ColorAttachment extends RenderAttachment implements Colorbuffer {
510 public ColorAttachment(final int iFormat, final int samples, final int width, final int height, final int name) {
511 super(Type.COLOR, iFormat, samples, width, height, name);
512 }
513 @Override
514 public final boolean isTextureAttachment() { return false; }
515 @Override
516 public final TextureAttachment getTextureAttachment() { throw GLException("Not a TextureAttachment, but ColorAttachment"); }
517 @Override
518 public final ColorAttachment getColorAttachment() { return this; }
519 }
520
521 /** Texture FBO attachment */
522 public static class TextureAttachment extends Attachment implements Colorbuffer {
523 /** details of the texture setup */
525
526 /**
527 * @param type allowed types are [ {@link Type#COLOR_TEXTURE}, {@link Type#DEPTH_TEXTURE}, {@link Type#STENCIL_TEXTURE} ]
528 * @param iFormat
529 * @param width
530 * @param height
531 * @param dataFormat
532 * @param dataType
533 * @param magFilter
534 * @param minFilter
535 * @param wrapS
536 * @param wrapT
537 * @param name
538 */
539 public TextureAttachment(final Type type, final int iFormat, final int width, final int height, final int dataFormat, final int dataType,
540 final int magFilter, final int minFilter, final int wrapS, final int wrapT, final int name) {
541 super(validateType(type), iFormat, width, height, name);
542 this.setStorageDefinition(defStorageDefinition);
543 this.dataFormat = dataFormat;
544 this.dataType = dataType;
545 this.magFilter = magFilter;
546 this.minFilter = minFilter;
547 this.wrapS = wrapS;
548 this.wrapT = wrapT;
549 }
550
551 private static Type validateType(final Type type) {
552 switch(type) {
553 case COLOR_TEXTURE:
554 case DEPTH_TEXTURE:
555 case STENCIL_TEXTURE:
556 return type;
557 default:
558 throw IllegalArgumentException("Invalid type: "+type);
559 }
560 }
561
562 /**
563 * Initializes the texture and set it's parameter, if uninitialized, i.e. name is <code>zero</code>.
564 * @throws GLException if texture generation and setup fails. The just created texture name will be deleted in this case.
565 */
566 @Override
567 public boolean initialize(final GL gl) throws GLException {
568 final boolean init = 0 == getName();
569 if( init ) {
570 final boolean checkError = DEBUG_FBO || GLContext.DEBUG_FBO_GL;
571 if( checkError ) {
572 checkPreGLError(gl);
573 }
574 final int[] name = new int[] { -1 };
575 gl.glGenTextures(1, name, 0);
576 if(0 == name[0]) {
577 throw GLException("null texture, "+this);
578 }
579 setName(name[0]);
580
581 gl.glBindTexture(GL.GL_TEXTURE_2D, name[0]);
582 if( 0 < magFilter ) {
583 gl.glTexParameteri(GL.GL_TEXTURE_2D, GL.GL_TEXTURE_MAG_FILTER, magFilter);
584 }
585 if( 0 < minFilter ) {
586 gl.glTexParameteri(GL.GL_TEXTURE_2D, GL.GL_TEXTURE_MIN_FILTER, minFilter);
587 }
588 if( 0 < wrapS ) {
589 gl.glTexParameteri(GL.GL_TEXTURE_2D, GL.GL_TEXTURE_WRAP_S, wrapS);
590 }
591 if( 0 < wrapT ) {
592 gl.glTexParameteri(GL.GL_TEXTURE_2D, GL.GL_TEXTURE_WRAP_T, wrapT);
593 }
594 if( checkError ) {
595 boolean preTexImage2D = true;
596 int glerr = gl.glGetError();
597 if(GL.GL_NO_ERROR == glerr) {
598 preTexImage2D = false;
599 setStorage(gl);
600 glerr = gl.glGetError();
601 }
602 if(GL.GL_NO_ERROR != glerr) {
603 gl.glDeleteTextures(1, name, 0);
604 setName(0);
605 throw GLException("GL Error "+toHexString(glerr)+" while creating (pre TexImage2D "+preTexImage2D+") "+this);
606 }
607 } else {
608 setStorage(gl);
609 }
610 if(DEBUG_FBO) {
611 System.err.println("Attachment.init.X: "+this);
612 }
613 }
614 return init;
615 }
616 private final StorageDefinition defStorageDefinition = new StorageDefinition() {
617 @Override
618 public void setStorage(final GL gl, final Attachment a) {
619 // a == this.super for this instance
620 gl.glTexImage2D(GL.GL_TEXTURE_2D, 0, format, getWidth(), getHeight(), 0, dataFormat, dataType, null);
621 } };
622
623 @Override
624 public void free(final GL gl) {
625 final int[] name = new int[] { getName() };
626 if( 0 != name[0] ) {
627 if(DEBUG_FBO) {
628 System.err.println("Attachment.free.0: "+this);
629 }
630 gl.glDeleteTextures(1, name, 0);
631 setName(0);
632 }
633 }
634
635 @Override
636 public final boolean isTextureAttachment() { return true; }
637 @Override
638 public final TextureAttachment getTextureAttachment() { return this; }
639 @Override
640 public final ColorAttachment getColorAttachment() { throw GLException("Not a ColorAttachment, but TextureAttachment"); }
641
642 @Override
643 public String toString() {
644 return getClass().getSimpleName()+"[type "+type+", target GL_TEXTURE_2D, level 0, format "+toHexString(format)+
645 ", "+getWidth()+"x"+getHeight()+", border 0, dataFormat "+toHexString(dataFormat)+
646 ", dataType "+toHexString(dataType)+
647 "; min/mag "+toHexString(minFilter)+"/"+toHexString(magFilter)+
648 ", wrap S/T "+toHexString(wrapS)+"/"+toHexString(wrapT)+
649 "; name "+toHexString(getName())+", obj "+toHexString(objectHashCode())+"]";
650 }
651 }
652 static String toHexString(final int v) {
653 return "0x"+Integer.toHexString(v);
654 }
655
656 /**
657 * Creates a color {@link TextureAttachment}, i.e. type {@link Type#COLOR_TEXTURE},
658 * selecting the texture data type and format automatically.
659 *
660 * <p>Using default min/mag filter {@link GL#GL_NEAREST} and default wrapS/wrapT {@link GL#GL_CLAMP_TO_EDGE}.</p>
661 *
662 * @param gl the used {@link GLContext}'s {@link GL} object
663 * @param alpha set to <code>true</code> if you request alpha channel, otherwise <code>false</code>;
664 * @param width texture width
665 * @param height texture height
666 * @return the created and uninitialized color {@link TextureAttachment}
667 */
668 public static final TextureAttachment createColorTextureAttachment(final GL gl, final boolean alpha, final int width, final int height) {
669 return createColorTextureAttachment(gl, alpha, width, height, GL.GL_NEAREST, GL.GL_NEAREST, GL.GL_CLAMP_TO_EDGE, GL.GL_CLAMP_TO_EDGE);
670 }
671
672 /**
673 * Creates a color {@link TextureAttachment}, i.e. type {@link Type#COLOR_TEXTURE},
674 * selecting the texture data type and format automatically.
675 * <p>
676 * For GLES3, sampling-sink format <b>must be equal</b> w/ the sampling-source {@link Colorbuffer},
677 * see details below. Implementation aligns w/ {@link #createColorAttachment(boolean)}
678 * and is enforced via {@link #sampleSinkExFormatMismatch(GL)}.
679 * </p>
680 * <p>
681 * ES3 BlitFramebuffer Requirements: OpenGL ES 3.0.2 p194: 4.3.2 Copying Pixels
682 * <pre>
683 * If SAMPLE_BUFFERS for the read framebuffer is greater than zero, no copy
684 * is performed and an INVALID_OPERATION error is generated if the formats of
685 * the read and draw framebuffers are not identical or if the source and destination
686 * rectangles are not defined with the same (X0, Y 0) and (X1, Y 1) bounds.
687 * </pre>
688 * Texture and Renderbuffer format details:
689 * <pre>
690 * ES2 Base iFormat: OpenGL ES 2.0.24 p66: 3.7.1 Texture Image Specification, Table 3.8
691 * ALPHA, LUMINANCE, LUMINANCE_ALPHA, RGB, RGBA
692 *
693 * ES3 Base iFormat: OpenGL ES 3.0.2 p125: 3.8.3 Texture Image Specification, Table 3.11
694 * ALPHA, LUMINANCE, LUMINANCE_ALPHA, RGB, RGBA
695 * DEPTH_COMPONENT, STENCIL_COMPONENT, RED, RG
696 *
697 * ES3 Required Texture and Renderbuffer iFormat: OpenGL ES 3.0.2 p126: 3.8.3 Texture Image Specification
698 * - RGBA32I, RGBA32UI, RGBA16I, RGBA16UI, RGBA8, RGBA8I,
699 * RGBA8UI, SRGB8_ALPHA8, RGB10_A2, RGB10_A2UI, RGBA4, and
700 * RGB5_A1.
701 * - RGB8 and RGB565.
702 * - RG32I, RG32UI, RG16I, RG16UI, RG8, RG8I, and RG8UI.
703 * - R32I, R32UI, R16I, R16UI, R8, R8I, and R8UI.
704 * </pre>
705 * </p>
706 *
707 * @param gl the used {@link GLContext}'s {@link GL} object
708 * @param alpha set to <code>true</code> if you request alpha channel, otherwise <code>false</code>;
709 * @param width texture width
710 * @param height texture height
711 * @param magFilter if > 0 value for {@link GL#GL_TEXTURE_MAG_FILTER}
712 * @param minFilter if > 0 value for {@link GL#GL_TEXTURE_MIN_FILTER}
713 * @param wrapS if > 0 value for {@link GL#GL_TEXTURE_WRAP_S}
714 * @param wrapT if > 0 value for {@link GL#GL_TEXTURE_WRAP_T}
715 * @return the created and uninitialized color {@link TextureAttachment}
716 */
717 public static final TextureAttachment createColorTextureAttachment(final GL gl, final boolean alpha, final int width, final int height,
718 final int magFilter, final int minFilter, final int wrapS, final int wrapT) {
719 final int internalFormat, dataFormat, dataType;
720 if(gl.isGLES3()) {
721 internalFormat = alpha ? GL.GL_RGBA8 : GL.GL_RGB8;
722 dataFormat = alpha ? GL.GL_RGBA : GL.GL_RGB;
723 dataType = GL.GL_UNSIGNED_BYTE;
724 } else if(gl.isGLES()) {
725 internalFormat = alpha ? GL.GL_RGBA : GL.GL_RGB;
726 dataFormat = alpha ? GL.GL_RGBA : GL.GL_RGB;
727 dataType = GL.GL_UNSIGNED_BYTE;
728 } else {
729 internalFormat = alpha ? GL.GL_RGBA8 : GL.GL_RGB8;
730 // textureInternalFormat = alpha ? GL.GL_RGBA : GL.GL_RGB;
731 // textureInternalFormat = alpha ? 4 : 3;
732 dataFormat = alpha ? GL.GL_BGRA : GL.GL_RGB;
733 dataType = alpha ? GL2GL3.GL_UNSIGNED_INT_8_8_8_8_REV : GL.GL_UNSIGNED_BYTE;
734 }
735 return createColorTextureAttachment(internalFormat, width, height, dataFormat, dataType, magFilter, minFilter, wrapS, wrapT);
736 }
737
738 public static final TextureAttachment createColorTextureAttachment(final GL gl, final int internalFormat, final int width, final int height,
739 final int magFilter, final int minFilter, final int wrapS, final int wrapT) {
740 final int dataFormat, dataType;
741 final boolean alpha = hasAlpha(internalFormat);
742 if( gl.isGLES() ) {
743 dataFormat = alpha ? GL.GL_RGBA : GL.GL_RGB;
744 dataType = GL.GL_UNSIGNED_BYTE;
745 } else {
746 dataFormat = alpha ? GL.GL_BGRA : GL.GL_RGB;
747 dataType = alpha ? GL2GL3.GL_UNSIGNED_INT_8_8_8_8_REV : GL.GL_UNSIGNED_BYTE;
748 }
749 return createColorTextureAttachment(internalFormat, width, height, dataFormat, dataType, magFilter, minFilter, wrapS, wrapT);
750 }
751
752 /**
753 * Creates a color {@link TextureAttachment}, i.e. type {@link Type#COLOR_TEXTURE}.
754 *
755 * @param internalFormat internalFormat parameter to {@link GL#glTexImage2D(int, int, int, int, int, int, int, int, long)}
756 * @param width texture width
757 * @param height texture height
758 * @param dataFormat format parameter to {@link GL#glTexImage2D(int, int, int, int, int, int, int, int, long)}
759 * @param dataType type parameter to {@link GL#glTexImage2D(int, int, int, int, int, int, int, int, long)}
760 * @param magFilter if > 0 value for {@link GL#GL_TEXTURE_MAG_FILTER}
761 * @param minFilter if > 0 value for {@link GL#GL_TEXTURE_MIN_FILTER}
762 * @param wrapS if > 0 value for {@link GL#GL_TEXTURE_WRAP_S}
763 * @param wrapT if > 0 value for {@link GL#GL_TEXTURE_WRAP_T}
764 * @return the created and uninitialized color {@link TextureAttachment}
765 */
766 public static final TextureAttachment createColorTextureAttachment(final int internalFormat, final int width, final int height, final int dataFormat, final int dataType,
767 final int magFilter, final int minFilter, final int wrapS, final int wrapT) {
768 return new TextureAttachment(Type.COLOR_TEXTURE, internalFormat, width, height, dataFormat, dataType,
769 magFilter, minFilter, wrapS, wrapT, 0 /* name */);
770 }
771
772 private static boolean hasAlpha(final int format) {
773 switch(format) {
774 case GL.GL_RGBA8:
775 case GL.GL_RGBA4:
776 case GL.GL_RGBA:
777 case GL.GL_BGRA:
778 case 4:
779 return true;
780 default:
781 return false;
782 }
783 }
784
785 private boolean initialized;
786 private boolean fullFBOSupport;
787 private boolean rgba8Avail;
788 private boolean depth24Avail;
789 private boolean depth32Avail;
790 private boolean stencil01Avail;
791 private boolean stencil04Avail;
792 private boolean stencil08Avail;
793 private boolean stencil16Avail;
794 private boolean packedDepthStencilAvail;
795 private int maxColorAttachments, maxSamples, maxTextureSize, maxRenderbufferSize;
796
797 private int width, height, samples;
798 private int vStatus;
799 private boolean ignoreStatus;
800 private int fbName;
801 private boolean bound;
802
803 private int colorbufferCount;
804 private int textureAttachmentCount;
805 private Colorbuffer[] colorbufferAttachments; // colorbuffer attachment points
806 private RenderAttachment depth, stencil; // depth and stencil maybe equal in case of packed-depth-stencil
807 private boolean modified; // size, sampleCount, or any attachment modified
808
809 private FBObject samplingSink; // MSAA sink
810 private Colorbuffer samplingColorSink;
811 private boolean samplingSinkDirty;
812
813 //
814 // ColorAttachment helper ..
815 //
816
817 private final void validateColorAttachmentPointRange(final int point) {
818 if(!initialized) {
819 throw GLException("FBO not initialized");
820 }
821 if(maxColorAttachments != colorbufferAttachments.length) {
822 throw InternalError(String.format("maxColorAttachments %d, array.length %d",
823 maxColorAttachments, colorbufferAttachments.length) );
824 }
825 if(0 > point || point >= maxColorAttachments) {
826 throw IllegalArgumentException(String.format("attachment point out of range: %d, should be within [0..%d], %s",
827 point, maxColorAttachments-1, this.toString() ) );
828 }
829 }
830
831 private final void validateAddColorAttachment(final int point, final Colorbuffer ca) {
832 validateColorAttachmentPointRange(point);
833 if( null != colorbufferAttachments[point] ) {
834 throw IllegalStateException(String.format("Cannot attach %s at %d, attachment point already in use by %s, %s",
835 ca.toString(), point, colorbufferAttachments[point].toString(), this.toString() ) );
836 }
837 }
838
839 private final void addColorAttachment(final int point, final Colorbuffer ca, final boolean validate) {
840 final Colorbuffer c = colorbufferAttachments[point];
841 if( validate ) {
842 validateColorAttachmentPointRange(point);
843 if( null == ca ) {
844 throw IllegalArgumentException("Colorbuffer is null");
845 }
846 if( null != c ) {
847 throw IllegalStateException(String.format("Cannot attach %s at %d, attachment point already in use by %s, %s",
848 ca.toString(), point, c.toString(), this.toString() ) );
849 }
850 }
851 colorbufferAttachments[point] = ca;
852 colorbufferCount++;
853 if( ca.isTextureAttachment() ) {
854 textureAttachmentCount++;
855 }
856 modified = true;
857 }
858
859 private final void removeColorAttachment(final int point, final Colorbuffer ca) {
860 validateColorAttachmentPointRange(point);
861 if( null == ca ) {
862 throw IllegalArgumentException("Colorbuffer is null");
863 }
864 final Colorbuffer c = colorbufferAttachments[point];
865 if( c != ca ) {
866 throw IllegalStateException(String.format("Cannot detach %s at %d, slot is holding other: %s, %s",
867 ca.toString(), point, c.toString(), this.toString() ) );
868 }
869 colorbufferAttachments[point] = null;
870 colorbufferCount--;
871 if( ca.isTextureAttachment() ) {
872 textureAttachmentCount--;
873 }
874 modified = true;
875 }
876
877 /**
878 * Return the {@link Colorbuffer} attachment at <code>attachmentPoint</code> if it is attached to this FBO, otherwise null.
879 *
880 * @see #attachColorbuffer(GL, boolean)
881 * @see #attachColorbuffer(GL, boolean)
882 * @see #attachTexture2D(GL, int, boolean, int, int, int, int)
883 * @see #attachTexture2D(GL, int, int, int, int, int, int, int, int)
884 */
885 public final Colorbuffer getColorbuffer(final int attachmentPoint) {
886 validateColorAttachmentPointRange(attachmentPoint);
887 return colorbufferAttachments[attachmentPoint];
888 }
889
890 /**
891 * Finds the passed {@link Colorbuffer} within the valid range of attachment points
892 * using <i>reference</i> comparison only.
893 * <p>
894 * Note: Slow. Implementation uses a logN array search to save resources, i.e. not using a HashMap.
895 * </p>
896 * @param ca the {@link Colorbuffer} to look for.
897 * @return -1 if the {@link Colorbuffer} could not be found, otherwise [0..{@link #getMaxColorAttachments()}-1]
898 */
899 public final int getColorbufferAttachmentPoint(final Colorbuffer ca) {
900 for(int i=0; i<colorbufferAttachments.length; i++) {
901 if( colorbufferAttachments[i] == ca ) {
902 return i;
903 }
904 }
905 return -1;
906 }
907
908 /**
909 * Returns the passed {@link Colorbuffer} if it is attached to this FBO, otherwise null.
910 * Implementation compares the <i>reference</i> only.
911 *
912 * <p>
913 * Note: Slow. Uses {@link #getColorbufferAttachmentPoint(Colorbuffer)} to determine it's attachment point
914 * to be used for {@link #getColorbuffer(int)}
915 * </p>
916 *
917 * @see #attachColorbuffer(GL, boolean)
918 * @see #attachColorbuffer(GL, boolean)
919 * @see #attachTexture2D(GL, int, boolean, int, int, int, int)
920 * @see #attachTexture2D(GL, int, int, int, int, int, int, int, int)
921 */
922 public final Colorbuffer getColorbuffer(final Colorbuffer ca) {
923 final int p = getColorbufferAttachmentPoint(ca);
924 return p>=0 ? getColorbuffer(p) : null;
925 }
926
927 /**
928 * Returns true if any attached {@link Colorbuffer} uses alpha,
929 * otherwise false.
930 */
931 public final boolean hasAttachmentUsingAlpha() {
932 final int caCount = getColorbufferCount();
933 boolean hasAlpha = false;
934 for(int i=0; i<caCount; i++) {
935 final Attachment ca = (Attachment)getColorbuffer(i);
936 if( null == ca ) {
937 break;
938 }
939 if( hasAlpha(ca.format) ) {
940 hasAlpha = true;
941 break;
942 }
943 }
944 return hasAlpha;
945 }
946
947 /**
948 * Creates an uninitialized FBObject instance.
949 * <p>
950 * Call {@link #init(GL, int, int, int)} .. etc to use it.
951 * </p>
952 */
953 public FBObject() {
954 this.initialized = false;
955
956 // TBD @ init
957 this.fullFBOSupport = false;
958 this.rgba8Avail = false;
959 this.depth24Avail = false;
960 this.depth32Avail = false;
961 this.stencil01Avail = false;
962 this.stencil04Avail = false;
963 this.stencil08Avail = false;
964 this.stencil16Avail = false;
965 this.packedDepthStencilAvail = false;
966 this.maxColorAttachments=-1;
967 this.maxSamples=-1;
968 this.maxTextureSize = 0;
969 this.maxRenderbufferSize = 0;
970
971 this.width = 0;
972 this.height = 0;
973 this.samples = 0;
974 this.vStatus = -1;
975 this.ignoreStatus = false;
976 this.fbName = 0;
977 this.bound = false;
978
979 this.colorbufferAttachments = null; // at init ..
980 this.colorbufferCount = 0;
981 this.textureAttachmentCount = 0;
982 this.depth = null;
983 this.stencil = null;
984 this.modified = true;
985
986 this.samplingSink = null;
987 this.samplingColorSink = null;
988 this.samplingSinkDirty = true;
989 }
990
991 /**
992 * Initializes this FBO's instance.
993 * <p>
994 * The sampling sink is not initializes, allowing manual assignment via {@link #setSamplingSink(FBObject)}
995 * if {@code newSamples > 0}.
996 * </p>
997 *
998 * <p>Leaves the FBO bound</p>
999 *
1000 * @param gl the current GL context
1001 * @param newWidth the initial width, it's minimum is capped to 1
1002 * @param newHeight the initial height, it's minimum is capped to 1
1003 * @param newSamples if > 0, MSAA will be used, otherwise no multisampling. Will be capped to {@link #getMaxSamples()}.
1004 * @throws IllegalStateException if already initialized
1005 * @throws GLException in case of an error, i.e. size too big, etc ..
1006 */
1007 public void init(final GL gl, final int newWidth, final int newHeight, final int newSamples) throws IllegalStateException, GLException {
1008 if( initialized ) {
1009 throw IllegalStateException("FBO already initialized");
1010 }
1011 if( !gl.hasBasicFBOSupport() ) {
1012 throw GLException("FBO not supported w/ context: "+gl.getContext()+", "+this);
1013 }
1014 fullFBOSupport = gl.hasFullFBOSupport();
1015
1016 rgba8Avail = gl.isGL2ES3() || gl.isExtensionAvailable(GLExtensions.OES_rgb8_rgba8);
1017 depth24Avail = fullFBOSupport || gl.isExtensionAvailable(GLExtensions.OES_depth24);
1018 depth32Avail = fullFBOSupport || gl.isExtensionAvailable(GLExtensions.OES_depth32);
1019 stencil01Avail = fullFBOSupport || gl.isExtensionAvailable(GLExtensions.OES_stencil1);
1020 stencil04Avail = fullFBOSupport || gl.isExtensionAvailable(GLExtensions.OES_stencil4);
1021 stencil08Avail = fullFBOSupport || gl.isExtensionAvailable(GLExtensions.OES_stencil8);
1022 stencil16Avail = fullFBOSupport;
1023
1024 packedDepthStencilAvail = fullFBOSupport ||
1025 gl.isExtensionAvailable(GLExtensions.OES_packed_depth_stencil) ||
1026 gl.isExtensionAvailable(GLExtensions.EXT_packed_depth_stencil) ;
1027
1028 final boolean NV_fbo_color_attachments = gl.isExtensionAvailable(GLExtensions.NV_fbo_color_attachments);
1029
1030 final int val[] = new int[1];
1031
1032 checkPreGLError(gl);
1033
1034 int realMaxColorAttachments = 1;
1035 maxColorAttachments = 1;
1036 if( fullFBOSupport || NV_fbo_color_attachments ) {
1037 try {
1038 val[0] = 0;
1039 gl.glGetIntegerv(GL2ES2.GL_MAX_COLOR_ATTACHMENTS, val, 0);
1040 realMaxColorAttachments = 1 <= val[0] ? val[0] : 1; // cap minimum to 1
1041 } catch (final GLException gle) { gle.printStackTrace(); }
1042 }
1043 maxColorAttachments = realMaxColorAttachments <= 8 ? realMaxColorAttachments : 8; // cap to limit array size
1044
1045 colorbufferAttachments = new Colorbuffer[maxColorAttachments];
1046 colorbufferCount = 0;
1047 textureAttachmentCount = 0;
1048
1049 maxSamples = gl.getMaxRenderbufferSamples(); // if > 0 implies fullFBOSupport
1050 gl.glGetIntegerv(GL.GL_MAX_TEXTURE_SIZE, val, 0);
1051 final int _maxTextureSize = val[0];
1052 if( 0 < USER_MAX_TEXTURE_SIZE ) {
1053 maxTextureSize = USER_MAX_TEXTURE_SIZE;
1054 } else {
1055 maxTextureSize = _maxTextureSize;
1056 }
1057 gl.glGetIntegerv(GL.GL_MAX_RENDERBUFFER_SIZE, val, 0);
1058 maxRenderbufferSize = val[0];
1059
1060 this.width = 0 < newWidth ? newWidth : 1;
1061 this.height = 0 < newHeight ? newHeight : 1;
1062 this.samples = newSamples <= maxSamples ? newSamples : maxSamples;
1063
1064 if(DEBUG_FBO) {
1065 System.err.println("FBObject.init() START: "+width+"x"+height+", "+newSamples+" -> "+this.samples+" samples");
1066 System.err.println("fullFBOSupport: "+fullFBOSupport);
1067 System.err.println("maxColorAttachments: "+maxColorAttachments+"/"+realMaxColorAttachments+" [capped/real]");
1068 System.err.println("maxSamples: "+maxSamples);
1069 System.err.println("maxTextureSize: "+_maxTextureSize+" -> "+maxTextureSize);
1070 System.err.println("maxRenderbufferSize: "+maxRenderbufferSize);
1071 System.err.println("rgba8: "+rgba8Avail);
1072 System.err.println("depth24: "+depth24Avail);
1073 System.err.println("depth32: "+depth32Avail);
1074 System.err.println("stencil01: "+stencil01Avail);
1075 System.err.println("stencil04: "+stencil04Avail);
1076 System.err.println("stencil08: "+stencil08Avail);
1077 System.err.println("stencil16: "+stencil16Avail);
1078 System.err.println("packedDepthStencil: "+packedDepthStencilAvail);
1079 System.err.println("NV_fbo_color_attachments: "+NV_fbo_color_attachments);
1080 System.err.println(gl.getContext().getGLVersion());
1081 System.err.println(JoglVersion.getGLStrings(gl, null, false).toString());
1082 }
1083
1084 checkPreGLError(gl);
1085
1086 if( width > maxRenderbufferSize || height > maxRenderbufferSize ) {
1087 throw GLException("Size "+width+"x"+height+" exceeds on of the maxima renderbuffer size "+maxRenderbufferSize+": \n\t"+this);
1088 }
1089
1090 modified = true;
1091 samplingSinkDirty = true;
1092
1093 // generate fbo ..
1094 gl.glGenFramebuffers(1, val, 0);
1095 fbName = val[0];
1096 if(0 == fbName) {
1097 throw GLException("null framebuffer");
1098 }
1099
1100 // bind fbo ..
1101 gl.glBindFramebuffer(GL.GL_FRAMEBUFFER, fbName);
1102 checkNoError(gl, gl.glGetError(), "FBObject Init.bindFB"); // throws GLException if error
1103 if(!gl.glIsFramebuffer(fbName)) {
1104 checkNoError(gl, GL.GL_INVALID_VALUE, "FBObject Init.isFB"); // throws GLException
1105 }
1106 bound = true;
1107 initialized = true;
1108
1109 vStatus = GL.GL_FRAMEBUFFER_INCOMPLETE_MISSING_ATTACHMENT; // always incomplete w/o attachments!
1110 if(DEBUG_FBO) {
1111 System.err.println("FBObject.init() END: "+this);
1112 ExceptionUtils.dumpStack(System.err);
1113 }
1114 }
1115
1116 /**
1117 * Resets this FBO's instance.
1118 * <p>
1119 * In case the new parameters are compatible with the current ones
1120 * no action will be performed and method returns immediately.<br>
1121 * Otherwise all attachments will be recreated
1122 * to match the new given parameters.
1123 * </p>
1124 * <p>
1125 * {@link #resetSamplingSink(GL)} is being issued immediately
1126 * to match the new configuration.
1127 * </p>
1128 *
1129 * <p>Leaves the FBO bound state untouched</p>
1130 *
1131 * @param gl the current GL context
1132 * @param newWidth the new width, it's minimum is capped to 1
1133 * @param newHeight the new height, it's minimum is capped to 1
1134 * @param newSamples if > 0, MSAA will be used, otherwise no multisampling. Will be capped to {@link #getMaxSamples()}.
1135 * @return {@code true} if this instance has been modified, otherwise {@code false}.
1136 * @throws IllegalStateException if not initialized via {@link #init(GL, int, int, int)}.
1137 * @throws GLException in case of an error, i.e. size too big, etc ..
1138 */
1139 public final boolean reset(final GL gl, int newWidth, int newHeight, int newSamples) throws GLException, IllegalStateException {
1140 if( !initialized ) {
1141 throw IllegalStateException("FBO not initialized");
1142 }
1143
1144 newSamples = newSamples <= maxSamples ? newSamples : maxSamples; // clamp
1145
1146 if( newWidth != width || newHeight != height || newSamples != samples ) {
1147 if( 0 >= newWidth ) { newWidth = 1; }
1148 if( 0 >= newHeight ) { newHeight = 1; }
1149 if( textureAttachmentCount > 0 && ( newWidth > 2 + maxTextureSize || newHeight > 2 + maxTextureSize ) ) {
1150 throw GLException("Size "+newWidth+"x"+newHeight+" exceeds on of the maximum texture size "+maxTextureSize+": \n\t"+this);
1151 }
1152 if( newWidth > maxRenderbufferSize || newHeight > maxRenderbufferSize ) {
1153 throw GLException("Size "+newWidth+"x"+newHeight+" exceeds on of the maxima renderbuffer size "+maxRenderbufferSize+": \n\t"+this);
1154 }
1155
1156 if(DEBUG_FBO) {
1157 System.err.println("FBObject.reset - START - "+width+"x"+height+", "+samples+" -> "+newWidth+"x"+newHeight+", "+newSamples+"; "+this);
1158 }
1159
1160 final boolean wasBound = isBound();
1161
1162 final int sampleCountChange;
1163 if( 0 < samples && 0 < newSamples || 0 == samples && 0 == newSamples ) {
1164 sampleCountChange = 0; // keep MSAA settings
1165 } else if( 0 == samples && 0 < newSamples ) {
1166 sampleCountChange = 1; // add MSAA
1167 } else if( 0 < samples && 0 == newSamples ) {
1168 sampleCountChange = -1; // remove MSAA
1169 } else {
1170 throw IllegalArgumentException("Error in sampleCount change: "+samples+" -> "+newSamples);
1171 }
1172 width = newWidth;
1173 height = newHeight;
1174 samples = newSamples;
1175
1176 modified = true;
1177 samplingSinkDirty = true;
1178
1179 detachAllImpl(gl, true, true, sampleCountChange);
1181
1182 if(!wasBound) {
1183 unbind(gl);
1184 }
1185
1186 if(DEBUG_FBO) {
1187 System.err.println("FBObject.reset - END - wasBound, "+wasBound+", "+this);
1188 }
1189 return true;
1190 } else {
1191 return false;
1192 }
1193 }
1194
1195 /**
1196 * Simply resets this instance's size only, w/o validation.
1197 *
1198 * <p>Leaves the FBO bound</p>
1199 *
1200 * @param gl the current GL context
1201 * @param newWidth the new width, it's minimum is capped to 1
1202 * @param newHeight the new height, it's minimum is capped to 1
1203 */
1204 private final void resetSizeImpl(final GL gl, final int newWidth, final int newHeight) {
1205 if(DEBUG_FBO) {
1206 System.err.println("FBObject.resetSize - START - "+width+"x"+height+", "+samples+" -> "+newWidth+"x"+newHeight);
1207 }
1208
1209 final int sampleCountChange = 0; // keep MSAA settings
1210 width = newWidth;
1211 height = newHeight;
1212
1213 modified = true;
1214 samplingSinkDirty = true;
1215
1216 detachAllImpl(gl, true, true, sampleCountChange);
1217
1218 if(DEBUG_FBO) {
1219 System.err.println("FBObject.resetSize - END - "+this);
1220 }
1221 }
1222
1223 private void validateAttachmentSize(final Attachment a) {
1224 final int aWidth = a.getWidth();
1225 final int aHeight = a.getHeight();
1226
1227 if( a instanceof TextureAttachment && ( aWidth > 2 + maxTextureSize || aHeight > 2 + maxTextureSize ) ) {
1228 throw GLException("Size "+aWidth+"x"+aHeight+" of "+a+" exceeds on of the maximum texture size "+maxTextureSize+": \n\t"+this);
1229 }
1230 if( aWidth > maxRenderbufferSize || aHeight > maxRenderbufferSize ) {
1231 throw GLException("Size "+aWidth+"x"+aHeight+" of "+a+" exceeds on of the maxima renderbuffer size "+maxRenderbufferSize+": \n\t"+this);
1232 }
1233 }
1234
1235 /**
1236 * Writes the internal format of the attachments to the given GLCapabilities object.
1237 * @param caps the destination for format bits
1238 */
1239 public final void formatToGLCapabilities(final GLCapabilities caps) {
1240 caps.setSampleBuffers(samples > 0);
1241 caps.setNumSamples(samples);
1242 caps.setDepthBits(0);
1243 caps.setStencilBits(0);
1244
1245 final Colorbuffer cb = samples > 0 ? getSamplingSink() : getColorbuffer(0);
1246 if(null != cb) {
1247 cb.formatToGLCapabilities(caps, rgba8Avail);
1248 }
1249 if(null != depth) {
1250 depth.formatToGLCapabilities(caps, rgba8Avail);
1251 }
1252 if(null != stencil && stencil != depth) {
1253 stencil.formatToGLCapabilities(caps, rgba8Avail);
1254 }
1255 }
1256
1257 /**
1258 * Note that the status may reflect an incomplete state during transition of attachments.
1259 * @return The FB status. {@link GL.GL_FRAMEBUFFER_COMPLETE} if ok, otherwise return GL FBO error state or -1
1260 * @see #validateStatus()
1261 */
1262 public final int getStatus() {
1263 return vStatus;
1264 }
1265
1266 /** return the {@link #getStatus()} as a string. */
1267 public final String getStatusString() {
1268 return getStatusString(vStatus);
1269 }
1270
1271 public static final String getStatusString(final int fbStatus) {
1272 switch(fbStatus) {
1273 case -1:
1274 return "NOT A FBO";
1275
1276 case GL.GL_FRAMEBUFFER_COMPLETE:
1277 return "OK";
1278
1279 case GL.GL_FRAMEBUFFER_INCOMPLETE_ATTACHMENT:
1280 return("FBO incomplete attachment\n");
1281 case GL.GL_FRAMEBUFFER_INCOMPLETE_MISSING_ATTACHMENT:
1282 return("FBO missing attachment");
1283 case GL.GL_FRAMEBUFFER_INCOMPLETE_DIMENSIONS:
1284 return("FBO attached images must have same dimensions");
1285 case GL.GL_FRAMEBUFFER_INCOMPLETE_FORMATS:
1286 return("FBO attached images must have same format");
1287 case GL2GL3.GL_FRAMEBUFFER_INCOMPLETE_DRAW_BUFFER:
1288 return("FBO missing draw buffer");
1289 case GL2GL3.GL_FRAMEBUFFER_INCOMPLETE_READ_BUFFER:
1290 return("FBO missing read buffer");
1291 case GL.GL_FRAMEBUFFER_INCOMPLETE_MULTISAMPLE:
1292 return("FBO missing multisample buffer");
1293 case GL3ES3.GL_FRAMEBUFFER_INCOMPLETE_LAYER_TARGETS:
1294 return("FBO missing layer targets");
1295
1296 case GL.GL_FRAMEBUFFER_UNSUPPORTED:
1297 return("Unsupported FBO format");
1298 case GL2ES3.GL_FRAMEBUFFER_UNDEFINED:
1299 return("FBO undefined");
1300
1301 case 0:
1302 return("FBO implementation fault");
1303 default:
1304 return("FBO incomplete, implementation ERROR "+toHexString(fbStatus));
1305 }
1306 }
1307
1308 /**
1309 * The status may even be valid if incomplete during transition of attachments.
1310 * @see #getStatus()
1311 */
1312 public final boolean isStatusValid() {
1313 switch(vStatus) {
1314 case GL.GL_FRAMEBUFFER_COMPLETE:
1315 return true;
1316
1317 case GL.GL_FRAMEBUFFER_INCOMPLETE_ATTACHMENT:
1318 case GL.GL_FRAMEBUFFER_INCOMPLETE_MISSING_ATTACHMENT:
1319 case GL.GL_FRAMEBUFFER_INCOMPLETE_DIMENSIONS:
1320 case GL.GL_FRAMEBUFFER_INCOMPLETE_FORMATS:
1321 case GL2GL3.GL_FRAMEBUFFER_INCOMPLETE_DRAW_BUFFER:
1322 case GL2GL3.GL_FRAMEBUFFER_INCOMPLETE_READ_BUFFER:
1323 case GL.GL_FRAMEBUFFER_INCOMPLETE_MULTISAMPLE:
1324 case GL3ES3.GL_FRAMEBUFFER_INCOMPLETE_LAYER_TARGETS:
1325 if(0 == colorbufferCount || null == depth) {
1326 // we are in transition
1327 return true;
1328 }
1329
1330 case GL.GL_FRAMEBUFFER_UNSUPPORTED:
1331 case GL2ES3.GL_FRAMEBUFFER_UNDEFINED:
1332
1333 case 0:
1334 default:
1335 if(DEBUG_FBO) {
1336 System.err.println("Framebuffer " + fbName + " is incomplete, status = " + toHexString(vStatus) +
1337 " : " + getStatusString(vStatus));
1338 }
1339 return false;
1340 }
1341 }
1342
1343 private static int checkPreGLError(final GL gl) {
1344 final int glerr = gl.glGetError();
1345 if(DEBUG_FBO && GL.GL_NO_ERROR != glerr) {
1346 System.err.println("Pre-existing GL error: "+toHexString(glerr));
1347 ExceptionUtils.dumpStack(System.err);
1348 }
1349 return glerr;
1350 }
1351
1352 private final boolean checkNoError(final GL gl, final int err, final String exceptionMessage) throws GLException {
1353 if(GL.GL_NO_ERROR != err) {
1354 if(null != gl) {
1355 destroy(gl);
1356 }
1357 if(null != exceptionMessage) {
1358 throw GLException(exceptionMessage+" GL Error "+toHexString(err)+" of "+this.toString());
1359 }
1360 return false;
1361 }
1362 return true;
1363 }
1364
1365 private final void checkInitialized() throws GLException {
1366 if(!initialized) {
1367 throw GLException("FBO not initialized, call init(GL) first.");
1368 }
1369 }
1370
1371 /**
1372 * Attaches a {@link Colorbuffer}, i.e. {@link TextureAttachment}, to this FBO's instance at the given attachment point,
1373 * selecting the texture data type and format automatically.
1374 *
1375 * <p>Using default min/mag filter {@link GL#GL_NEAREST} and default wrapS/wrapT {@link GL#GL_CLAMP_TO_EDGE}.</p>
1376 *
1377 * <p>Leaves the FBO bound.</p>
1378 *
1379 * @param gl the current GL context
1380 * @param attachmentPoint the color attachment point ranging from [0..{@link #getMaxColorAttachments()}-1]
1381 * @param alpha set to <code>true</code> if you request alpha channel, otherwise <code>false</code>;
1382 * @return TextureAttachment instance describing the new attached texture colorbuffer if bound and configured successfully, otherwise GLException is thrown
1383 * @throws GLException in case the texture colorbuffer couldn't be allocated or MSAA has been chosen
1384 * @see #createColorTextureAttachment(GLProfile, boolean, int, int)
1385 */
1386 public final TextureAttachment attachTexture2D(final GL gl, final int attachmentPoint, final boolean alpha) throws GLException {
1387 return attachColorbuffer(gl, attachmentPoint,
1388 createColorTextureAttachment(gl, alpha, width, height)).getTextureAttachment();
1389 }
1390
1391 /**
1392 * Attaches a {@link Colorbuffer}, i.e. {@link TextureAttachment}, to this FBO's instance at the given attachment point,
1393 * selecting the texture data type and format automatically.
1394 *
1395 * <p>Leaves the FBO bound.</p>
1396 *
1397 * @param gl the current GL context
1398 * @param attachmentPoint the color attachment point ranging from [0..{@link #getMaxColorAttachments()}-1]
1399 * @param alpha set to <code>true</code> if you request alpha channel, otherwise <code>false</code>;
1400 * @param magFilter if > 0 value for {@link GL#GL_TEXTURE_MAG_FILTER}
1401 * @param minFilter if > 0 value for {@link GL#GL_TEXTURE_MIN_FILTER}
1402 * @param wrapS if > 0 value for {@link GL#GL_TEXTURE_WRAP_S}
1403 * @param wrapT if > 0 value for {@link GL#GL_TEXTURE_WRAP_T}
1404 * @return TextureAttachment instance describing the new attached texture colorbuffer if bound and configured successfully, otherwise GLException is thrown
1405 * @throws GLException in case the texture colorbuffer couldn't be allocated or MSAA has been chosen
1406 * @see #createColorTextureAttachment(GLProfile, boolean, int, int, int, int, int, int)
1407 */
1408 public final TextureAttachment attachTexture2D(final GL gl, final int attachmentPoint, final boolean alpha, final int magFilter, final int minFilter, final int wrapS, final int wrapT) throws GLException {
1409 return attachColorbuffer(gl, attachmentPoint,
1410 createColorTextureAttachment(gl, alpha, width, height, magFilter, minFilter, wrapS, wrapT)).getTextureAttachment();
1411 }
1412
1413 /**
1414 * Attaches a {@link Colorbuffer}, i.e. {@link TextureAttachment}, to this FBO's instance at the given attachment point.
1415 *
1416 * <p>Leaves the FBO bound.</p>
1417 *
1418 * @param gl the current GL context
1419 * @param attachmentPoint the color attachment point ranging from [0..{@link #getMaxColorAttachments()}-1]
1420 * @param internalFormat internalFormat parameter to {@link GL#glTexImage2D(int, int, int, int, int, int, int, int, long)}
1421 * @param dataFormat format parameter to {@link GL#glTexImage2D(int, int, int, int, int, int, int, int, long)}
1422 * @param dataType type parameter to {@link GL#glTexImage2D(int, int, int, int, int, int, int, int, long)}
1423 * @param magFilter if > 0 value for {@link GL#GL_TEXTURE_MAG_FILTER}
1424 * @param minFilter if > 0 value for {@link GL#GL_TEXTURE_MIN_FILTER}
1425 * @param wrapS if > 0 value for {@link GL#GL_TEXTURE_WRAP_S}
1426 * @param wrapT if > 0 value for {@link GL#GL_TEXTURE_WRAP_T}
1427 * @return TextureAttachment instance describing the new attached texture colorbuffer if bound and configured successfully, otherwise GLException is thrown
1428 * @throws GLException in case the texture colorbuffer couldn't be allocated or MSAA has been chosen
1429 * @see #createColorTextureAttachment(int, int, int, int, int, int, int, int, int)
1430 */
1431 public final TextureAttachment attachTexture2D(final GL gl, final int attachmentPoint,
1432 final int internalFormat, final int dataFormat, final int dataType,
1433 final int magFilter, final int minFilter, final int wrapS, final int wrapT) throws GLException {
1434 return attachColorbuffer(gl, attachmentPoint,
1435 createColorTextureAttachment(internalFormat, width, height, dataFormat, dataType, magFilter, minFilter, wrapS, wrapT)).getTextureAttachment();
1436 }
1437
1438 /**
1439 * Creates a {@link ColorAttachment}, selecting the format automatically.
1440 * <p>
1441 * For GLES3, sampling-sink {@link Colorbuffer} format <b>must be equal</b> w/ the sampling-source {@link Colorbuffer}.
1442 * Implementation aligns w/ {@link #createColorTextureAttachment(GLProfile, boolean, int, int, int, int, int, int)}
1443 * and is enforced via {@link #sampleSinkExFormatMismatch(GL)}.
1444 * </p>
1445 *
1446 * @param alpha set to <code>true</code> if you request alpha channel, otherwise <code>false</code>;
1447 * @return uninitialized ColorAttachment instance describing the new attached colorbuffer
1448 */
1449 public final ColorAttachment createColorAttachment(final boolean alpha) {
1450 final int internalFormat;
1451
1452 if( rgba8Avail ) {
1453 internalFormat = alpha ? GL.GL_RGBA8 : GL.GL_RGB8 ;
1454 } else {
1455 internalFormat = alpha ? GL.GL_RGBA4 : GL.GL_RGB565;
1456 }
1457 return createColorAttachment(internalFormat, samples, width, height);
1458 }
1459
1460 /**
1461 * Creates a {@link ColorAttachment}, selecting the format automatically.
1462 * <p>
1463 * For GLES3, sampling-sink {@link Colorbuffer} format <b>must be equal</b> w/ the sampling-source {@link Colorbuffer}.
1464 * Implementation aligns w/ {@link #createColorTextureAttachment(GLProfile, boolean, int, int, int, int, int, int)}
1465 * and is enforced via {@link #sampleSinkExFormatMismatch(GL)}.
1466 * </p>
1467 *
1468 * @param alpha set to <code>true</code> if you request alpha channel, otherwise <code>false</code>;
1469 * @return uninitialized ColorAttachment instance describing the new attached colorbuffer
1470 */
1471 public static final ColorAttachment createColorAttachment(final int internalFormat, final int samples, final int width, final int height) {
1472 return new ColorAttachment(internalFormat, samples, width, height, 0 /* name not yet determined */);
1473 }
1474
1475 public static final RenderAttachment createRenderAttachment(final Type type, final int internalFormat, final int samples, final int width, final int height) {
1476 return new RenderAttachment(type, internalFormat, samples, width, height, 0 /* name not yet determined */);
1477 }
1478
1479 /**
1480 * Attaches a newly created and {@link Colorbuffer#initialize(GL) initialized} {@link Colorbuffer}, i.e. a {@link ColorAttachment},
1481 * at the given attachment point.
1482 * <p>
1483 * The {@link ColorAttachment} is created using {@code alpha} if {@code true} and current {@code sample count} and {@code size}.
1484 * </p>
1485 *
1486 * <p>Leaves the FBO bound.</p>
1487 *
1488 * @param gl the current GL context
1489 * @param attachmentPoint the color attachment point ranging from [0..{@link #getMaxColorAttachments()}-1]
1490 * @param alpha set to <code>true</code> if you request alpha channel, otherwise <code>false</code>;
1491 * @return ColorAttachment instance describing the new attached colorbuffer if bound and configured successfully, otherwise GLException is thrown
1492 * @throws GLException in case the colorbuffer couldn't be allocated
1493 * @see #createColorAttachment(boolean)
1494 */
1495 public final ColorAttachment attachColorbuffer(final GL gl, final int attachmentPoint, final boolean alpha) throws GLException {
1496 return attachColorbuffer(gl, attachmentPoint, createColorAttachment(alpha)).getColorAttachment();
1497 }
1498
1499 /**
1500 * Attaches a newly created and {@link Colorbuffer#initialize(GL) initialized} {@link Colorbuffer}, i.e. a {@link ColorAttachment},
1501 * at the given attachment point.
1502 * <p>
1503 * The {@link ColorAttachment} is created using the given {@code internalFormat} and current {@code sample count} and {@code size}.
1504 * </p>
1505 *
1506 * <p>Leaves the FBO bound.</p>
1507 *
1508 * @param gl the current GL context
1509 * @param attachmentPoint the color attachment point ranging from [0..{@link #getMaxColorAttachments()}-1]
1510 * @param internalFormat usually {@link GL#GL_RGBA4}, {@link GL#GL_RGB5_A1}, {@link GL#GL_RGB565}, {@link GL#GL_RGB8} or {@link GL#GL_RGBA8}
1511 * @return ColorAttachment instance describing the new attached colorbuffer if bound and configured successfully, otherwise GLException is thrown
1512 * @throws GLException in case the colorbuffer couldn't be allocated
1513 * @throws IllegalArgumentException if <code>internalFormat</code> doesn't reflect a colorbuffer
1514 */
1515 public final ColorAttachment attachColorbuffer(final GL gl, final int attachmentPoint, final int internalFormat) throws GLException, IllegalArgumentException {
1516 final Attachment.Type atype = Attachment.Type.determine(internalFormat);
1517 if( Attachment.Type.COLOR != atype ) {
1518 throw IllegalArgumentException("colorformat invalid: "+toHexString(internalFormat)+", "+this);
1519 }
1520
1521 return attachColorbuffer(gl, attachmentPoint, createColorAttachment(internalFormat, samples, width, height)).getColorAttachment();
1522 }
1523
1524 /**
1525 * Attaches a {@link Colorbuffer} at the given attachment point
1526 * and {@link Colorbuffer#initialize(GL) initializes} it, if not done yet.
1527 * <p>
1528 * {@link Colorbuffer} may be a {@link ColorAttachment} or {@link TextureAttachment}.
1529 * </p>
1530 * <p>
1531 * If {@link Colorbuffer} is a {@link TextureAttachment} and is uninitialized, i.e. it's texture name is <code>zero</code>,
1532 * a new texture name is generated and setup w/ the texture parameter.<br/>
1533 * Otherwise, i.e. texture name is not <code>zero</code>, the passed TextureAttachment <code>texA</code> is
1534 * considered complete and assumed matching this FBO requirement. A GL error may occur is the latter is untrue.
1535 * </p>
1536 *
1537 * <p>Leaves the FBO bound.</p>
1538 *
1539 * @param gl
1540 * @param attachmentPoint the color attachment point ranging from [0..{@link #getMaxColorAttachments()}-1]
1541 * @param colbuf the to be attached {@link Colorbuffer}
1542 * @return given {@link Colorbuffer} instance if bound and configured successfully, otherwise GLException is thrown
1543 * @throws GLException in case the colorbuffer couldn't be allocated or MSAA has been chosen in case of a {@link TextureAttachment}
1544 */
1545 public final Colorbuffer attachColorbuffer(final GL gl, final int attachmentPoint, final Colorbuffer colbuf) throws GLException {
1546 bind(gl);
1547 return attachColorbufferImpl(gl, attachmentPoint, colbuf);
1548 }
1549
1550 private final Colorbuffer attachColorbufferImpl(final GL gl, final int attachmentPoint, final Colorbuffer colbuf) throws GLException {
1551 validateAddColorAttachment(attachmentPoint, colbuf);
1552 validateAttachmentSize((Attachment)colbuf);
1553
1554 final boolean initializedColorbuf = colbuf.initialize(gl);
1555 addColorAttachment(attachmentPoint, colbuf, false);
1556
1557 if( colbuf.isTextureAttachment() ) {
1558 final TextureAttachment texA = colbuf.getTextureAttachment();
1559 if( samples > 0 ) {
1560 removeColorAttachment(attachmentPoint, texA);
1561 if( initializedColorbuf ) {
1562 texA.free(gl);
1563 }
1564 throw GLException("Texture2D not supported w/ MSAA. If you have enabled MSAA with exisiting texture attachments, you may want to detach them via detachAllTexturebuffer(gl).");
1565 }
1566
1567 // Set up the color buffer for use as a renderable texture:
1568 gl.glFramebufferTexture2D(GL.GL_FRAMEBUFFER,
1569 GL.GL_COLOR_ATTACHMENT0 + attachmentPoint,
1570 GL.GL_TEXTURE_2D, texA.getName(), 0);
1571
1572 if(!ignoreStatus) {
1573 updateStatus(gl);
1574 if( !isStatusValid() ) {
1575 detachColorbuffer(gl, attachmentPoint, true);
1576 throw GLException("attachTexture2D "+texA+" at "+attachmentPoint+" failed: "+getStatusString()+", "+this);
1577 }
1578 }
1579 } else {
1580 final ColorAttachment colA = colbuf.getColorAttachment();
1581
1582 // Attach the color buffer
1583 gl.glFramebufferRenderbuffer(GL.GL_FRAMEBUFFER,
1584 GL.GL_COLOR_ATTACHMENT0 + attachmentPoint,
1585 GL.GL_RENDERBUFFER, colA.getName());
1586
1587 if(!ignoreStatus) {
1588 updateStatus(gl);
1589 if( !isStatusValid() ) {
1590 detachColorbuffer(gl, attachmentPoint, true);
1591 throw GLException("attachColorbuffer "+colA+" at "+attachmentPoint+" failed: "+getStatusString()+", "+this);
1592 }
1593 }
1594 }
1595 if(DEBUG_FBO) {
1596 System.err.println("FBObject.attachColorbuffer.X: [attachmentPoint "+attachmentPoint+", colbuf "+colbuf+"]: "+this);
1597 }
1598 return colbuf;
1599 }
1600
1601 private final int getDepthIFormat(final int reqBits) {
1602 if( 32 <= reqBits && depth32Avail ) {
1603 return GL.GL_DEPTH_COMPONENT32;
1604 } else if( 24 <= reqBits && ( depth24Avail || depth32Avail ) ) {
1605 if( depth24Avail ) {
1606 return GL.GL_DEPTH_COMPONENT24;
1607 } else {
1608 return GL.GL_DEPTH_COMPONENT32;
1609 }
1610 } else {
1611 return GL.GL_DEPTH_COMPONENT16;
1612 }
1613 }
1614 private final int getStencilIFormat(final int reqBits) {
1615 if( 16 <= reqBits && stencil16Avail ) {
1616 return GL2GL3.GL_STENCIL_INDEX16;
1617 } else if( 8 <= reqBits && ( stencil08Avail || stencil16Avail ) ) {
1618 if( stencil08Avail ) {
1619 return GL.GL_STENCIL_INDEX8;
1620 } else {
1621 return GL2GL3.GL_STENCIL_INDEX16;
1622 }
1623 } else if( 4 <= reqBits && ( stencil04Avail || stencil08Avail || stencil16Avail ) ) {
1624 if( stencil04Avail ) {
1625 return GL.GL_STENCIL_INDEX4;
1626 } else if( stencil08Avail ) {
1627 return GL.GL_STENCIL_INDEX8;
1628 } else {
1629 return GL2GL3.GL_STENCIL_INDEX16;
1630 }
1631 } else if( 1 <= reqBits && ( stencil01Avail || stencil04Avail || stencil08Avail || stencil16Avail ) ) {
1632 if( stencil01Avail ) {
1633 return GL.GL_STENCIL_INDEX1;
1634 } else if( stencil04Avail ) {
1635 return GL.GL_STENCIL_INDEX4;
1636 } else if( stencil08Avail ) {
1637 return GL.GL_STENCIL_INDEX8;
1638 } else {
1639 return GL2GL3.GL_STENCIL_INDEX16;
1640 }
1641 } else {
1642 throw GLException("stencil buffer n/a");
1643 }
1644 }
1645
1646 /** Request default bit count for depth- or stencil buffer (depth 24 bits, stencil 8 bits), value {@value} */
1647 public static final int DEFAULT_BITS = 0;
1648
1649 /**
1650 * Request current context drawable's <i>requested</i>
1651 * {@link GLCapabilitiesImmutable#getDepthBits() depth-} or {@link GLCapabilitiesImmutable#getStencilBits() stencil-}bits; value {@value} */
1652 public static final int REQUESTED_BITS = -1;
1653
1654 /**
1655 * Request current context drawable's <i>chosen</i>
1656 * {@link GLCapabilitiesImmutable#getDepthBits() depth-} or {@link GLCapabilitiesImmutable#getStencilBits() stencil-}bits; value {@value} */
1657 public static final int CHOSEN_BITS = -2;
1658
1659 /** Request maximum bit count for depth- or stencil buffer (depth 32 bits, stencil 16 bits), value {@value} */
1660 public static final int MAXIMUM_BITS = -3;
1661
1662
1663 /**
1664 * Attaches one depth, stencil or packed-depth-stencil buffer to this FBO's instance,
1665 * selecting the internalFormat automatically.
1666 * <p>
1667 * Stencil and depth buffer can be attached only once.
1668 * </p>
1669 * <p>
1670 * In case the bit-count is not supported,
1671 * the next available one is chosen, i.e. next higher (preferred) or lower bit-count.
1672 * </p>
1673 * <p>
1674 * Use {@link #getDepthAttachment()} and/or {@link #getStencilAttachment()} to retrieve details
1675 * about the attached buffer. The details cannot be returned, since it's possible 2 buffers
1676 * are being created, depth and stencil.
1677 * </p>
1678 *
1679 * <p>Leaves the FBO bound.</p>
1680 *
1681 * @param gl
1682 * @param atype either {@link Type#DEPTH}, {@link Type#STENCIL} or {@link Type#DEPTH_STENCIL}
1683 * @param reqBits desired bits for depth or stencil,
1684 * may use generic values {@link #DEFAULT_BITS}, {@link #REQUESTED_BITS}, {@link #CHOSEN_BITS} or {@link #MAXIMUM_BITS}.
1685 * @throws GLException in case the renderbuffer couldn't be allocated or one is already attached.
1686 * @throws IllegalArgumentException
1687 * @see #getDepthAttachment()
1688 * @see #getStencilAttachment()
1689 */
1690 public final void attachRenderbuffer(final GL gl, final Attachment.Type atype, final int reqBits) throws GLException, IllegalArgumentException {
1691 final int reqDepth, reqStencil;
1692 if( MAXIMUM_BITS > reqBits ) {
1693 throw IllegalArgumentException("reqBits out of range, shall be >= "+MAXIMUM_BITS);
1694 } else if( MAXIMUM_BITS == reqBits ) {
1695 reqDepth = 32;
1696 reqStencil = 16;
1697 } else if( CHOSEN_BITS == reqBits ) {
1698 final GLCapabilitiesImmutable caps = gl.getContext().getGLDrawable().getChosenGLCapabilities();
1699 reqDepth = caps.getDepthBits();
1700 reqStencil = caps.getStencilBits();
1701 } else if( REQUESTED_BITS == reqBits ) {
1702 final GLCapabilitiesImmutable caps = gl.getContext().getGLDrawable().getRequestedGLCapabilities();
1703 reqDepth = caps.getDepthBits();
1704 reqStencil = caps.getStencilBits();
1705 } else if( DEFAULT_BITS == reqBits ) {
1706 reqDepth = 24;
1707 reqStencil = 8;
1708 } else {
1709 reqDepth = reqBits;
1710 reqStencil = reqBits;
1711 }
1712 final int internalFormat;
1713 int internalStencilFormat = -1;
1714
1715 switch ( atype ) {
1716 case DEPTH:
1717 internalFormat = getDepthIFormat(reqDepth);
1718 break;
1719
1720 case STENCIL:
1721 internalFormat = getStencilIFormat(reqStencil);
1722 break;
1723
1724 case DEPTH_STENCIL:
1725 if( packedDepthStencilAvail ) {
1726 internalFormat = GL.GL_DEPTH24_STENCIL8;
1727 } else {
1728 internalFormat = getDepthIFormat(reqDepth);
1729 internalStencilFormat = getStencilIFormat(reqStencil);
1730 }
1731 break;
1732 default:
1733 throw IllegalArgumentException("only depth/stencil types allowed, was "+atype+", "+this);
1734 }
1735 attachRenderbufferImpl(gl, atype, internalFormat);
1736
1737 if(0<=internalStencilFormat) {
1738 attachRenderbufferImpl(gl, Attachment.Type.STENCIL, internalStencilFormat);
1739 }
1740 }
1741
1742 /**
1743 * Attaches one depth, stencil or packed-depth-stencil buffer to this FBO's instance,
1744 * depending on the <code>internalFormat</code>.
1745 * <p>
1746 * Stencil and depth buffer can be attached only once.
1747 * </p>
1748 * <p>
1749 * Use {@link #getDepthAttachment()} and/or {@link #getStencilAttachment()} to retrieve details
1750 * about the attached buffer. The details cannot be returned, since it's possible 2 buffers
1751 * are being created, depth and stencil.
1752 * </p>
1753 *
1754 * <p>Leaves the FBO bound.</p>
1755 *
1756 * @param gl the current GL context
1757 * @param internalFormat {@link GL#GL_DEPTH_COMPONENT16}, {@link GL#GL_DEPTH_COMPONENT24}, {@link GL#GL_DEPTH_COMPONENT32},
1758 * {@link GL#GL_STENCIL_INDEX1}, {@link GL#GL_STENCIL_INDEX4}, {@link GL#GL_STENCIL_INDEX8}
1759 * or {@link GL#GL_DEPTH24_STENCIL8}
1760 * @throws GLException in case the renderbuffer couldn't be allocated or one is already attached.
1761 * @throws IllegalArgumentException
1762 * @see #getDepthAttachment()
1763 * @see #getStencilAttachment()
1764 */
1765 public final void attachRenderbuffer(final GL gl, final int internalFormat) throws GLException, IllegalArgumentException {
1766 final Attachment.Type atype = Attachment.Type.determine(internalFormat);
1767 if( Attachment.Type.DEPTH != atype && Attachment.Type.STENCIL != atype && Attachment.Type.DEPTH_STENCIL != atype ) {
1768 throw IllegalArgumentException("renderformat invalid: "+toHexString(internalFormat)+", "+this);
1769 }
1770 attachRenderbufferImpl(gl, atype, internalFormat);
1771 }
1772
1773 protected final void attachRenderbufferImpl(final GL gl, final Attachment.Type atype, final int internalFormat) throws GLException {
1774 if( null != depth && ( Attachment.Type.DEPTH == atype || Attachment.Type.DEPTH_STENCIL == atype ) ) {
1775 throw GLException("FBO depth buffer already attached (rb "+depth+"), type is "+atype+", "+toHexString(internalFormat)+", "+this);
1776 }
1777 if( null != stencil && ( Attachment.Type.STENCIL== atype || Attachment.Type.DEPTH_STENCIL == atype ) ) {
1778 throw GLException("FBO stencil buffer already attached (rb "+stencil+"), type is "+atype+", "+toHexString(internalFormat)+", "+this);
1779 }
1780 bind(gl);
1781
1782 attachRenderbufferImpl2(gl, atype, internalFormat);
1783 }
1784
1785 private final void attachRenderbufferImpl2(final GL gl, final Attachment.Type atype, final int internalFormat) throws GLException {
1786 // atype and current depth and stencil instance are already validated in 'attachRenderbufferImpl(..)'
1787 if( Attachment.Type.DEPTH == atype ) {
1788 if(null == depth) {
1789 depth = createRenderAttachment(Type.DEPTH, internalFormat, samples, width, height);
1790 } else {
1791 depth.setSize(width, height);
1792 depth.setSamples(samples);
1793 }
1794 validateAttachmentSize(depth);
1795 depth.initialize(gl);
1796 } else if( Attachment.Type.STENCIL == atype ) {
1797 if(null == stencil) {
1798 stencil = createRenderAttachment(Type.STENCIL, internalFormat, samples, width, height);
1799 } else {
1800 stencil.setSize(width, height);
1801 stencil.setSamples(samples);
1802 }
1803 validateAttachmentSize(stencil);
1804 stencil.initialize(gl);
1805 } else if( Attachment.Type.DEPTH_STENCIL == atype ) {
1806 if(null == depth) {
1807 if(null != stencil) {
1808 throw InternalError("XXX: DEPTH_STENCIL, depth was null, stencil not: "+this.toString());
1809 }
1810 depth = createRenderAttachment(Type.DEPTH_STENCIL, internalFormat, samples, width, height);
1811 } else {
1812 depth.setSize(width, height);
1813 depth.setSamples(samples);
1814 }
1815 validateAttachmentSize(depth);
1816 depth.initialize(gl);
1817 // DEPTH_STENCIL shares buffer w/ depth and stencil
1818 stencil = depth;
1819 }
1820
1821 // Attach the buffer
1822 if( Attachment.Type.DEPTH == atype ) {
1823 gl.glFramebufferRenderbuffer(GL.GL_FRAMEBUFFER, GL.GL_DEPTH_ATTACHMENT, GL.GL_RENDERBUFFER, depth.getName());
1824 } else if( Attachment.Type.STENCIL == atype ) {
1825 gl.glFramebufferRenderbuffer(GL.GL_FRAMEBUFFER, GL.GL_STENCIL_ATTACHMENT, GL.GL_RENDERBUFFER, stencil.getName());
1826 } else if( Attachment.Type.DEPTH_STENCIL == atype ) {
1827 gl.glFramebufferRenderbuffer(GL.GL_FRAMEBUFFER, GL.GL_DEPTH_ATTACHMENT, GL.GL_RENDERBUFFER, depth.getName());
1828 gl.glFramebufferRenderbuffer(GL.GL_FRAMEBUFFER, GL.GL_STENCIL_ATTACHMENT, GL.GL_RENDERBUFFER, stencil.getName());
1829 }
1830
1831 modified = true;
1832
1833 if(!ignoreStatus) {
1834 updateStatus(gl);
1835 if( !isStatusValid() ) {
1836 detachRenderbuffer(gl, atype, true);
1837 throw GLException("renderbuffer [attachmentType "+atype+", iformat "+toHexString(internalFormat)+"] failed: "+this.getStatusString()+", "+this.toString());
1838 }
1839 }
1840
1841 if(DEBUG_FBO) {
1842 System.err.println("FBObject.attachRenderbuffer.X: [attachmentType "+atype+", iformat "+toHexString(internalFormat)+"]: "+this);
1843 }
1844 }
1845
1846 /**
1847 * Detaches a {@link Colorbuffer}, i.e. {@link ColorAttachment} or {@link TextureAttachment}.
1848 * <p>Leaves the FBO bound!</p>
1849 *
1850 * @param gl
1851 * @param attachmentPoint
1852 * @param dispose true if the Colorbuffer shall be disposed
1853 * @return the detached Colorbuffer
1854 * @throws IllegalArgumentException
1855 */
1856 public final Colorbuffer detachColorbuffer(final GL gl, final int attachmentPoint, final boolean dispose) throws IllegalArgumentException {
1857 bind(gl);
1858
1859 final Colorbuffer res = detachColorbufferImpl(gl, attachmentPoint, dispose ? DetachAction.DISPOSE : DetachAction.NONE, 0);
1860 if(null == res) {
1861 throw IllegalArgumentException("ColorAttachment at "+attachmentPoint+", not attached, "+this);
1862 }
1863 if(DEBUG_FBO) {
1864 System.err.println("FBObject.detachColorbuffer.X: [attachmentPoint "+attachmentPoint+", dispose "+dispose+"]: "+res+", "+this);
1865 }
1866 return res;
1867 }
1868
1869 private final Colorbuffer detachColorbufferImpl(final GL gl, final int attachmentPoint, final DetachAction detachAction, final int sampleCountChange) {
1870 final Colorbuffer colbufOld = colorbufferAttachments[attachmentPoint]; // shortcut, don't validate here
1871
1872 if(null == colbufOld) {
1873 return null;
1874 }
1875
1876 removeColorAttachment(attachmentPoint, colbufOld);
1877
1878 if( colbufOld.isTextureAttachment() ) {
1879 final TextureAttachment texA = colbufOld.getTextureAttachment();
1880 if( 0 != texA.getName() ) {
1881 gl.glFramebufferTexture2D(GL.GL_FRAMEBUFFER,
1882 GL.GL_COLOR_ATTACHMENT0 + attachmentPoint,
1883 GL.GL_TEXTURE_2D, 0, 0);
1884 gl.glBindTexture(GL.GL_TEXTURE_2D, 0);
1885 switch(detachAction) {
1886 case DISPOSE:
1887 case RECREATE:
1888 texA.free(gl);
1889 break;
1890 default:
1891 }
1892 }
1893 if(DetachAction.RECREATE == detachAction) {
1894 final Colorbuffer colbufNew;
1895 if( 0 < sampleCountChange ) {
1896 // switch to MSAA: TextureAttachment -> ColorAttachment
1897 colbufNew = createColorAttachment(hasAlpha(texA.format));
1898 } else {
1899 // keep MSAA settings
1900 texA.setSize(width, height);
1901 colbufNew = texA;
1902 }
1903 attachColorbufferImpl(gl, attachmentPoint, colbufNew);
1904 }
1905 } else {
1906 final ColorAttachment colA = colbufOld.getColorAttachment();
1907 if( 0 != colA.getName() ) {
1908 gl.glFramebufferRenderbuffer(GL.GL_FRAMEBUFFER,
1909 GL.GL_COLOR_ATTACHMENT0+attachmentPoint,
1910 GL.GL_RENDERBUFFER, 0);
1911 switch(detachAction) {
1912 case DISPOSE:
1913 case RECREATE:
1914 colA.free(gl);
1915 break;
1916 default:
1917 }
1918 }
1919 if(DetachAction.RECREATE == detachAction) {
1920 final Colorbuffer colbufNew;
1921 if( 0 <= sampleCountChange || null == samplingColorSink ) {
1922 // keep ColorAttachment,
1923 // including 'switch to non-MSAA' if no samplingColorSink is available
1924 // to determine whether a TextureAttachment or ColorAttachment is desired!
1925 colA.setSize(width, height);
1926 colA.setSamples(samples);
1927 colbufNew = colA;
1928 } else {
1929 // switch to non MSAA
1930 if( samplingColorSink.isTextureAttachment() ) {
1931 final TextureAttachment samplingTextureSink = samplingColorSink.getTextureAttachment();
1932 colbufNew = createColorTextureAttachment(samplingTextureSink.format, width, height,
1933 samplingTextureSink.dataFormat, samplingTextureSink.dataType,
1934 samplingTextureSink.magFilter, samplingTextureSink.minFilter,
1935 samplingTextureSink.wrapS, samplingTextureSink.wrapT);
1936 } else {
1937 colbufNew = createColorAttachment(samplingColorSink.getFormat(), 0, width, height);
1938 }
1939 }
1940 attachColorbuffer(gl, attachmentPoint, colbufNew);
1941 }
1942 }
1943 return colbufOld;
1944 }
1945
1946 private final void freeAllColorbufferImpl(final GL gl) {
1947 for(int i=0; i<maxColorAttachments; i++) {
1948 final Colorbuffer colbuf = colorbufferAttachments[i]; // shortcut, don't validate here
1949
1950 if(null == colbuf) {
1951 return;
1952 }
1953
1954 if( colbuf.isTextureAttachment() ) {
1955 final TextureAttachment texA = colbuf.getTextureAttachment();
1956 if( 0 != texA.getName() ) {
1957 gl.glFramebufferTexture2D(GL.GL_FRAMEBUFFER,
1958 GL.GL_COLOR_ATTACHMENT0 + i,
1959 GL.GL_TEXTURE_2D, 0, 0);
1960 gl.glBindTexture(GL.GL_TEXTURE_2D, 0);
1961 }
1962 texA.free(gl);
1963 } else {
1964 final ColorAttachment colA = colbuf.getColorAttachment();
1965 if( 0 != colA.getName() ) {
1966 gl.glFramebufferRenderbuffer(GL.GL_FRAMEBUFFER,
1967 GL.GL_COLOR_ATTACHMENT0 + i,
1968 GL.GL_RENDERBUFFER, 0);
1969 }
1970 colA.free(gl);
1971 }
1972 }
1973 }
1974
1975 /**
1976 *
1977 * @param gl
1978 * @param dispose true if the Colorbuffer shall be disposed
1979 * @param reqAType {@link Type#DEPTH}, {@link Type#DEPTH} or {@link Type#DEPTH_STENCIL}
1980 */
1981 public final void detachRenderbuffer(final GL gl, final Attachment.Type atype, final boolean dispose) throws IllegalArgumentException {
1982 bind(gl);
1983 final RenderAttachment res = detachRenderbufferImpl(gl, atype, dispose ? DetachAction.DISPOSE : DetachAction.NONE);
1984 if(null == res) {
1985 throw IllegalArgumentException("RenderAttachment type "+atype+", not attached, "+this);
1986 }
1987 if(DEBUG_FBO) {
1988 System.err.println("FBObject.detachRenderbuffer.X: [attachmentType "+atype+", dispose "+dispose+"]: "+this);
1989 }
1990 }
1991
1992 public final boolean isDepthStencilPackedFormat() {
1993 final boolean res = null != depth && null != stencil &&
1994 depth.format == stencil.format ;
1995 if(res) {
1996 if(depth.getName() != stencil.getName() ) {
1997 throw InternalError("depth/stencil packed format not sharing: depth "+depth+", stencil "+stencil);
1998 }
1999 if(depth != stencil) {
2000 throw InternalError("depth/stencil packed format not a shared reference: depth "+depth+", stencil "+stencil);
2001 }
2002 }
2003 return res;
2004 }
2005
2006 private final RenderAttachment detachRenderbufferImpl(final GL gl, Attachment.Type atype, final DetachAction detachAction) throws IllegalArgumentException {
2007 switch ( atype ) {
2008 case DEPTH:
2009 case STENCIL:
2010 case DEPTH_STENCIL:
2011 break;
2012 default:
2013 throw IllegalArgumentException("only depth/stencil types allowed, was "+atype+", "+this);
2014 }
2015 if( null == depth && null == stencil ) {
2016 return null; // nop
2017 }
2018 final boolean packed = isDepthStencilPackedFormat();
2019 if( packed ) {
2020 // Note: DEPTH_STENCIL shares buffer w/ depth and stencil
2021 atype = Attachment.Type.DEPTH_STENCIL;
2022 }
2023 final RenderAttachment renderOld;
2024 switch ( atype ) {
2025 case DEPTH:
2026 renderOld = depth;
2027 if( null != renderOld ) {
2028 final int format = renderOld.format;
2029 if( 0 != renderOld.getName() ) {
2030 gl.glFramebufferRenderbuffer(GL.GL_FRAMEBUFFER, GL.GL_DEPTH_ATTACHMENT, GL.GL_RENDERBUFFER, 0);
2031 switch(detachAction) {
2032 case DISPOSE:
2033 case RECREATE:
2034 renderOld.free(gl);
2035 break;
2036 default:
2037 }
2038 }
2039 if(DetachAction.RECREATE == detachAction) {
2040 attachRenderbufferImpl2(gl, atype, format);
2041 } else {
2042 depth = null;
2043 }
2044 }
2045 break;
2046 case STENCIL:
2047 renderOld = stencil;
2048 if( null != renderOld ) {
2049 final int format = renderOld.format;
2050 if(0 != renderOld.getName()) {
2051 gl.glFramebufferRenderbuffer(GL.GL_FRAMEBUFFER, GL.GL_STENCIL_ATTACHMENT, GL.GL_RENDERBUFFER, 0);
2052 switch(detachAction) {
2053 case DISPOSE:
2054 case RECREATE:
2055 renderOld.free(gl);
2056 break;
2057 default:
2058 }
2059 }
2060 if(DetachAction.RECREATE == detachAction) {
2061 attachRenderbufferImpl2(gl, atype, format);
2062 } else {
2063 stencil = null;
2064 }
2065 }
2066 break;
2067 case DEPTH_STENCIL:
2068 renderOld = depth;
2069 if( null != renderOld ) {
2070 final int format = renderOld.format;
2071 if(0 != renderOld.getName()) {
2072 gl.glFramebufferRenderbuffer(GL.GL_FRAMEBUFFER, GL.GL_DEPTH_ATTACHMENT, GL.GL_RENDERBUFFER, 0);
2073 if(packed) {
2074 gl.glFramebufferRenderbuffer(GL.GL_FRAMEBUFFER, GL.GL_STENCIL_ATTACHMENT, GL.GL_RENDERBUFFER, 0);
2075 }
2076 switch(detachAction) {
2077 case DISPOSE:
2078 case RECREATE:
2079 renderOld.free(gl);
2080 break;
2081 default:
2082 }
2083 }
2084 if(DetachAction.RECREATE == detachAction) {
2085 attachRenderbufferImpl2(gl, packed ? Attachment.Type.DEPTH_STENCIL : Attachment.Type.DEPTH, format);
2086 } else {
2087 depth = null;
2088 if(packed) {
2089 stencil = null;
2090 }
2091 }
2092 }
2093 if( !packed && null != stencil ) {
2094 final int format = stencil.format;
2095 if(0 != stencil.getName()) {
2096 gl.glFramebufferRenderbuffer(GL.GL_FRAMEBUFFER, GL.GL_STENCIL_ATTACHMENT, GL.GL_RENDERBUFFER, 0);
2097 switch(detachAction) {
2098 case DISPOSE:
2099 case RECREATE:
2100 stencil.free(gl);
2101 break;
2102 default:
2103 }
2104 }
2105 if(DetachAction.RECREATE == detachAction) {
2106 attachRenderbufferImpl2(gl, Attachment.Type.STENCIL, format);
2107 } else {
2108 stencil = null;
2109 }
2110 }
2111 break;
2112 default:
2113 throw InternalError("XXX"); // handled by caller
2114 }
2115 modified = true;
2116 return renderOld;
2117 }
2118
2119 private final void freeAllRenderbufferImpl(final GL gl) throws IllegalArgumentException {
2120 // Note: DEPTH_STENCIL shares buffer w/ depth and stencil
2121 final boolean packed = isDepthStencilPackedFormat();
2122 if( null != depth ) {
2123 if(0 != depth.getName()) {
2124 gl.glFramebufferRenderbuffer(GL.GL_FRAMEBUFFER, GL.GL_DEPTH_ATTACHMENT, GL.GL_RENDERBUFFER, 0);
2125 if(packed) {
2126 gl.glFramebufferRenderbuffer(GL.GL_FRAMEBUFFER, GL.GL_STENCIL_ATTACHMENT, GL.GL_RENDERBUFFER, 0);
2127 }
2128 depth.free(gl);
2129 }
2130 }
2131 if( !packed && null != stencil ) {
2132 if(0 != stencil.getName()) {
2133 gl.glFramebufferRenderbuffer(GL.GL_FRAMEBUFFER, GL.GL_STENCIL_ATTACHMENT, GL.GL_RENDERBUFFER, 0);
2134 stencil.free(gl);
2135 }
2136 }
2137 }
2138
2139 /**
2140 * Detaches all {@link ColorAttachment}s, {@link TextureAttachment}s and {@link RenderAttachment}s
2141 * and disposes them.
2142 * <p>Leaves the FBO bound, if initialized!</p>
2143 * <p>
2144 * An attached sampling sink texture will be detached as well, see {@link #getSamplingTextureSink()}.
2145 * </p>
2146 * @param gl the current GL context
2147 */
2148 public final void detachAll(final GL gl) {
2149 if(null != samplingSink) {
2150 samplingSink.detachAll(gl);
2151 }
2152 detachAllImpl(gl, true/* detachNonColorbuffer */, false /* recreate */, 0);
2153 }
2154
2155 /**
2156 * Detaches all {@link ColorAttachment}s and {@link TextureAttachment}s
2157 * and disposes them.
2158 * <p>Leaves the FBO bound, if initialized!</p>
2159 * <p>
2160 * An attached sampling sink texture will be detached as well, see {@link #getSamplingTextureSink()}.
2161 * </p>
2162 * @param gl the current GL context
2163 */
2164 public final void detachAllColorbuffer(final GL gl) {
2165 if(null != samplingSink) {
2166 samplingSink.detachAllColorbuffer(gl);
2167 }
2168 detachAllImpl(gl, false/* detachNonColorbuffer */, false /* recreate */, 0);
2169 }
2170
2171 /**
2172 * Detaches all {@link TextureAttachment}s and disposes them.
2173 * <p>Leaves the FBO bound, if initialized!</p>
2174 * <p>
2175 * An attached sampling sink texture will be detached as well, see {@link #getSamplingTextureSink()}.
2176 * </p>
2177 * @param gl the current GL context
2178 */
2179 public final void detachAllTexturebuffer(final GL gl) {
2180 if( !isInitialized() ) {
2181 return;
2182 }
2183 if(null != samplingSink) {
2184 samplingSink.detachAllTexturebuffer(gl);
2185 }
2186 bind(gl);
2187 for(int i=0; i<maxColorAttachments; i++) {
2188 if( colorbufferAttachments[i].isTextureAttachment() ) {
2189 detachColorbufferImpl(gl, i, DetachAction.DISPOSE, 0);
2190 }
2191 }
2192 if(DEBUG_FBO) {
2193 System.err.println("FBObject.detachAllTexturebuffer.X: "+this);
2194 }
2195 }
2196
2197 public final void detachAllRenderbuffer(final GL gl) {
2198 if( !isInitialized() ) {
2199 return;
2200 }
2201 if(null != samplingSink) {
2202 samplingSink.detachAllRenderbuffer(gl);
2203 }
2204 bind(gl);
2205 detachRenderbufferImpl(gl, Attachment.Type.DEPTH_STENCIL, DetachAction.DISPOSE);
2206 }
2207
2208 private final void detachAllImpl(final GL gl, final boolean detachNonColorbuffer, final boolean recreate, final int sampleCountChange) {
2209 if( !isInitialized() ) {
2210 return;
2211 }
2212 ignoreStatus = recreate; // ignore status on single calls only if recreate -> reset
2213 try {
2214 bind(gl);
2215 if(FBOResizeQuirk) {
2216 if(detachNonColorbuffer && recreate) {
2217 // free all colorbuffer & renderbuffer 1st
2218 freeAllColorbufferImpl(gl);
2219 freeAllRenderbufferImpl(gl);
2220 }
2221 }
2222 for(int i=0; i<maxColorAttachments; i++) {
2223 detachColorbufferImpl(gl, i, recreate ? DetachAction.RECREATE : DetachAction.DISPOSE, sampleCountChange);
2224 }
2225 if( !recreate && colorbufferCount>0 ) {
2226 throw InternalError("Non zero ColorAttachments "+this);
2227 }
2228
2229 if(detachNonColorbuffer) {
2230 detachRenderbufferImpl(gl, Attachment.Type.DEPTH_STENCIL, recreate ? DetachAction.RECREATE : DetachAction.DISPOSE);
2231 }
2232 if(ignoreStatus) { // post validate
2233 /* if(true) {
2234 throw GLException("Simulating bug 617, reset FBO failure");
2235 } */
2236 updateStatus(gl);
2237 if(!isStatusValid()) {
2238 throw GLException("detachAllImpl failed: "+getStatusString()+", "+this);
2239 }
2240 }
2241 } finally {
2242 ignoreStatus = false;
2243 }
2244 if(DEBUG_FBO) {
2245 System.err.println("FBObject.detachAll.X: [resetNonColorbuffer "+detachNonColorbuffer+", recreate "+recreate+"]: "+this);
2246 }
2247 }
2248
2249 /**
2250 * @param gl the current GL context
2251 */
2252 public final void destroy(final GL gl) {
2253 if(!initialized) {
2254 return;
2255 }
2256 if(DEBUG_FBO) {
2257 System.err.println("FBObject.destroy.0: "+this);
2258 // Thread.dumpStack();
2259 }
2260 if( null != samplingSink && samplingSink.isInitialized() ) {
2261 samplingSink.destroy(gl);
2262 }
2263
2264 detachAllImpl(gl, true /* detachNonColorbuffer */, false /* recreate */, 0);
2265
2266 // cache FB names, preset exposed to zero,
2267 // braking ties w/ GL/GLContext link to getReadFramebuffer()/getWriteFramebuffer()
2268 final int fb_cache = fbName;
2269 fbName = 0;
2270
2271 final int name[] = new int[1];
2272 if(0!=fb_cache) {
2273 name[0] = fb_cache;
2274 gl.glDeleteFramebuffers(1, name, 0);
2275 }
2276 initialized = false;
2277 bound = false;
2278 if(DEBUG_FBO) {
2279 System.err.println("FBObject.destroy.X: "+this);
2280 }
2281 }
2282
2283 private final boolean sampleSinkSizeMismatch() {
2284 return samplingSink.getWidth() != width || samplingSink.getHeight() != height ;
2285 }
2286 private final boolean sampleSinkDepthStencilMismatch() {
2287 if ( ( null != depth && ( null == samplingSink.depth || depth.format != samplingSink.depth.format ) )
2288 ||
2289 ( null == depth && null != samplingSink.depth )
2290 ) {
2291 return true;
2292 }
2293
2294 if ( ( null != stencil && ( null == samplingSink.stencil || stencil.format != samplingSink.stencil.format ) )
2295 ||
2296 ( null == stencil && null != samplingSink.stencil )
2297 ) {
2298 return true;
2299 }
2300
2301 return false;
2302 }
2303 /**
2304 * For GLES3, sampling-sink {@link Colorbuffer} <i>internal format</i> <b>must be equal</b> w/ the sampling-source {@link Colorbuffer}.
2305 * Implementation aligns w/ {@link #createColorTextureAttachment(GLProfile, boolean, int, int, int, int, int, int)}
2306 * and {@link #createColorAttachment(boolean)}.
2307 */
2308 private final boolean sampleSinkExFormatMismatch(final GL gl) {
2309 if( null != samplingColorSink && getColorbufferCount() > 0 && gl.isGL2ES3() ) {
2310 final Attachment ca = (Attachment)getColorbuffer(0); // should be at attachment-point 0
2311 // We cannot comply w/ attachment's format other than attachment point 0!
2312 // return ( null != ca && ca.format != samplingColorSink.getFormat() ) ||
2313 // hasAlpha(samplingColorSink.getFormat()) != hasAttachmentUsingAlpha();
2314 return null != ca && ca.format != samplingColorSink.getFormat();
2315 }
2316 return false;
2317 }
2318
2319 /**
2320 * Manually validates the MSAA sampling sink, if used.
2321 * <p>
2322 * If MSAA is being used and no sampling sink is attached via {@link #setSamplingSink(FBObject)}
2323 * a new sampling sink is being created.
2324 * </p>
2325 * <p>
2326 * If the sampling sink size or attributes differs from the source, its attachments are reset
2327 * to match the source.
2328 * </p>
2329 * <p>
2330 * Automatically called by {@link #reset(GL, int, int, int, boolean)}
2331 * and {@link #syncSamplingSink(GL)}.
2332 * </p>
2333 * <p>
2334 * It is recommended to call this method after initializing the FBO and attaching renderbuffer etc for the 1st time
2335 * if access to sampling sink resources is required.
2336 * </p>
2337 *
2338 * <p>Leaves the FBO bound state untouched</p>
2339 *
2340 * @param gl the current GL context
2341 * @return {@code true} if this instance has been modified, otherwise {@code false}.
2342 * @throws GLException in case of an error, i.e. size too big, etc ..
2343 */
2344 public final boolean resetSamplingSink(final GL gl) throws GLException {
2345 if(DEBUG_FBO) {
2346 System.err.println("FBObject.resetSamplingSink.0");
2347 ExceptionUtils.dumpStack(System.err);
2348 }
2349
2350 if( 0 == samples ) {
2351 final boolean modifiedInstance;
2352 // MSAA off
2353 if( null != samplingSink ) {
2354 // cleanup
2355 if( samplingSink.initialized ) {
2356 samplingSink.detachAll(gl);
2357 }
2358 samplingSink = null;
2359 samplingColorSink = null;
2360 modifiedInstance = true;
2361 } else {
2362 modifiedInstance = false;
2363 }
2364 this.modified = false;
2365 if(DEBUG_FBO) {
2366 System.err.println("FBObject.resetSamplingSink.X1: zero samples, mod "+modifiedInstance+"\n\tTHIS "+this);
2367 }
2368 return modifiedInstance;
2369 }
2370
2371 boolean modifiedInstance = false;
2372
2373 if( null == samplingSink ) {
2374 samplingSink = new FBObject();
2375 samplingSink.init(gl, width, height, 0);
2376 samplingColorSink = null;
2377 modifiedInstance = true;
2378 } else if( !samplingSink.initialized ) {
2379 throw InternalError("InitState Mismatch: samplingSink set, but not initialized "+samplingSink);
2380 } else if( null == samplingColorSink || 0 == samplingColorSink.getName() ) {
2381 throw InternalError("InitState Mismatch: samplingColorSink set, but not initialized "+samplingColorSink+", "+samplingSink);
2382 }
2383
2384 if(DEBUG_FBO) {
2385 System.err.println("FBObject.resetSamplingSink.1: mod "+modifiedInstance+"\n\tTHIS "+this+",\n\tSINK "+samplingSink);
2386 }
2387 boolean sampleSinkExFormatMismatch = sampleSinkExFormatMismatch(gl);
2388 boolean sampleSinkSizeMismatch = sampleSinkSizeMismatch();
2389 boolean sampleSinkDepthStencilMismatch = sampleSinkDepthStencilMismatch();
2390
2391 if( modifiedInstance ) {
2392 // samplingColorSink == null
2393 // must match size, format and colorbuffer do not exist yet
2394 if( sampleSinkExFormatMismatch || sampleSinkSizeMismatch ) {
2395 throw InternalError("InitState Mismatch: Matching exFormat "+!sampleSinkExFormatMismatch+
2396 ", size "+!sampleSinkSizeMismatch +", "+this);
2397 }
2398 } else {
2399 // samplingColorSink != null
2400 if(!sampleSinkExFormatMismatch && !sampleSinkSizeMismatch && !sampleSinkDepthStencilMismatch) {
2401 if(DEBUG_FBO) {
2402 System.err.println("FBObject.resetSamplingSink.X2: Matching: exFormat "+!sampleSinkExFormatMismatch+
2403 ", size "+!sampleSinkSizeMismatch +", depthStencil "+!sampleSinkDepthStencilMismatch+
2404 ", mod "+modifiedInstance);
2405 }
2406 // all properties match ..
2407 samplingSink.modified = false;
2408 this.modified = false;
2409 return modifiedInstance;
2410 }
2411 }
2412
2413 final boolean wasBound;
2414 if( isBound() ) {
2415 markUnbound(); // automatic GL unbind by sampleSink binding
2416 wasBound = true;
2417 } else {
2418 wasBound = false;
2419 }
2420
2421 if(DEBUG_FBO) {
2422 System.err.println("FBObject.resetSamplingSink.2: wasBound "+wasBound+", matching: exFormat "+!sampleSinkExFormatMismatch+
2423 ", size "+!sampleSinkSizeMismatch +", depthStencil "+!sampleSinkDepthStencilMismatch);
2424 }
2425
2426 modifiedInstance = true;
2427
2428 if( sampleSinkDepthStencilMismatch ) { // includes 1st init
2429 samplingSink.detachAllRenderbuffer(gl);
2430 }
2431
2432 final boolean samplingColorSinkShallBeTA = null == samplingColorSink || samplingColorSink.isTextureAttachment();
2433
2434 if( sampleSinkExFormatMismatch ) {
2435 samplingSink.detachAllColorbuffer(gl);
2436 samplingColorSink = null;
2437 } else if( sampleSinkSizeMismatch ) {
2438 samplingSink.resetSizeImpl(gl, width, height);
2439 samplingColorSink = samplingSink.getColorbuffer(0);
2440 }
2441
2442 if( null == samplingColorSink ) { // sampleSinkFormatMismatch || 1st init
2443 final Colorbuffer cb0 = getColorbuffer(0); // align with colorbuffer at attachment-point 0
2444 if( null != cb0 ) {
2445 // match pre-existing format
2446 if( samplingColorSinkShallBeTA ) {
2447 samplingColorSink = createColorTextureAttachment(gl, cb0.getFormat(), width, height,
2448 GL.GL_NEAREST, GL.GL_NEAREST,
2449 GL.GL_CLAMP_TO_EDGE, GL.GL_CLAMP_TO_EDGE);
2450 } else {
2451 samplingColorSink = createColorAttachment(cb0.getFormat(), 0, width, height);
2452 }
2453 samplingSink.attachColorbuffer(gl, 0, samplingColorSink);
2454 } else {
2455 // match default format
2456 final boolean hasAlpha = hasAttachmentUsingAlpha();
2457 if( samplingColorSinkShallBeTA ) {
2458 samplingColorSink = samplingSink.attachTexture2D(gl, 0, hasAlpha);
2459 } else {
2460 samplingColorSink = samplingSink.attachColorbuffer(gl, 0, hasAlpha);
2461 }
2462 }
2463 }
2464
2465 if( sampleSinkDepthStencilMismatch ) { // includes 1st init
2466 samplingSink.attachRenderbuffer(gl, depth.format);
2467 if( null != stencil && !isDepthStencilPackedFormat() ) {
2468 samplingSink.attachRenderbuffer(gl, stencil.format);
2469 }
2470 }
2471
2472 sampleSinkExFormatMismatch = sampleSinkExFormatMismatch(gl);
2473 sampleSinkSizeMismatch = sampleSinkSizeMismatch();
2474 sampleSinkDepthStencilMismatch = sampleSinkDepthStencilMismatch();
2475 if(sampleSinkExFormatMismatch || sampleSinkSizeMismatch || sampleSinkDepthStencilMismatch) {
2476 throw InternalError("Samples sink mismatch after reset: \n\tTHIS "+this+",\n\t SINK "+samplingSink+
2477 "\n\t Mismatch. Matching: exFormat "+!sampleSinkExFormatMismatch+
2478 ", size "+!sampleSinkSizeMismatch +", depthStencil "+!sampleSinkDepthStencilMismatch);
2479 }
2480
2481 samplingSink.modified = false;
2482 samplingSink.unbind(gl);
2483 this.modified = false;
2484
2485 if(wasBound) {
2486 bind(gl);
2487 }
2488
2489 if(DEBUG_FBO) {
2490 System.err.println("FBObject.resetSamplingSink.XX: END mod "+modifiedInstance+"\n\tTHIS "+this+",\n\tSINK "+samplingSink+
2491 "\n\t Matching: exFormat "+!sampleSinkExFormatMismatch+
2492 ", size "+!sampleSinkSizeMismatch +", depthStencil "+!sampleSinkDepthStencilMismatch);
2493 }
2494 return modifiedInstance;
2495 }
2496
2497 /**
2498 * Setting this FBO sampling sink.
2499 * @param newSamplingSink the new and initialized FBO sampling sink to use, or null to remove current sampling sink
2500 * @return the previous sampling sink or null if none was attached
2501 * @throws GLException if this FBO doesn't use MSAA or the given sink uses MSAA itself
2502 * @throws IllegalStateException if the {@code newSamplingSink} is not null and not initialized
2503 */
2504 public FBObject setSamplingSink(final FBObject newSamplingSink) throws IllegalStateException, GLException {
2505 final FBObject prev = samplingSink;
2506 if( null == newSamplingSink) {
2507 samplingSink = null;
2508 samplingColorSink = null;
2509 } else if( samples > 0 ) {
2510 if( !newSamplingSink.isInitialized() ) {
2511 throw IllegalStateException("SamplingSink not initialized: "+newSamplingSink);
2512 }
2513 if( newSamplingSink.getNumSamples() > 0 ) {
2514 throw GLException("SamplingSink FBO cannot use MSAA itself: "+newSamplingSink);
2515 }
2516 samplingSink = newSamplingSink;
2517 samplingColorSink = newSamplingSink.getColorbuffer(0);
2518 } else {
2519 throw GLException("Setting SamplingSink for non MSAA FBO not allowed: "+this);
2520 }
2521 modified = true;
2522 samplingSinkDirty = true;
2523 return prev;
2524 }
2525
2526 /**
2527 * Bind this FBO, i.e. bind write framebuffer to {@link #getWriteFramebuffer()}.
2528 *
2529 * <p>
2530 * If multisampling is used, it sets the read framebuffer to the sampling sink {@link #getWriteFramebuffer()}.
2531 * </p>
2532 * <p>
2533 * In case you have attached more than one color buffer,
2534 * you may want to setup {@link GL2ES3#glDrawBuffers(int, int[], int)}.
2535 * </p>
2536 * @param gl the current GL context
2537 * @throws GLException
2538 */
2539 public final void bind(final GL gl) throws GLException {
2540 if(!bound || fbName != gl.getBoundFramebuffer(GL.GL_FRAMEBUFFER)) {
2541 checkInitialized();
2542 if( fullFBOSupport ) {
2543 gl.glBindFramebuffer(GL.GL_DRAW_FRAMEBUFFER, getWriteFramebuffer()); // this fb, msaa or normal
2544 gl.glBindFramebuffer(GL.GL_READ_FRAMEBUFFER, getReadFramebuffer()); // msaa: sampling sink, normal: this fb
2545 } else {
2546 gl.glBindFramebuffer(GL.GL_FRAMEBUFFER, getWriteFramebuffer()); // normal: read/write
2547 }
2548 bound = true;
2549 samplingSinkDirty = true;
2550 }
2551 }
2552
2553 /**
2554 * Unbind this FBO, i.e. bind read and write framebuffer to default, see {@link GLBase#getDefaultDrawFramebuffer()}.
2555 *
2556 * <p>If full FBO is supported, sets the read and write framebuffer individually to default, hence not disturbing
2557 * an optional operating MSAA FBO, see {@link GLBase#getDefaultReadFramebuffer()} and {@link GLBase#getDefaultDrawFramebuffer()}</p>
2558 *
2559 * @param gl the current GL context
2560 * @throws GLException
2561 */
2562 public final void unbind(final GL gl) throws GLException {
2563 if(bound) {
2564 if(fullFBOSupport) {
2565 // default read/draw buffers, may utilize GLContext/GLDrawable override of
2566 // GLContext.getDefaultDrawFramebuffer() and GLContext.getDefaultReadFramebuffer()
2567 gl.glBindFramebuffer(GL.GL_DRAW_FRAMEBUFFER, 0);
2568 gl.glBindFramebuffer(GL.GL_READ_FRAMEBUFFER, 0);
2569 } else {
2570 gl.glBindFramebuffer(GL.GL_FRAMEBUFFER, 0); // default draw buffer
2571 }
2572 bound = false;
2573 }
2574 }
2575
2576 /**
2577 * Method simply marks this FBO unbound w/o interfering w/ the bound framebuffer as perfomed by {@link #unbind(GL)}.
2578 * <p>
2579 * Only use this method if a subsequent {@link #unbind(GL)}, {@link #use(GL, TextureAttachment)} or {@link #bind(GL)}
2580 * follows on <i>any</i> FBO.
2581 * </p>
2582 */
2583 public final void markUnbound() {
2584 bound = false;
2585 }
2586
2587 /**
2588 * Returns <code>true</code> if framebuffer object is bound via {@link #bind(GL)}, otherwise <code>false</code>.
2589 * <p>
2590 * Method verifies the bound state via {@link GL#getBoundFramebuffer(int)}.
2591 * </p>
2592 * @param gl the current GL context
2593 */
2594 public final boolean isBound(final GL gl) {
2595 bound = bound && fbName == gl.getBoundFramebuffer(GL.GL_FRAMEBUFFER) ;
2596 return bound;
2597 }
2598
2599 /** Returns <code>true</code> if framebuffer object is bound via {@link #bind(GL)}, otherwise <code>false</code>. */
2600 public final boolean isBound() { return bound; }
2601
2602 /**
2603 * If multisampling is being used and flagged dirty by a previous call of {@link #bind(GL)} or after initialization,
2604 * the msaa-buffers are sampled to it's sink {@link #getSamplingTextureSink()}.
2605 * <p>
2606 * Method also resets the sampling sink configuration via {@link #resetSamplingSink(GL)} if used and required.
2607 * </p>
2608 * <p>
2609 * Method is called automatically by {@link #use(GL, TextureAttachment)}.
2610 * </p>
2611 * <p>
2612 * Method always resets the framebuffer binding to default in the end.
2613 * If full FBO is supported, sets the read and write framebuffer individually to default after sampling, hence not disturbing
2614 * an optional operating MSAA FBO, see {@link GLBase#getDefaultReadFramebuffer()} and {@link GLBase#getDefaultDrawFramebuffer()}
2615 * </p>
2616 * <p>
2617 * In case you use this FBO w/o the {@link GLFBODrawable} and intend to employ {@link GL#glReadPixels(int, int, int, int, int, int, java.nio.Buffer) glReadPixels(..)}
2618 * you may want to call {@link GL#glBindFramebuffer(int, int) glBindFramebuffer}({@link GL2ES3#GL_READ_FRAMEBUFFER}, {@link #getReadFramebuffer()});
2619 * </p>
2620 * <p>Leaves the FBO unbound.</p>
2621 *
2622 * @param gl the current GL context
2623 * @param ta {@link TextureAttachment} to use, prev. attached w/ {@link #attachTexture2D(GL, int, boolean, int, int, int, int) attachTexture2D(..)}
2624 * @throws IllegalArgumentException
2625 */
2626 public final void syncSamplingSink(final GL gl) {
2627 markUnbound();
2628 if(samples>0 && samplingSinkDirty) { // implies fullFBOSupport
2629 samplingSinkDirty = false;
2630 if( isModified() ) {
2632 }
2633 final boolean checkError = DEBUG_FBO || GLContext.DEBUG_FBO_GL;
2634 if( checkError ) {
2635 checkPreGLError(gl);
2636 }
2637 gl.glBindFramebuffer(GL.GL_READ_FRAMEBUFFER, fbName); // read from this MSAA fb
2638 gl.glBindFramebuffer(GL.GL_DRAW_FRAMEBUFFER, samplingSink.getWriteFramebuffer()); // write to sampling sink
2639 ((GL2ES3)gl).glBlitFramebuffer(0, 0, width, height, 0, 0, width, height, // since MSAA is supported, casting to GL2ES3 is OK
2640 GL.GL_COLOR_BUFFER_BIT, GL.GL_NEAREST);
2641 if( checkError ) {
2642 checkNoError(null, gl.glGetError(), "FBObject syncSampleSink"); // throws GLException if error
2643 }
2644 } else {
2645 modified = false;
2646 }
2647 if(fullFBOSupport) {
2648 // default read/draw buffers, may utilize GLContext/GLDrawable override of
2649 // GLContext.getDefaultDrawFramebuffer() and GLContext.getDefaultReadFramebuffer()
2650 gl.glBindFramebuffer(GL.GL_DRAW_FRAMEBUFFER, 0);
2651 gl.glBindFramebuffer(GL.GL_READ_FRAMEBUFFER, 0);
2652 } else {
2653 gl.glBindFramebuffer(GL.GL_FRAMEBUFFER, 0); // default draw buffer
2654 }
2655 }
2656
2657 /**
2658 * {@link #syncSamplingSink(GL) Synchronize the sampling sink} and bind the given {@link TextureAttachment}, if not <code>null</code>.
2659 *
2660 * <p>If using a {@link TextureAttachment} and multiple texture units, ensure you call {@link GL#glActiveTexture(int)} first!</p>
2661 *
2662 * <p>{@link #syncSamplingSink(GL)} is being called</p>
2663 *
2664 * <p>Leaves the FBO unbound!</p>
2665 *
2666 * @param gl the current GL context
2667 * @param ta {@link TextureAttachment} to use, prev. attached w/ {@link #attachTexture2D(GL, int, boolean, int, int, int, int) attachTexture2D(..)},
2668 * may be <code>null</code> in case no {@link TextureAttachment} is used.
2669 * @throws IllegalArgumentException
2670 */
2671 public final void use(final GL gl, final TextureAttachment ta) throws IllegalArgumentException {
2673 if( null != ta ) {
2674 gl.glBindTexture(GL.GL_TEXTURE_2D, ta.getName()); // use it ..
2675 }
2676 }
2677
2678 /**
2679 * Unbind texture, ie bind 'non' texture 0
2680 *
2681 * <p>Leaves the FBO unbound.</p>
2682 */
2683 public final void unuse(final GL gl) {
2684 unbind(gl);
2685 gl.glBindTexture(GL.GL_TEXTURE_2D, 0); // don't use it
2686 }
2687
2688 /** @see GL#hasFullFBOSupport() */
2689 public final boolean hasFullFBOSupport() throws GLException { checkInitialized(); return this.fullFBOSupport; }
2690
2691 /**
2692 * Returns <code>true</code> if renderbuffer accepts internal format {@link GL#GL_RGB8} and {@link GL#GL_RGBA8}, otherwise <code>false</code>.
2693 * @throws GLException if {@link #init(GL)} hasn't been called.
2694 */
2695 public final boolean supportsRGBA8() throws GLException { checkInitialized(); return rgba8Avail; }
2696
2697 /**
2698 * Returns <code>true</code> if {@link GL#GL_DEPTH_COMPONENT16}, {@link GL#GL_DEPTH_COMPONENT24} or {@link GL#GL_DEPTH_COMPONENT32} is supported, otherwise <code>false</code>.
2699 * @param bits 16, 24 or 32 bits
2700 * @throws GLException if {@link #init(GL)} hasn't been called.
2701 */
2702 public final boolean supportsDepth(final int bits) throws GLException {
2703 checkInitialized();
2704 switch(bits) {
2705 case 16: return true;
2706 case 24: return depth24Avail;
2707 case 32: return depth32Avail;
2708 default: return false;
2709 }
2710 }
2711
2712 /**
2713 * Returns <code>true</code> if {@link GL#GL_STENCIL_INDEX1}, {@link GL#GL_STENCIL_INDEX4}, {@link GL#GL_STENCIL_INDEX8} or {@link GL2GL3#GL_STENCIL_INDEX16} is supported, otherwise <code>false</code>.
2714 * @param bits 1, 4, 8 or 16 bits
2715 * @throws GLException if {@link #init(GL)} hasn't been called.
2716 */
2717 public final boolean supportsStencil(final int bits) throws GLException {
2718 checkInitialized();
2719 switch(bits) {
2720 case 1: return stencil01Avail;
2721 case 4: return stencil04Avail;
2722 case 8: return stencil08Avail;
2723 case 16: return stencil16Avail;
2724 default: return false;
2725 }
2726 }
2727
2728 /**
2729 * Returns <code>true</code> if {@link GL#GL_DEPTH24_STENCIL8} is supported, otherwise <code>false</code>.
2730 * @throws GLException if {@link #init(GL)} hasn't been called.
2731 */
2732 public final boolean supportsPackedDepthStencil() throws GLException { checkInitialized(); return packedDepthStencilAvail; }
2733
2734 /**
2735 * Returns the maximum number of colorbuffer attachments.
2736 * @throws GLException if {@link #init(GL)} hasn't been called.
2737 */
2738 public final int getMaxColorAttachments() throws GLException { checkInitialized(); return maxColorAttachments; }
2739
2740 public final int getMaxTextureSize() throws GLException { checkInitialized(); return this.maxTextureSize; }
2741 public final int getMaxRenderbufferSize() throws GLException { checkInitialized(); return this.maxRenderbufferSize; }
2742
2743 /** @see GL#getMaxRenderbufferSamples() */
2744 public final int getMaxSamples() throws GLException { checkInitialized(); return this.maxSamples; }
2745
2746 /**
2747 * Returns <code>true</code> if this instance has been initialized with {@link #reset(GL, int, int)}
2748 * or {@link #reset(GL, int, int, int, boolean)}, otherwise <code>false</code>
2749 */
2750 public final boolean isInitialized() { return initialized; }
2751 /** Returns the width */
2752 public final int getWidth() { return width; }
2753 /** Returns the height */
2754 public final int getHeight() { return height; }
2755 /** Returns the number of samples for multisampling (MSAA). zero if no multisampling is used. */
2756 public final int getNumSamples() { return samples; }
2757 /** Returns the framebuffer name to render to. */
2758 public final int getWriteFramebuffer() { return fbName; }
2759 /** Returns the framebuffer name to read from. Depending on multisampling, this may be a different framebuffer. */
2760 public final int getReadFramebuffer() {
2761 return 0 < samples ? ( null != samplingSink ? samplingSink.getReadFramebuffer() : 0 ) : fbName;
2762 }
2763
2764 public final int getDefaultDrawBuffer() { return GL.GL_COLOR_ATTACHMENT0; }
2765
2766 public final int getDefaultReadBuffer() { return GL.GL_COLOR_ATTACHMENT0; }
2767
2768 /** Return the number of attached {@link Colorbuffer}s */
2769 public final int getColorbufferCount() { return colorbufferCount; }
2770 /** Return the number of attached {@link TextureAttachment}s */
2771 public final int getTextureAttachmentCount() { return textureAttachmentCount; }
2772 /** Return the stencil {@link RenderAttachment} attachment, if exist. Maybe share the same {@link Attachment#getName()} as {@link #getDepthAttachment()}, if packed depth-stencil is being used. */
2773 public final RenderAttachment getStencilAttachment() { return stencil; }
2774 /** Return the depth {@link RenderAttachment} attachment. Maybe share the same {@link Attachment#getName()} as {@link #getStencilAttachment()}, if packed depth-stencil is being used. */
2775 public final RenderAttachment getDepthAttachment() { return depth; }
2776
2777 /** Return the complete multisampling {@link FBObject} sink, if using multisampling. */
2778 public final FBObject getSamplingSinkFBO() { return samplingSink; }
2779
2780 /** Return the multisampling {@link Colorbuffer} sink, if using multisampling. */
2781 public final Colorbuffer getSamplingSink() { return samplingColorSink; }
2782
2783 /**
2784 * Returns <code>true</code> if the multisampling colorbuffer (msaa-buffer)
2785 * has been flagged dirty by a previous call of {@link #bind(GL)},
2786 * otherwise <code>false</code>.
2787 */
2788 public final boolean isSamplingBufferDirty() { return samplingSinkDirty; }
2789
2790 /**
2791 * Returns <code>true</code> if size, sample-count or any attachment of this instance
2792 * or its {@link #getSamplingSink() sampling-sink} has been modified since last {@link #syncSamplingSink(GL) sync},
2793 * {@link #use(GL, TextureAttachment) use}, {@link #reset(GL, int, int, int) reset}
2794 * or {@link #resetSamplingSink(GL) resetSamplingSink}.
2795 * <p>
2796 * Otherwise method returns <code>false</code>.
2797 * </p>
2798 */
2799 public final boolean isModified() { return modified || ( null != samplingSink && samplingSink.modified ); }
2800
2801 int objectHashCode() { return super.hashCode(); }
2802
2803 @Override
2804 public final String toString() {
2805 final String caps = null != colorbufferAttachments ? Arrays.asList(colorbufferAttachments).toString() : null ;
2806 return "FBO[name r/w "+fbName+"/"+getReadFramebuffer()+", init "+initialized+", bound "+bound+", size "+width+"x"+height+
2807 ", samples "+samples+"/"+maxSamples+", modified "+modified+"/"+isModified()+", depth "+depth+", stencil "+stencil+
2808 ", colorbuffer attachments: "+colorbufferCount+"/"+maxColorAttachments+", with "+textureAttachmentCount+" textures"+
2809 ": "+caps+", msaa["+samplingColorSink+", hasSink "+(null != samplingSink)+
2810 ", dirty "+samplingSinkDirty+"], state "+getStatusString()+", obj "+toHexString(objectHashCode())+"]";
2811 }
2812
2813 private final void updateStatus(final GL gl) {
2814 if( 0 == fbName ) {
2815 vStatus = -1;
2816 } else {
2817 vStatus = gl.glCheckFramebufferStatus(GL.GL_FRAMEBUFFER);
2818 }
2819 }
2820 };
2821 /**@}*/
2822
2823} // namespace gamp::render::gl
2824
2825#endif /* GAMP_RENDER_GL_FBOJECT_HPP_ */
2826
#define E_FILE_LINE
Interface abstraction to allow custom definitions of Attachment's storage.
Definition FBObject.hpp:159
virtual void setStorage(const GL &gl, const Attachment &a)=0
Set or create the Attachment's storage after generating its name and binding it to the target.
static class ColorAttachment extends RenderAttachment implements gamp::render::gl::FBObject::Colorbuffer dataType
final boolean isStatusValid()
The status may even be valid if incomplete during transition of attachments.
static final TextureAttachment createColorTextureAttachment(final GL gl, final boolean alpha, final int width, final int height, final int magFilter, final int minFilter, final int wrapS, final int wrapT)
Creates a color TextureAttachment, i.e.
Definition FBObject.hpp:717
final void detachAllRenderbuffer(final GL gl)
final void bind(final GL gl)
Bind this FBO, i.e.
void init(final GL gl, final int newWidth, final int newHeight, final int newSamples)
Initializes this FBO's instance.
static final int REQUESTED_BITS
Request current context drawable's requested depth- or stencil-bits; value {@value}...
static class ColorAttachment extends RenderAttachment implements gamp::render::gl::FBObject::Colorbuffer dataFormat
Texture FBO attachment.
final ColorAttachment attachColorbuffer(final GL gl, final int attachmentPoint, final boolean alpha)
Attaches a newly created and initialized Colorbuffer, i.e.
final ColorAttachment attachColorbuffer(final GL gl, final int attachmentPoint, final int internalFormat)
Attaches a newly created and initialized Colorbuffer, i.e.
static class ColorAttachment extends RenderAttachment implements gamp::render::gl::FBObject::Colorbuffer wrapT
final void unuse(final GL gl)
Unbind texture, ie bind 'non' texture 0.
final boolean isBound()
Returns true if framebuffer object is bound via bind(GL), otherwise false.
final int getTextureAttachmentCount()
Return the number of attached TextureAttachments.
final boolean isDepthStencilPackedFormat()
RenderAttachment(final Type type, final int iFormat, final int samples, final int width, final int height, final int name)
Definition FBObject.hpp:400
final boolean isInitialized()
Returns true if this instance has been initialized with reset(GL, int, int) or reset(GL,...
final void syncSamplingSink(final GL gl)
If multisampling is being used and flagged dirty by a previous call of bind(GL) or after initializati...
final boolean resetSamplingSink(final GL gl)
Manually validates the MSAA sampling sink, if used.
final ColorAttachment getColorAttachment()
Definition FBObject.hpp:640
static final RenderAttachment createRenderAttachment(final Type type, final int internalFormat, final int samples, final int width, final int height)
final boolean isModified()
Returns true if size, sample-count or any attachment of this instance or its sampling-sink has been m...
static class ColorAttachment extends RenderAttachment implements gamp::render::gl::FBObject::Colorbuffer wrapS
final boolean isBound(final GL gl)
Returns true if framebuffer object is bound via bind(GL), otherwise false.
final void detachAllColorbuffer(final GL gl)
Detaches all ColorAttachments and TextureAttachments and disposes them.
final TextureAttachment getTextureAttachment()
Definition FBObject.hpp:638
FBObject setSamplingSink(final FBObject newSamplingSink)
Setting this FBO sampling sink.
final boolean isSamplingBufferDirty()
Returns true if the multisampling colorbuffer (msaa-buffer) has been flagged dirty by a previous call...
final void attachRenderbufferImpl(final GL gl, final Attachment.Type atype, final int internalFormat)
static class ColorAttachment extends RenderAttachment implements gamp::render::gl::FBObject::Colorbuffer magFilter
boolean equals(final Object o)
Definition FBObject.hpp:429
static final int MAXIMUM_BITS
Request maximum bit count for depth- or stencil buffer (depth 32 bits, stencil 16 bits),...
final Colorbuffer getSamplingSink()
Return the multisampling Colorbuffer sink, if using multisampling.
final void markUnbound()
Method simply marks this FBO unbound w/o interfering w/ the bound framebuffer as perfomed by unbind(G...
final Colorbuffer attachColorbuffer(final GL gl, final int attachmentPoint, final Colorbuffer colbuf)
Attaches a Colorbuffer at the given attachment point and initializes it, if not done yet.
final int getMaxColorAttachments()
Returns the maximum number of colorbuffer attachments.
final void attachRenderbuffer(final GL gl, final Attachment.Type atype, final int reqBits)
Attaches one depth, stencil or packed-depth-stencil buffer to this FBO's instance,...
boolean initialize(final GL gl)
Definition FBObject.hpp:451
final int getStatus()
Note that the status may reflect an incomplete state during transition of attachments.
final boolean supportsStencil(final int bits)
Returns true if GL#GL_STENCIL_INDEX1, GL#GL_STENCIL_INDEX4, GL#GL_STENCIL_INDEX8 or GL2GL3#GL_STENCIL...
final boolean reset(final GL gl, int newWidth, int newHeight, int newSamples)
Resets this FBO's instance.
final int getWidth()
Returns the width.
static String toHexString(final int v)
Definition FBObject.hpp:652
final void use(final GL gl, final TextureAttachment ta)
Synchronize the sampling sink and bind the given TextureAttachment, if not null.
final boolean supportsPackedDepthStencil()
Returns true if GL#GL_DEPTH24_STENCIL8 is supported, otherwise false.
final void detachAllTexturebuffer(final GL gl)
Detaches all TextureAttachments and disposes them.
final void setSamples(final int s)
Definition FBObject.hpp:408
final int getWriteFramebuffer()
Returns the framebuffer name to render to.
TextureAttachment(final Type type, final int iFormat, final int width, final int height, final int dataFormat, final int dataType, final int magFilter, final int minFilter, final int wrapS, final int wrapT, final int name)
Definition FBObject.hpp:539
static final int DEFAULT_BITS
Request default bit count for depth- or stencil buffer (depth 24 bits, stencil 8 bits),...
final TextureAttachment attachTexture2D(final GL gl, final int attachmentPoint, final boolean alpha)
Attaches a Colorbuffer, i.e.
static final int CHOSEN_BITS
Request current context drawable's chosen depth- or stencil-bits; value {@value}.
final RenderAttachment getStencilAttachment()
Return the stencil RenderAttachment attachment, if exist.
final int getHeight()
Returns the height.
final RenderAttachment getDepthAttachment()
Return the depth RenderAttachment attachment.
final FBObject getSamplingSinkFBO()
Return the complete multisampling FBObject sink, if using multisampling.
final TextureAttachment attachTexture2D(final GL gl, final int attachmentPoint, final boolean alpha, final int magFilter, final int minFilter, final int wrapS, final int wrapT)
Attaches a Colorbuffer, i.e.
FBObject()
Creates an uninitialized FBObject instance.
Definition FBObject.hpp:953
final boolean supportsRGBA8()
Returns true if renderbuffer accepts internal format GL#GL_RGB8 and GL#GL_RGBA8, otherwise false.
final boolean isTextureAttachment()
Definition FBObject.hpp:636
final int getSamples()
number of samples, or zero for no multisampling
Definition FBObject.hpp:407
final int getColorbufferAttachmentPoint(final Colorbuffer ca)
Finds the passed Colorbuffer within the valid range of attachment points using reference comparison o...
Definition FBObject.hpp:899
static final ColorAttachment createColorAttachment(final int internalFormat, final int samples, final int width, final int height)
Creates a ColorAttachment, selecting the format automatically.
final String getStatusString()
return the getStatus() as a string.
final void formatToGLCapabilities(final GLCapabilities caps)
Writes the internal format of the attachments to the given GLCapabilities object.
static final String getStatusString(final int fbStatus)
final void detachRenderbuffer(final GL gl, final Attachment.Type atype, final boolean dispose)
final boolean hasFullFBOSupport()
static final TextureAttachment createColorTextureAttachment(final GL gl, final int internalFormat, final int width, final int height, final int magFilter, final int minFilter, final int wrapS, final int wrapT)
Definition FBObject.hpp:738
static final TextureAttachment createColorTextureAttachment(final GL gl, final boolean alpha, final int width, final int height)
Creates a color TextureAttachment, i.e.
Definition FBObject.hpp:668
final int getNumSamples()
Returns the number of samples for multisampling (MSAA).
final Colorbuffer detachColorbuffer(final GL gl, final int attachmentPoint, final boolean dispose)
Detaches a Colorbuffer, i.e.
static class ColorAttachment extends RenderAttachment implements gamp::render::gl::FBObject::Colorbuffer minFilter
void free(final GL gl)
Definition FBObject.hpp:490
final ColorAttachment createColorAttachment(final boolean alpha)
Creates a ColorAttachment, selecting the format automatically.
final int getReadFramebuffer()
Returns the framebuffer name to read from.
final void detachAll(final GL gl)
Detaches all ColorAttachments, TextureAttachments and RenderAttachments and disposes them.
final int getMaxRenderbufferSize()
final void attachRenderbuffer(final GL gl, final int internalFormat)
Attaches one depth, stencil or packed-depth-stencil buffer to this FBO's instance,...
final boolean hasAttachmentUsingAlpha()
Returns true if any attached Colorbuffer uses alpha, otherwise false.
Definition FBObject.hpp:931
final Colorbuffer getColorbuffer(final int attachmentPoint)
Return the Colorbuffer attachment at attachmentPoint if it is attached to this FBO,...
Definition FBObject.hpp:885
final Colorbuffer getColorbuffer(final Colorbuffer ca)
Returns the passed Colorbuffer if it is attached to this FBO, otherwise null.
Definition FBObject.hpp:922
final TextureAttachment attachTexture2D(final GL gl, final int attachmentPoint, final int internalFormat, final int dataFormat, final int dataType, final int magFilter, final int minFilter, final int wrapS, final int wrapT)
Attaches a Colorbuffer, i.e.
final boolean supportsDepth(final int bits)
Returns true if GL#GL_DEPTH_COMPONENT16, GL#GL_DEPTH_COMPONENT24 or GL#GL_DEPTH_COMPONENT32 is suppor...
final void unbind(final GL gl)
Unbind this FBO, i.e.
final int getColorbufferCount()
Return the number of attached Colorbuffers.
static final TextureAttachment createColorTextureAttachment(final int internalFormat, final int width, final int height, final int dataFormat, final int dataType, final int magFilter, final int minFilter, final int wrapS, final int wrapT)
Creates a color TextureAttachment, i.e.
Definition FBObject.hpp:766
final void destroy(final GL gl)
Specifies a set of OpenGL capabilities.
OpenGL Rendering Context.
consteval_cxx20 std::string_view name() noexcept
@ null
Denotes a func::null_target_t.
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.