50 constexpr static bool DEBUG_FBO =
true;
51 constexpr static int USER_MAX_TEXTURE_SIZE = 0;
52 constexpr static bool FBOResizeQuirk =
false;
54 enum class DetachAction { NONE, DISPOSE, RECREATE };
57 class ColorAttachment;
65 virtual ~Colorbuffer() =
default;
107 virtual int getFormat() = 0;
116 virtual int getName() = 0;
123 NONE, DEPTH, STENCIL, DEPTH_STENCIL, COLOR, COLOR_TEXTURE, DEPTH_TEXTURE, STENCIL_TEXTURE
130 static Type determine(GLenum format) {
138 case GL_DEPTH_COMPONENT16:
139 case GL_DEPTH_COMPONENT24:
140 case GL_DEPTH_COMPONENT32:
142 case GL_STENCIL_INDEX1:
143 case GL_STENCIL_INDEX4:
144 case GL_STENCIL_INDEX8:
146 case GL_DEPTH24_STENCIL8:
147 return Type.DEPTH_STENCIL;
195 protected Attachment(
final Type type,
final int iFormat,
final int width,
final int height,
final int name) {
197 this.format = iFormat;
199 this.height = height;
207 public void setStorageDefinition(
final StorageDefinition sd) { this.storageDefinition = sd; }
212 protected final void setStorage(
final GL gl) { storageDefinition.
setStorage(
gl,
this); }
219 public final void formatToGLCapabilities(
final GLCapabilities caps,
final boolean rgba8Avail) {
224 _format = rgba8Avail ?
GL.GL_RGBA8 :
GL.GL_RGBA4;
228 _format = rgba8Avail ?
GL.GL_RGB8 :
GL.GL_RGB565;
236 caps.setGreenBits(4);
238 caps.setAlphaBits(4);
242 caps.setGreenBits(5);
244 caps.setAlphaBits(1);
248 caps.setGreenBits(6);
250 caps.setAlphaBits(0);
254 caps.setGreenBits(8);
256 caps.setAlphaBits(0);
260 caps.setGreenBits(8);
262 caps.setAlphaBits(8);
264 case GL.GL_DEPTH_COMPONENT16:
265 caps.setDepthBits(16);
267 case GL.GL_DEPTH_COMPONENT24:
268 caps.setDepthBits(24);
270 case GL.GL_DEPTH_COMPONENT32:
271 caps.setDepthBits(32);
273 case GL.GL_STENCIL_INDEX1:
274 caps.setStencilBits(1);
276 case GL.GL_STENCIL_INDEX4:
277 caps.setStencilBits(4);
279 case GL.GL_STENCIL_INDEX8:
280 caps.setStencilBits(8);
282 case GL.GL_DEPTH24_STENCIL8:
283 caps.setDepthBits(24);
284 caps.setStencilBits(8);
287 throw IllegalArgumentException(
"format invalid: "+
toHexString(format));
292 public final int getFormat() {
return format; }
295 public final int getWidth() {
return width; }
297 public final int getHeight() {
return height; }
298 final void setSize(
final int w,
final int h) { width = w; height =
h; }
301 public final int getName() {
return name; }
302 final void setName(
final int n) { name = n; }
316 public abstract
boolean initialize(
final GL gl)
throws GLException;
328 public abstract
void free(
final GL gl)
throws GLException;
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 &&
355 public int hashCode() {
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;
365 int objectHashCode() {
return super.hashCode(); }
368 public String toString() {
369 return getClass().getSimpleName()+
"[type "+type+
", format "+
toHexString(format)+
", "+width+
"x"+height+
373 public static Type getType(
final int attachmentPoint,
final int maxColorAttachments) {
374 if(
GL.GL_COLOR_ATTACHMENT0 <= attachmentPoint && attachmentPoint <
GL.GL_COLOR_ATTACHMENT0+maxColorAttachments ) {
377 switch(attachmentPoint) {
378 case GL.GL_DEPTH_ATTACHMENT:
380 case GL.GL_STENCIL_ATTACHMENT:
383 throw IllegalArgumentException(
"Invalid attachment point "+
toHexString(attachmentPoint));
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;
410 private static Type validateType(
final Type
type) {
418 throw IllegalArgumentException(
"Invalid type: "+
type);
430 if(
this == o )
return true;
432 return super.equals(o) &&
445 int hash = super.hashCode();
446 hash = ((hash << 5) - hash) + samples;
452 final boolean init = 0 == getName();
454 final boolean checkError = DEBUG_FBO ||
GLContext.DEBUG_FBO_GL;
458 final int[]
name =
new int[] { -1 };
459 gl.glGenRenderbuffers(1,
name, 0);
462 gl.glBindRenderbuffer(
GL.GL_RENDERBUFFER, getName());
465 final int glerr =
gl.glGetError();
466 if(
GL.GL_NO_ERROR != glerr) {
467 gl.glDeleteRenderbuffers(1,
name, 0);
469 throw GLException(
"GL Error "+
toHexString(glerr)+
" while creating "+
this);
473 System.err.println(
"Attachment.init.X: "+
this);
478 private final StorageDefinition defStorageDefinition =
new StorageDefinition() {
480 public void setStorage(
final GL gl,
final Attachment a) {
491 final int[]
name =
new int[] { getName() };
494 System.err.println(
"Attachment.free.0: "+
this);
496 gl.glDeleteRenderbuffers(1,
name, 0);
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);
514 public final boolean isTextureAttachment() {
return false; }
516 public final TextureAttachment getTextureAttachment() {
throw GLException(
"Not a TextureAttachment, but ColorAttachment"); }
518 public final ColorAttachment getColorAttachment() {
return this; }
522 public static class TextureAttachment extends Attachment implements Colorbuffer {
541 super(validateType(
type), iFormat, width, height,
name);
542 this.setStorageDefinition(defStorageDefinition);
551 private static Type validateType(
final Type
type) {
555 case STENCIL_TEXTURE:
558 throw IllegalArgumentException(
"Invalid type: "+
type);
568 final boolean init = 0 == getName();
570 final boolean checkError = DEBUG_FBO ||
GLContext.DEBUG_FBO_GL;
574 final int[]
name =
new int[] { -1 };
575 gl.glGenTextures(1,
name, 0);
577 throw GLException(
"null texture, "+
this);
581 gl.glBindTexture(
GL.GL_TEXTURE_2D,
name[0]);
583 gl.glTexParameteri(
GL.GL_TEXTURE_2D,
GL.GL_TEXTURE_MAG_FILTER,
magFilter);
586 gl.glTexParameteri(
GL.GL_TEXTURE_2D,
GL.GL_TEXTURE_MIN_FILTER,
minFilter);
589 gl.glTexParameteri(
GL.GL_TEXTURE_2D,
GL.GL_TEXTURE_WRAP_S,
wrapS);
592 gl.glTexParameteri(
GL.GL_TEXTURE_2D,
GL.GL_TEXTURE_WRAP_T,
wrapT);
595 boolean preTexImage2D =
true;
596 int glerr =
gl.glGetError();
597 if(
GL.GL_NO_ERROR == glerr) {
598 preTexImage2D =
false;
600 glerr =
gl.glGetError();
602 if(
GL.GL_NO_ERROR != glerr) {
603 gl.glDeleteTextures(1,
name, 0);
605 throw GLException(
"GL Error "+
toHexString(glerr)+
" while creating (pre TexImage2D "+preTexImage2D+
") "+
this);
611 System.err.println(
"Attachment.init.X: "+
this);
616 private final StorageDefinition defStorageDefinition =
new StorageDefinition() {
618 public void setStorage(
final GL gl,
final Attachment a) {
625 final int[]
name =
new int[] { getName() };
628 System.err.println(
"Attachment.free.0: "+
this);
630 gl.glDeleteTextures(1,
name, 0);
640 public final ColorAttachment
getColorAttachment() {
throw GLException(
"Not a ColorAttachment, but TextureAttachment"); }
644 return getClass().getSimpleName()+
"[type "+
type+
", target GL_TEXTURE_2D, level 0, format "+
toHexString(format)+
653 return "0x"+
Integer.toHexString(v);
721 internalFormat = alpha ?
GL.GL_RGBA8 :
GL.GL_RGB8;
724 }
else if(
gl.isGLES()) {
725 internalFormat = alpha ?
GL.GL_RGBA :
GL.GL_RGB;
729 internalFormat = alpha ?
GL.GL_RGBA8 :
GL.GL_RGB8;
733 dataType = alpha ? GL2GL3.GL_UNSIGNED_INT_8_8_8_8_REV :
GL.GL_UNSIGNED_BYTE;
741 final boolean alpha = hasAlpha(internalFormat);
747 dataType = alpha ? GL2GL3.GL_UNSIGNED_INT_8_8_8_8_REV :
GL.GL_UNSIGNED_BYTE;
772 private static boolean hasAlpha(
final int format) {
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;
797 private int width, height, samples;
799 private boolean ignoreStatus;
801 private boolean bound;
803 private int colorbufferCount;
804 private int textureAttachmentCount;
805 private Colorbuffer[] colorbufferAttachments;
807 private boolean modified;
810 private Colorbuffer samplingColorSink;
811 private boolean samplingSinkDirty;
817 private final void validateColorAttachmentPointRange(
final int point) {
819 throw GLException(
"FBO not initialized");
821 if(maxColorAttachments != colorbufferAttachments.length) {
822 throw InternalError(String.format(
"maxColorAttachments %d, array.length %d",
823 maxColorAttachments, colorbufferAttachments.length) );
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() ) );
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() ) );
839 private final void addColorAttachment(
final int point,
final Colorbuffer ca,
final boolean validate) {
840 final Colorbuffer c = colorbufferAttachments[point];
842 validateColorAttachmentPointRange(point);
844 throw IllegalArgumentException(
"Colorbuffer is null");
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() ) );
851 colorbufferAttachments[point] = ca;
853 if( ca.isTextureAttachment() ) {
854 textureAttachmentCount++;
859 private final void removeColorAttachment(
final int point,
final Colorbuffer ca) {
860 validateColorAttachmentPointRange(point);
862 throw IllegalArgumentException(
"Colorbuffer is null");
864 final Colorbuffer c = colorbufferAttachments[point];
866 throw IllegalStateException(String.format(
"Cannot detach %s at %d, slot is holding other: %s, %s",
867 ca.toString(), point, c.toString(),
this.toString() ) );
869 colorbufferAttachments[point] =
null;
871 if( ca.isTextureAttachment() ) {
872 textureAttachmentCount--;
886 validateColorAttachmentPointRange(attachmentPoint);
887 return colorbufferAttachments[attachmentPoint];
900 for(
int i=0; i<colorbufferAttachments.length; i++) {
901 if( colorbufferAttachments[i] == ca ) {
933 boolean hasAlpha =
false;
934 for(
int i=0; i<caCount; i++) {
939 if( hasAlpha(ca.format) ) {
954 this.initialized =
false;
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;
968 this.maxTextureSize = 0;
969 this.maxRenderbufferSize = 0;
975 this.ignoreStatus =
false;
979 this.colorbufferAttachments = null;
980 this.colorbufferCount = 0;
981 this.textureAttachmentCount = 0;
984 this.modified =
true;
986 this.samplingSink = null;
987 this.samplingColorSink = null;
988 this.samplingSinkDirty =
true;
1007 public void init(
final GL gl,
final int newWidth,
final int newHeight,
final int newSamples)
throws IllegalStateException, GLException {
1009 throw IllegalStateException(
"FBO already initialized");
1011 if( !
gl.hasBasicFBOSupport() ) {
1012 throw GLException(
"FBO not supported w/ context: "+
gl.getContext()+
", "+
this);
1014 fullFBOSupport =
gl.hasFullFBOSupport();
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;
1024 packedDepthStencilAvail = fullFBOSupport ||
1025 gl.isExtensionAvailable(GLExtensions.OES_packed_depth_stencil) ||
1026 gl.isExtensionAvailable(GLExtensions.EXT_packed_depth_stencil) ;
1028 final boolean NV_fbo_color_attachments =
gl.isExtensionAvailable(GLExtensions.NV_fbo_color_attachments);
1030 final int val[] =
new int[1];
1032 checkPreGLError(
gl);
1034 int realMaxColorAttachments = 1;
1035 maxColorAttachments = 1;
1036 if( fullFBOSupport || NV_fbo_color_attachments ) {
1039 gl.glGetIntegerv(GL2ES2.GL_MAX_COLOR_ATTACHMENTS, val, 0);
1040 realMaxColorAttachments = 1 <= val[0] ? val[0] : 1;
1041 }
catch (
final GLException gle) { gle.printStackTrace(); }
1043 maxColorAttachments = realMaxColorAttachments <= 8 ? realMaxColorAttachments : 8;
1045 colorbufferAttachments =
new Colorbuffer[maxColorAttachments];
1046 colorbufferCount = 0;
1047 textureAttachmentCount = 0;
1049 maxSamples =
gl.getMaxRenderbufferSamples();
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;
1055 maxTextureSize = _maxTextureSize;
1057 gl.glGetIntegerv(
GL.GL_MAX_RENDERBUFFER_SIZE, val, 0);
1058 maxRenderbufferSize = val[0];
1060 this.width = 0 < newWidth ? newWidth : 1;
1061 this.height = 0 < newHeight ? newHeight : 1;
1062 this.samples = newSamples <= maxSamples ? newSamples : maxSamples;
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());
1084 checkPreGLError(
gl);
1086 if( width > maxRenderbufferSize || height > maxRenderbufferSize ) {
1087 throw GLException(
"Size "+width+
"x"+height+
" exceeds on of the maxima renderbuffer size "+maxRenderbufferSize+
": \n\t"+
this);
1091 samplingSinkDirty =
true;
1094 gl.glGenFramebuffers(1, val, 0);
1097 throw GLException(
"null framebuffer");
1101 gl.glBindFramebuffer(
GL.GL_FRAMEBUFFER, fbName);
1102 checkNoError(
gl,
gl.glGetError(),
"FBObject Init.bindFB");
1103 if(!
gl.glIsFramebuffer(fbName)) {
1104 checkNoError(
gl,
GL.GL_INVALID_VALUE,
"FBObject Init.isFB");
1109 vStatus =
GL.GL_FRAMEBUFFER_INCOMPLETE_MISSING_ATTACHMENT;
1111 System.err.println(
"FBObject.init() END: "+
this);
1112 ExceptionUtils.dumpStack(System.err);
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");
1144 newSamples = newSamples <= maxSamples ? newSamples : maxSamples;
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);
1152 if( newWidth > maxRenderbufferSize || newHeight > maxRenderbufferSize ) {
1153 throw GLException(
"Size "+newWidth+
"x"+newHeight+
" exceeds on of the maxima renderbuffer size "+maxRenderbufferSize+
": \n\t"+
this);
1157 System.err.println(
"FBObject.reset - START - "+width+
"x"+height+
", "+samples+
" -> "+newWidth+
"x"+newHeight+
", "+newSamples+
"; "+
this);
1160 final boolean wasBound =
isBound();
1162 final int sampleCountChange;
1163 if( 0 < samples && 0 < newSamples || 0 == samples && 0 == newSamples ) {
1164 sampleCountChange = 0;
1165 }
else if( 0 == samples && 0 < newSamples ) {
1166 sampleCountChange = 1;
1167 }
else if( 0 < samples && 0 == newSamples ) {
1168 sampleCountChange = -1;
1170 throw IllegalArgumentException(
"Error in sampleCount change: "+samples+
" -> "+newSamples);
1174 samples = newSamples;
1177 samplingSinkDirty =
true;
1179 detachAllImpl(
gl,
true,
true, sampleCountChange);
1187 System.err.println(
"FBObject.reset - END - wasBound, "+wasBound+
", "+
this);
1204 private final void resetSizeImpl(
final GL gl,
final int newWidth,
final int newHeight) {
1206 System.err.println(
"FBObject.resetSize - START - "+width+
"x"+height+
", "+samples+
" -> "+newWidth+
"x"+newHeight);
1209 final int sampleCountChange = 0;
1214 samplingSinkDirty =
true;
1216 detachAllImpl(
gl,
true,
true, sampleCountChange);
1219 System.err.println(
"FBObject.resetSize - END - "+
this);
1223 private void validateAttachmentSize(
final Attachment a) {
1224 final int aWidth = a.getWidth();
1225 final int aHeight = a.getHeight();
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);
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);
1240 caps.setSampleBuffers(samples > 0);
1241 caps.setNumSamples(samples);
1242 caps.setDepthBits(0);
1243 caps.setStencilBits(0);
1247 cb.formatToGLCapabilities(caps, rgba8Avail);
1250 depth.formatToGLCapabilities(caps, rgba8Avail);
1252 if(null != stencil && stencil != depth) {
1253 stencil.formatToGLCapabilities(caps, rgba8Avail);
1276 case GL.GL_FRAMEBUFFER_COMPLETE:
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");
1296 case GL.GL_FRAMEBUFFER_UNSUPPORTED:
1297 return(
"Unsupported FBO format");
1298 case GL2ES3.GL_FRAMEBUFFER_UNDEFINED:
1299 return(
"FBO undefined");
1302 return(
"FBO implementation fault");
1304 return(
"FBO incomplete, implementation ERROR "+
toHexString(fbStatus));
1314 case GL.GL_FRAMEBUFFER_COMPLETE:
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) {
1330 case GL.GL_FRAMEBUFFER_UNSUPPORTED:
1331 case GL2ES3.GL_FRAMEBUFFER_UNDEFINED:
1336 System.err.println(
"Framebuffer " + fbName +
" is incomplete, status = " +
toHexString(vStatus) +
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);
1352 private final boolean checkNoError(
final GL gl,
final int err,
final String exceptionMessage)
throws GLException {
1353 if(
GL.GL_NO_ERROR != err) {
1357 if(null != exceptionMessage) {
1365 private final void checkInitialized() throws GLException {
1367 throw GLException(
"FBO not initialized, call init(GL) first.");
1450 final int internalFormat;
1453 internalFormat = alpha ?
GL.GL_RGBA8 :
GL.GL_RGB8 ;
1455 internalFormat = alpha ?
GL.GL_RGBA4 :
GL.GL_RGB565;
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 );
1495 public final ColorAttachment
attachColorbuffer(
final GL gl,
final int attachmentPoint,
final boolean alpha)
throws GLException {
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);
1545 public final Colorbuffer
attachColorbuffer(
final GL gl,
final int attachmentPoint,
final Colorbuffer colbuf)
throws GLException {
1547 return attachColorbufferImpl(
gl, attachmentPoint, colbuf);
1550 private final Colorbuffer attachColorbufferImpl(
final GL gl,
final int attachmentPoint,
final Colorbuffer colbuf)
throws GLException {
1551 validateAddColorAttachment(attachmentPoint, colbuf);
1552 validateAttachmentSize((Attachment)colbuf);
1554 final boolean initializedColorbuf = colbuf.initialize(
gl);
1555 addColorAttachment(attachmentPoint, colbuf,
false);
1557 if( colbuf.isTextureAttachment() ) {
1560 removeColorAttachment(attachmentPoint, texA);
1561 if( initializedColorbuf ) {
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).");
1568 gl.glFramebufferTexture2D(
GL.GL_FRAMEBUFFER,
1569 GL.GL_COLOR_ATTACHMENT0 + attachmentPoint,
1570 GL.GL_TEXTURE_2D, texA.getName(), 0);
1576 throw GLException(
"attachTexture2D "+texA+
" at "+attachmentPoint+
" failed: "+
getStatusString()+
", "+
this);
1580 final ColorAttachment colA = colbuf.getColorAttachment();
1583 gl.glFramebufferRenderbuffer(
GL.GL_FRAMEBUFFER,
1584 GL.GL_COLOR_ATTACHMENT0 + attachmentPoint,
1585 GL.GL_RENDERBUFFER, colA.getName());
1591 throw GLException(
"attachColorbuffer "+colA+
" at "+attachmentPoint+
" failed: "+
getStatusString()+
", "+
this);
1596 System.err.println(
"FBObject.attachColorbuffer.X: [attachmentPoint "+attachmentPoint+
", colbuf "+colbuf+
"]: "+
this);
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;
1608 return GL.GL_DEPTH_COMPONENT32;
1611 return GL.GL_DEPTH_COMPONENT16;
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;
1621 return GL2GL3.GL_STENCIL_INDEX16;
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;
1629 return GL2GL3.GL_STENCIL_INDEX16;
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;
1639 return GL2GL3.GL_STENCIL_INDEX16;
1642 throw GLException(
"stencil buffer n/a");
1690 public final void attachRenderbuffer(
final GL gl,
final Attachment.Type atype,
final int reqBits)
throws GLException, IllegalArgumentException {
1691 final int reqDepth, reqStencil;
1693 throw IllegalArgumentException(
"reqBits out of range, shall be >= "+
MAXIMUM_BITS);
1698 final GLCapabilitiesImmutable caps =
gl.getContext().getGLDrawable().getChosenGLCapabilities();
1699 reqDepth = caps.getDepthBits();
1700 reqStencil = caps.getStencilBits();
1702 final GLCapabilitiesImmutable caps =
gl.getContext().getGLDrawable().getRequestedGLCapabilities();
1703 reqDepth = caps.getDepthBits();
1704 reqStencil = caps.getStencilBits();
1710 reqStencil = reqBits;
1712 final int internalFormat;
1713 int internalStencilFormat = -1;
1717 internalFormat = getDepthIFormat(reqDepth);
1721 internalFormat = getStencilIFormat(reqStencil);
1725 if( packedDepthStencilAvail ) {
1726 internalFormat =
GL.GL_DEPTH24_STENCIL8;
1728 internalFormat = getDepthIFormat(reqDepth);
1729 internalStencilFormat = getStencilIFormat(reqStencil);
1733 throw IllegalArgumentException(
"only depth/stencil types allowed, was "+atype+
", "+
this);
1737 if(0<=internalStencilFormat) {
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);
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);
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);
1782 attachRenderbufferImpl2(
gl, atype, internalFormat);
1785 private final void attachRenderbufferImpl2(
final GL gl,
final Attachment.Type atype,
final int internalFormat)
throws GLException {
1787 if( Attachment.Type.DEPTH == atype ) {
1791 depth.setSize(width, height);
1792 depth.setSamples(samples);
1794 validateAttachmentSize(depth);
1795 depth.initialize(gl);
1796 }
else if( Attachment.Type.STENCIL == atype ) {
1797 if(null == stencil) {
1800 stencil.setSize(width, height);
1801 stencil.setSamples(samples);
1803 validateAttachmentSize(stencil);
1804 stencil.initialize(gl);
1805 }
else if( Attachment.Type.DEPTH_STENCIL == atype ) {
1807 if(null != stencil) {
1808 throw InternalError(
"XXX: DEPTH_STENCIL, depth was null, stencil not: "+this.
toString());
1812 depth.setSize(width, height);
1813 depth.setSamples(samples);
1815 validateAttachmentSize(depth);
1816 depth.initialize(gl);
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());
1842 System.err.println(
"FBObject.attachRenderbuffer.X: [attachmentType "+atype+
", iformat "+
toHexString(internalFormat)+
"]: "+
this);
1856 public final Colorbuffer
detachColorbuffer(
final GL gl,
final int attachmentPoint,
final boolean dispose)
throws IllegalArgumentException {
1859 final Colorbuffer res = detachColorbufferImpl(
gl, attachmentPoint, dispose ? DetachAction.DISPOSE : DetachAction.NONE, 0);
1861 throw IllegalArgumentException(
"ColorAttachment at "+attachmentPoint+
", not attached, "+
this);
1864 System.err.println(
"FBObject.detachColorbuffer.X: [attachmentPoint "+attachmentPoint+
", dispose "+dispose+
"]: "+res+
", "+
this);
1869 private final Colorbuffer detachColorbufferImpl(
final GL gl,
final int attachmentPoint,
final DetachAction detachAction,
final int sampleCountChange) {
1870 final Colorbuffer colbufOld = colorbufferAttachments[attachmentPoint];
1872 if(null == colbufOld) {
1876 removeColorAttachment(attachmentPoint, colbufOld);
1878 if( colbufOld.isTextureAttachment() ) {
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) {
1893 if(DetachAction.RECREATE == detachAction) {
1894 final Colorbuffer colbufNew;
1895 if( 0 < sampleCountChange ) {
1900 texA.setSize(width, height);
1903 attachColorbufferImpl(gl, attachmentPoint, colbufNew);
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) {
1919 if(DetachAction.RECREATE == detachAction) {
1920 final Colorbuffer colbufNew;
1921 if( 0 <= sampleCountChange || null == samplingColorSink ) {
1925 colA.setSize(width, height);
1926 colA.setSamples(samples);
1930 if( samplingColorSink.isTextureAttachment() ) {
1931 final TextureAttachment samplingTextureSink = samplingColorSink.getTextureAttachment();
1933 samplingTextureSink.dataFormat, samplingTextureSink.dataType,
1934 samplingTextureSink.magFilter, samplingTextureSink.minFilter,
1935 samplingTextureSink.wrapS, samplingTextureSink.wrapT);
1946 private final void freeAllColorbufferImpl(
final GL gl) {
1947 for(
int i=0; i<maxColorAttachments; i++) {
1948 final Colorbuffer colbuf = colorbufferAttachments[i];
1950 if(null == colbuf) {
1954 if( colbuf.isTextureAttachment() ) {
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);
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);
1981 public final void detachRenderbuffer(
final GL gl,
final Attachment.Type atype,
final boolean dispose)
throws IllegalArgumentException {
1983 final RenderAttachment res = detachRenderbufferImpl(
gl, atype, dispose ? DetachAction.DISPOSE : DetachAction.NONE);
1985 throw IllegalArgumentException(
"RenderAttachment type "+atype+
", not attached, "+
this);
1988 System.err.println(
"FBObject.detachRenderbuffer.X: [attachmentType "+atype+
", dispose "+dispose+
"]: "+
this);
1993 final boolean res = null != depth && null != stencil &&
1994 depth.format == stencil.format ;
1996 if(depth.getName() != stencil.getName() ) {
1997 throw InternalError(
"depth/stencil packed format not sharing: depth "+depth+
", stencil "+stencil);
1999 if(depth != stencil) {
2000 throw InternalError(
"depth/stencil packed format not a shared reference: depth "+depth+
", stencil "+stencil);
2006 private final RenderAttachment detachRenderbufferImpl(
final GL gl, Attachment.Type atype,
final DetachAction detachAction)
throws IllegalArgumentException {
2013 throw IllegalArgumentException(
"only depth/stencil types allowed, was "+atype+
", "+
this);
2015 if( null == depth && null == stencil ) {
2021 atype = Attachment.Type.DEPTH_STENCIL;
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) {
2039 if(DetachAction.RECREATE == detachAction) {
2040 attachRenderbufferImpl2(gl, atype, format);
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) {
2060 if(DetachAction.RECREATE == detachAction) {
2061 attachRenderbufferImpl2(gl, atype, format);
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);
2074 gl.glFramebufferRenderbuffer(
GL.GL_FRAMEBUFFER,
GL.GL_STENCIL_ATTACHMENT,
GL.GL_RENDERBUFFER, 0);
2076 switch(detachAction) {
2084 if(DetachAction.RECREATE == detachAction) {
2085 attachRenderbufferImpl2(gl, packed ? Attachment.Type.DEPTH_STENCIL : Attachment.Type.DEPTH, format);
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) {
2105 if(DetachAction.RECREATE == detachAction) {
2106 attachRenderbufferImpl2(gl, Attachment.Type.STENCIL, format);
2113 throw InternalError(
"XXX");
2119 private final void freeAllRenderbufferImpl(
final GL gl)
throws IllegalArgumentException {
2122 if( null != depth ) {
2123 if(0 != depth.getName()) {
2124 gl.glFramebufferRenderbuffer(
GL.GL_FRAMEBUFFER,
GL.GL_DEPTH_ATTACHMENT,
GL.GL_RENDERBUFFER, 0);
2126 gl.glFramebufferRenderbuffer(
GL.GL_FRAMEBUFFER,
GL.GL_STENCIL_ATTACHMENT,
GL.GL_RENDERBUFFER, 0);
2131 if( !packed && null != stencil ) {
2132 if(0 != stencil.getName()) {
2133 gl.glFramebufferRenderbuffer(
GL.GL_FRAMEBUFFER,
GL.GL_STENCIL_ATTACHMENT,
GL.GL_RENDERBUFFER, 0);
2149 if(null != samplingSink) {
2150 samplingSink.detachAll(
gl);
2152 detachAllImpl(
gl,
true,
false , 0);
2165 if(null != samplingSink) {
2166 samplingSink.detachAllColorbuffer(
gl);
2168 detachAllImpl(
gl,
false,
false , 0);
2183 if(null != samplingSink) {
2184 samplingSink.detachAllTexturebuffer(
gl);
2187 for(
int i=0; i<maxColorAttachments; i++) {
2189 detachColorbufferImpl(
gl, i, DetachAction.DISPOSE, 0);
2193 System.err.println(
"FBObject.detachAllTexturebuffer.X: "+
this);
2201 if(null != samplingSink) {
2202 samplingSink.detachAllRenderbuffer(
gl);
2205 detachRenderbufferImpl(
gl, Attachment.Type.DEPTH_STENCIL, DetachAction.DISPOSE);
2208 private final void detachAllImpl(
final GL gl,
final boolean detachNonColorbuffer,
final boolean recreate,
final int sampleCountChange) {
2212 ignoreStatus = recreate;
2215 if(FBOResizeQuirk) {
2216 if(detachNonColorbuffer && recreate) {
2218 freeAllColorbufferImpl(
gl);
2219 freeAllRenderbufferImpl(
gl);
2222 for(
int i=0; i<maxColorAttachments; i++) {
2223 detachColorbufferImpl(gl, i, recreate ? DetachAction.RECREATE : DetachAction.DISPOSE, sampleCountChange);
2225 if( !recreate && colorbufferCount>0 ) {
2226 throw InternalError(
"Non zero ColorAttachments "+
this);
2229 if(detachNonColorbuffer) {
2230 detachRenderbufferImpl(gl, Attachment.Type.DEPTH_STENCIL, recreate ? DetachAction.RECREATE : DetachAction.DISPOSE);
2238 throw GLException(
"detachAllImpl failed: "+
getStatusString()+
", "+
this);
2242 ignoreStatus =
false;
2245 System.err.println(
"FBObject.detachAll.X: [resetNonColorbuffer "+detachNonColorbuffer+
", recreate "+recreate+
"]: "+
this);
2257 System.err.println(
"FBObject.destroy.0: "+
this);
2260 if( null != samplingSink && samplingSink.isInitialized() ) {
2261 samplingSink.destroy(
gl);
2264 detachAllImpl(
gl,
true ,
false , 0);
2268 final int fb_cache = fbName;
2271 final int name[] =
new int[1];
2274 gl.glDeleteFramebuffers(1,
name, 0);
2276 initialized =
false;
2279 System.err.println(
"FBObject.destroy.X: "+
this);
2283 private final boolean sampleSinkSizeMismatch() {
2284 return samplingSink.getWidth() != width || samplingSink.getHeight() != height ;
2286 private final boolean sampleSinkDepthStencilMismatch() {
2287 if ( ( null != depth && ( null == samplingSink.depth || depth.format != samplingSink.depth.format ) )
2289 ( null == depth && null != samplingSink.depth )
2294 if ( ( null != stencil && ( null == samplingSink.stencil || stencil.format != samplingSink.stencil.format ) )
2296 ( null == stencil && null != samplingSink.stencil )
2308 private final boolean sampleSinkExFormatMismatch(
final GL gl) {
2314 return null != ca && ca.format != samplingColorSink.getFormat();
2346 System.err.println(
"FBObject.resetSamplingSink.0");
2347 ExceptionUtils.dumpStack(System.err);
2350 if( 0 == samples ) {
2351 final boolean modifiedInstance;
2353 if( null != samplingSink ) {
2355 if( samplingSink.initialized ) {
2356 samplingSink.detachAll(
gl);
2358 samplingSink = null;
2359 samplingColorSink = null;
2360 modifiedInstance =
true;
2362 modifiedInstance =
false;
2364 this.modified =
false;
2366 System.err.println(
"FBObject.resetSamplingSink.X1: zero samples, mod "+modifiedInstance+
"\n\tTHIS "+
this);
2368 return modifiedInstance;
2371 boolean modifiedInstance =
false;
2373 if( null == samplingSink ) {
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);
2385 System.err.println(
"FBObject.resetSamplingSink.1: mod "+modifiedInstance+
"\n\tTHIS "+
this+
",\n\tSINK "+samplingSink);
2387 boolean sampleSinkExFormatMismatch = sampleSinkExFormatMismatch(
gl);
2388 boolean sampleSinkSizeMismatch = sampleSinkSizeMismatch();
2389 boolean sampleSinkDepthStencilMismatch = sampleSinkDepthStencilMismatch();
2391 if( modifiedInstance ) {
2394 if( sampleSinkExFormatMismatch || sampleSinkSizeMismatch ) {
2395 throw InternalError(
"InitState Mismatch: Matching exFormat "+!sampleSinkExFormatMismatch+
2396 ", size "+!sampleSinkSizeMismatch +
", "+
this);
2400 if(!sampleSinkExFormatMismatch && !sampleSinkSizeMismatch && !sampleSinkDepthStencilMismatch) {
2402 System.err.println(
"FBObject.resetSamplingSink.X2: Matching: exFormat "+!sampleSinkExFormatMismatch+
2403 ", size "+!sampleSinkSizeMismatch +
", depthStencil "+!sampleSinkDepthStencilMismatch+
2404 ", mod "+modifiedInstance);
2407 samplingSink.modified =
false;
2408 this.modified =
false;
2409 return modifiedInstance;
2413 final boolean wasBound;
2422 System.err.println(
"FBObject.resetSamplingSink.2: wasBound "+wasBound+
", matching: exFormat "+!sampleSinkExFormatMismatch+
2423 ", size "+!sampleSinkSizeMismatch +
", depthStencil "+!sampleSinkDepthStencilMismatch);
2426 modifiedInstance =
true;
2428 if( sampleSinkDepthStencilMismatch ) {
2429 samplingSink.detachAllRenderbuffer(
gl);
2432 final boolean samplingColorSinkShallBeTA = null == samplingColorSink || samplingColorSink.isTextureAttachment();
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);
2442 if( null == samplingColorSink ) {
2446 if( samplingColorSinkShallBeTA ) {
2448 GL.GL_NEAREST,
GL.GL_NEAREST,
2449 GL.GL_CLAMP_TO_EDGE,
GL.GL_CLAMP_TO_EDGE);
2453 samplingSink.attachColorbuffer(
gl, 0, samplingColorSink);
2457 if( samplingColorSinkShallBeTA ) {
2458 samplingColorSink = samplingSink.attachTexture2D(
gl, 0, hasAlpha);
2460 samplingColorSink = samplingSink.attachColorbuffer(
gl, 0, hasAlpha);
2465 if( sampleSinkDepthStencilMismatch ) {
2466 samplingSink.attachRenderbuffer(
gl, depth.format);
2468 samplingSink.attachRenderbuffer(
gl, stencil.format);
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);
2481 samplingSink.modified =
false;
2482 samplingSink.unbind(
gl);
2483 this.modified =
false;
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);
2494 return modifiedInstance;
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);
2513 if( newSamplingSink.getNumSamples() > 0 ) {
2514 throw GLException(
"SamplingSink FBO cannot use MSAA itself: "+newSamplingSink);
2516 samplingSink = newSamplingSink;
2517 samplingColorSink = newSamplingSink.getColorbuffer(0);
2519 throw GLException(
"Setting SamplingSink for non MSAA FBO not allowed: "+
this);
2522 samplingSinkDirty =
true;
2539 public final void bind(
final GL gl)
throws GLException {
2540 if(!bound || fbName !=
gl.getBoundFramebuffer(
GL.GL_FRAMEBUFFER)) {
2542 if( fullFBOSupport ) {
2549 samplingSinkDirty =
true;
2564 if(fullFBOSupport) {
2567 gl.glBindFramebuffer(
GL.GL_DRAW_FRAMEBUFFER, 0);
2568 gl.glBindFramebuffer(
GL.GL_READ_FRAMEBUFFER, 0);
2570 gl.glBindFramebuffer(
GL.GL_FRAMEBUFFER, 0);
2595 bound = bound && fbName ==
gl.getBoundFramebuffer(
GL.GL_FRAMEBUFFER) ;
2628 if(samples>0 && samplingSinkDirty) {
2629 samplingSinkDirty =
false;
2633 final boolean checkError = DEBUG_FBO ||
GLContext.DEBUG_FBO_GL;
2635 checkPreGLError(
gl);
2637 gl.glBindFramebuffer(
GL.GL_READ_FRAMEBUFFER, fbName);
2638 gl.glBindFramebuffer(
GL.GL_DRAW_FRAMEBUFFER, samplingSink.getWriteFramebuffer());
2639 ((GL2ES3)
gl).glBlitFramebuffer(0, 0, width, height, 0, 0, width, height,
2640 GL.GL_COLOR_BUFFER_BIT,
GL.GL_NEAREST);
2642 checkNoError(null,
gl.glGetError(),
"FBObject syncSampleSink");
2647 if(fullFBOSupport) {
2650 gl.glBindFramebuffer(
GL.GL_DRAW_FRAMEBUFFER, 0);
2651 gl.glBindFramebuffer(
GL.GL_READ_FRAMEBUFFER, 0);
2653 gl.glBindFramebuffer(
GL.GL_FRAMEBUFFER, 0);
2674 gl.glBindTexture(
GL.GL_TEXTURE_2D, ta.getName());
2685 gl.glBindTexture(
GL.GL_TEXTURE_2D, 0);
2689 public final boolean hasFullFBOSupport() throws GLException { checkInitialized();
return this.fullFBOSupport; }
2695 public final boolean supportsRGBA8() throws GLException { checkInitialized();
return rgba8Avail; }
2705 case 16:
return true;
2706 case 24:
return depth24Avail;
2707 case 32:
return depth32Avail;
2708 default:
return false;
2720 case 1:
return stencil01Avail;
2721 case 4:
return stencil04Avail;
2722 case 8:
return stencil08Avail;
2723 case 16:
return stencil16Avail;
2724 default:
return false;
2740 public final int getMaxTextureSize() throws GLException { checkInitialized();
return this.maxTextureSize; }
2744 public final int getMaxSamples() throws GLException { checkInitialized();
return this.maxSamples; }
2761 return 0 < samples ? ( null != samplingSink ? samplingSink.getReadFramebuffer() : 0 ) : fbName;
2799 public final boolean isModified() {
return modified || ( null != samplingSink && samplingSink.modified ); }
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)+
2813 private final void updateStatus(
final GL gl) {
2817 vStatus =
gl.glCheckFramebufferStatus(
GL.GL_FRAMEBUFFER);