Browse Source

Fixed the rg11b10f canvas format, removed rgb9e5 due to lack of real support, improved love.graphics.hasCanvasFormat to do extra checks

--HG--
branch : Canvas-formats
Alex Szpakowski 11 years ago
parent
commit
4aeba24e94
2 changed files with 59 additions and 20 deletions
  1. 57 19
      src/modules/graphics/opengl/Canvas.cpp
  2. 2 1
      src/modules/graphics/opengl/Canvas.h

+ 57 - 19
src/modules/graphics/opengl/Canvas.cpp

@@ -525,12 +525,6 @@ bool Canvas::loadVolatile()
 		return false;
 	}
 
-	if (!isFormatSupported(format))
-	{
-		status = GL_FRAMEBUFFER_INCOMPLETE_ATTACHMENT;
-		return false;
-	}
-
 	glGenTextures(1, &texture);
 	gl.bindTexture(texture);
 
@@ -1017,14 +1011,11 @@ void Canvas::convertFormat(Canvas::Format format, GLenum &internalformat, GLenum
 		internalformat = GL_RGB10_A2;
 		type = GL_UNSIGNED_INT_10_10_10_2;
 		break;
-	case FORMAT_RGB9E5:
-		internalformat = GL_RGB9_E5;
-		externalformat = GL_RGB;
-		type = GL_UNSIGNED_INT_5_9_9_9_REV;
 	case FORMAT_RG11B10F:
 		internalformat = GL_R11F_G11F_B10F;
 		externalformat = GL_RGB;
 		type = GL_UNSIGNED_INT_10F_11F_11F_REV;
+		break;
 	case FORMAT_RGBA16F:
 		internalformat = GL_RGBA16F;
 		type = GL_FLOAT;
@@ -1052,11 +1043,14 @@ bool Canvas::isMultiCanvasSupported()
 	return gl.getMaxRenderTargets() >= 4;
 }
 
+bool Canvas::supportedFormats[] = {false};
+
 bool Canvas::isFormatSupported(Canvas::Format format)
 {
 	if (!isSupported())
 		return false;
 
+	bool supported = true;
 	format = getSizedFormat(format);
 
 	switch (format)
@@ -1065,23 +1059,68 @@ bool Canvas::isFormatSupported(Canvas::Format format)
 	case FORMAT_RGBA4:
 	case FORMAT_RGB5A1:
 	case FORMAT_RGB10A2:
-		return true;
+		supported = true;
+		break;
 	case FORMAT_RGB565:
-		return GLEE_VERSION_4_2 || GLEE_ARB_ES2_compatibility;
-	case FORMAT_RGB9E5:
-		return GLEE_VERSION_3_0 || GLEE_EXT_texture_shared_exponent;
+		supported = GLEE_VERSION_4_2 || GLEE_ARB_ES2_compatibility;
+		break;
 	case FORMAT_RG11B10F:
-		return GLEE_VERSION_3_0 || (GLEE_ARB_texture_float && GLEE_ARB_color_buffer_float
+		supported = GLEE_VERSION_3_0 || (GLEE_ARB_texture_float && GLEE_ARB_color_buffer_float
 			&& GLEE_EXT_packed_float);
+		break;
 	case FORMAT_RGBA16F:
 	case FORMAT_RGBA32F:
-		return GLEE_VERSION_3_0 || (GLEE_ARB_texture_float && GLEE_ARB_color_buffer_float);
+		supported = GLEE_VERSION_3_0 || (GLEE_ARB_texture_float && GLEE_ARB_color_buffer_float);
+		break;
 	case FORMAT_SRGB:
-		return GLEE_VERSION_3_0 || ((GLEE_ARB_framebuffer_sRGB || GLEE_EXT_framebuffer_sRGB)
+		supported = GLEE_VERSION_3_0 || ((GLEE_ARB_framebuffer_sRGB || GLEE_EXT_framebuffer_sRGB)
 			&& (GLEE_VERSION_2_1 || GLEE_EXT_texture_sRGB));
+		break;
 	default:
-		return false;
+		supported = false;
+		break;
 	}
+
+	if (!supported)
+		return false;
+
+	if (supportedFormats[format])
+		return true;
+
+	// Even though we might have the necessary OpenGL version or extension,
+	// drivers are still allowed to throw FRAMEBUFFER_UNSUPPORTED when attaching
+	// a texture to a FBO whose format the driver doesn't like. So we should
+	// test with an actual FBO.
+
+	GLenum internalformat = GL_RGBA;
+	GLenum externalformat = GL_RGBA;
+	GLenum textype = GL_UNSIGNED_BYTE;
+	convertFormat(format, internalformat, externalformat, textype);
+
+	GLuint texture = 0;
+	glGenTextures(1, &texture);
+	gl.bindTexture(texture);
+
+	Texture::Filter f;
+	f.min = f.mag = Texture::FILTER_NEAREST;
+	gl.setTextureFilter(f);
+
+	Texture::Wrap w;
+	gl.setTextureWrap(w);
+
+	glTexImage2D(GL_TEXTURE_2D, 0, internalformat, 2, 2, 0, externalformat, textype, nullptr);
+
+	GLuint fbo = 0;
+	GLenum status = strategy->createFBO(fbo, texture);
+	strategy->deleteFBO(fbo, 0, 0);
+
+	gl.deleteTexture(texture);
+
+	// Cache the result so we don't do this for every isFormatSupported call.
+	if (status == GL_FRAMEBUFFER_COMPLETE)
+		supportedFormats[format] = true;
+
+	return status == GL_FRAMEBUFFER_COMPLETE;
 }
 
 void Canvas::bindDefaultCanvas()
@@ -1109,7 +1148,6 @@ StringMap<Canvas::Format, Canvas::FORMAT_MAX_ENUM>::Entry Canvas::formatEntries[
 	{"rgb5a1", Canvas::FORMAT_RGB5A1},
 	{"rgb565", Canvas::FORMAT_RGB565},
 	{"rgb10a2", Canvas::FORMAT_RGB10A2},
-	{"rgb9e5", Canvas::FORMAT_RGB9E5},
 	{"rg11b10f", Canvas::FORMAT_RG11B10F},
 	{"rgba16f", Canvas::FORMAT_RGBA16F},
 	{"rgba32f", Canvas::FORMAT_RGBA32F},

+ 2 - 1
src/modules/graphics/opengl/Canvas.h

@@ -50,7 +50,6 @@ public:
 		FORMAT_RGB5A1,
 		FORMAT_RGB565,
 		FORMAT_RGB10A2,
-		FORMAT_RGB9E5,
 		FORMAT_RG11B10F,
 		FORMAT_RGBA16F,
 		FORMAT_RGBA32F,
@@ -158,6 +157,8 @@ private:
 	void setupGrab();
 	void drawv(const Matrix &t, const Vertex *v);
 
+	static bool supportedFormats[FORMAT_MAX_ENUM];
+
 	static StringMap<Format, FORMAT_MAX_ENUM>::Entry formatEntries[];
 	static StringMap<Format, FORMAT_MAX_ENUM> formats;