/* * Copyright 2011 Google Inc. * * Use of this source code is governed by a BSD-style license that can be * found in the LICENSE file. */ #include "gl/GrGLInterface.h" #include "gl/GrGLExtensions.h" #include "gl/GrGLUtil.h" #include <stdio.h> #if GR_GL_PER_GL_FUNC_CALLBACK namespace { void GrGLDefaultInterfaceCallback(const GrGLInterface*) {} } #endif const GrGLInterface* GrGLInterfaceAddTestDebugMarker(const GrGLInterface* interface, GrGLInsertEventMarkerProc insertEventMarkerFn, GrGLPushGroupMarkerProc pushGroupMarkerFn, GrGLPopGroupMarkerProc popGroupMarkerFn) { GrGLInterface* newInterface = GrGLInterface::NewClone(interface); if (!newInterface->fExtensions.has("GL_EXT_debug_marker")) { newInterface->fExtensions.add("GL_EXT_debug_marker"); } newInterface->fFunctions.fInsertEventMarker = insertEventMarkerFn; newInterface->fFunctions.fPushGroupMarker = pushGroupMarkerFn; newInterface->fFunctions.fPopGroupMarker = popGroupMarkerFn; return newInterface; } const GrGLInterface* GrGLInterfaceRemoveNVPR(const GrGLInterface* interface) { GrGLInterface* newInterface = GrGLInterface::NewClone(interface); newInterface->fExtensions.remove("GL_NV_path_rendering"); newInterface->fExtensions.remove("GL_CHROMIUM_path_rendering"); newInterface->fFunctions.fMatrixLoadf = nullptr; newInterface->fFunctions.fMatrixLoadIdentity = nullptr; newInterface->fFunctions.fPathCommands = nullptr; newInterface->fFunctions.fPathParameteri = nullptr; newInterface->fFunctions.fPathParameterf = nullptr; newInterface->fFunctions.fGenPaths = nullptr; newInterface->fFunctions.fDeletePaths = nullptr; newInterface->fFunctions.fIsPath = nullptr; newInterface->fFunctions.fPathStencilFunc = nullptr; newInterface->fFunctions.fStencilFillPath = nullptr; newInterface->fFunctions.fStencilStrokePath = nullptr; newInterface->fFunctions.fStencilFillPathInstanced = nullptr; newInterface->fFunctions.fStencilStrokePathInstanced = nullptr; newInterface->fFunctions.fCoverFillPath = nullptr; newInterface->fFunctions.fCoverStrokePath = nullptr; newInterface->fFunctions.fCoverFillPathInstanced = nullptr; newInterface->fFunctions.fCoverStrokePathInstanced = nullptr; newInterface->fFunctions.fStencilThenCoverFillPath = nullptr; newInterface->fFunctions.fStencilThenCoverStrokePath = nullptr; newInterface->fFunctions.fStencilThenCoverFillPathInstanced = nullptr; newInterface->fFunctions.fStencilThenCoverStrokePathInstanced = nullptr; newInterface->fFunctions.fProgramPathFragmentInputGen = nullptr; newInterface->fFunctions.fBindFragmentInputLocation = nullptr; return newInterface; } GrGLInterface::GrGLInterface() { fStandard = kNone_GrGLStandard; #if GR_GL_PER_GL_FUNC_CALLBACK fCallback = GrGLDefaultInterfaceCallback; fCallbackData = 0; #endif } GrGLInterface* GrGLInterface::NewClone(const GrGLInterface* interface) { SkASSERT(interface); GrGLInterface* clone = new GrGLInterface; clone->fStandard = interface->fStandard; clone->fExtensions = interface->fExtensions; clone->fFunctions = interface->fFunctions; #if GR_GL_PER_GL_FUNC_CALLBACK clone->fCallback = interface->fCallback; clone->fCallbackData = interface->fCallbackData; #endif return clone; } #ifdef SK_DEBUG static int kIsDebug = 1; #else static int kIsDebug = 0; #endif #define RETURN_FALSE_INTERFACE \ if (kIsDebug) { SkDebugf("%s:%d GrGLInterface::validate() failed.\n", __FILE__, __LINE__); } \ return false; bool GrGLInterface::validate() const { if (kNone_GrGLStandard == fStandard) { RETURN_FALSE_INTERFACE } if (!fExtensions.isInitialized()) { RETURN_FALSE_INTERFACE } // functions that are always required if (nullptr == fFunctions.fActiveTexture || nullptr == fFunctions.fAttachShader || nullptr == fFunctions.fBindAttribLocation || nullptr == fFunctions.fBindBuffer || nullptr == fFunctions.fBindTexture || nullptr == fFunctions.fBlendColor || // -> GL >= 1.4 or extension, ES >= 2.0 nullptr == fFunctions.fBlendEquation || // -> GL >= 1.4 or extension, ES >= 2.0 nullptr == fFunctions.fBlendFunc || nullptr == fFunctions.fBufferData || nullptr == fFunctions.fBufferSubData || nullptr == fFunctions.fClear || nullptr == fFunctions.fClearColor || nullptr == fFunctions.fClearStencil || nullptr == fFunctions.fColorMask || nullptr == fFunctions.fCompileShader || nullptr == fFunctions.fCopyTexSubImage2D || nullptr == fFunctions.fCreateProgram || nullptr == fFunctions.fCreateShader || nullptr == fFunctions.fCullFace || nullptr == fFunctions.fDeleteBuffers || nullptr == fFunctions.fDeleteProgram || nullptr == fFunctions.fDeleteShader || nullptr == fFunctions.fDeleteTextures || nullptr == fFunctions.fDepthMask || nullptr == fFunctions.fDisable || nullptr == fFunctions.fDisableVertexAttribArray || nullptr == fFunctions.fDrawArrays || nullptr == fFunctions.fDrawElements || nullptr == fFunctions.fEnable || nullptr == fFunctions.fEnableVertexAttribArray || nullptr == fFunctions.fFrontFace || nullptr == fFunctions.fGenBuffers || nullptr == fFunctions.fGenTextures || nullptr == fFunctions.fGetBufferParameteriv || nullptr == fFunctions.fGenerateMipmap || nullptr == fFunctions.fGetError || nullptr == fFunctions.fGetIntegerv || nullptr == fFunctions.fGetProgramInfoLog || nullptr == fFunctions.fGetProgramiv || nullptr == fFunctions.fGetShaderInfoLog || nullptr == fFunctions.fGetShaderiv || nullptr == fFunctions.fGetString || nullptr == fFunctions.fGetUniformLocation || #if 0 // Not included in Chrome yet nullptr == fFunctions.fIsTexture || #endif nullptr == fFunctions.fLinkProgram || nullptr == fFunctions.fLineWidth || nullptr == fFunctions.fPixelStorei || nullptr == fFunctions.fReadPixels || nullptr == fFunctions.fScissor || nullptr == fFunctions.fShaderSource || nullptr == fFunctions.fStencilFunc || nullptr == fFunctions.fStencilMask || nullptr == fFunctions.fStencilOp || nullptr == fFunctions.fTexImage2D || nullptr == fFunctions.fTexParameteri || nullptr == fFunctions.fTexParameteriv || nullptr == fFunctions.fTexSubImage2D || nullptr == fFunctions.fUniform1f || nullptr == fFunctions.fUniform1i || nullptr == fFunctions.fUniform1fv || nullptr == fFunctions.fUniform1iv || nullptr == fFunctions.fUniform2f || nullptr == fFunctions.fUniform2i || nullptr == fFunctions.fUniform2fv || nullptr == fFunctions.fUniform2iv || nullptr == fFunctions.fUniform3f || nullptr == fFunctions.fUniform3i || nullptr == fFunctions.fUniform3fv || nullptr == fFunctions.fUniform3iv || nullptr == fFunctions.fUniform4f || nullptr == fFunctions.fUniform4i || nullptr == fFunctions.fUniform4fv || nullptr == fFunctions.fUniform4iv || nullptr == fFunctions.fUniformMatrix2fv || nullptr == fFunctions.fUniformMatrix3fv || nullptr == fFunctions.fUniformMatrix4fv || nullptr == fFunctions.fUseProgram || nullptr == fFunctions.fVertexAttrib1f || nullptr == fFunctions.fVertexAttrib2fv || nullptr == fFunctions.fVertexAttrib3fv || nullptr == fFunctions.fVertexAttrib4fv || nullptr == fFunctions.fVertexAttribPointer || nullptr == fFunctions.fViewport || nullptr == fFunctions.fBindFramebuffer || nullptr == fFunctions.fBindRenderbuffer || nullptr == fFunctions.fCheckFramebufferStatus || nullptr == fFunctions.fDeleteFramebuffers || nullptr == fFunctions.fDeleteRenderbuffers || nullptr == fFunctions.fFinish || nullptr == fFunctions.fFlush || nullptr == fFunctions.fFramebufferRenderbuffer || nullptr == fFunctions.fFramebufferTexture2D || nullptr == fFunctions.fGetFramebufferAttachmentParameteriv || nullptr == fFunctions.fGetRenderbufferParameteriv || nullptr == fFunctions.fGenFramebuffers || nullptr == fFunctions.fGenRenderbuffers || nullptr == fFunctions.fRenderbufferStorage) { RETURN_FALSE_INTERFACE } GrGLVersion glVer = GrGLGetVersion(this); if (GR_GL_INVALID_VER == glVer) { RETURN_FALSE_INTERFACE } // TODO: Remove this once command buffer implements full ES3. bool ALLOW_MISSING_FUNCTIONS_FOR_INCOMPLETE_COMMAND_BUFFER_ES3 = false; if (kGLES_GrGLStandard == fStandard && glVer >= GR_GL_VER(3,0)) { const GrGLubyte* rendererUByte; GR_GL_CALL_RET(this, rendererUByte, GetString(GR_GL_RENDERER)); const char* renderer = reinterpret_cast<const char*>(rendererUByte); ALLOW_MISSING_FUNCTIONS_FOR_INCOMPLETE_COMMAND_BUFFER_ES3 = 0 == strcmp(renderer, "Chromium"); } // Now check that baseline ES/Desktop fns not covered above are present // and that we have fn pointers for any advertised fExtensions that we will // try to use. // these functions are part of ES2, we assume they are available // On the desktop we assume they are available if the extension // is present or GL version is high enough. if (kGLES_GrGLStandard == fStandard) { if (nullptr == fFunctions.fStencilFuncSeparate || nullptr == fFunctions.fStencilMaskSeparate || nullptr == fFunctions.fStencilOpSeparate) { RETURN_FALSE_INTERFACE } } else if (kGL_GrGLStandard == fStandard) { if (glVer >= GR_GL_VER(2,0)) { if (nullptr == fFunctions.fStencilFuncSeparate || nullptr == fFunctions.fStencilMaskSeparate || nullptr == fFunctions.fStencilOpSeparate) { RETURN_FALSE_INTERFACE } } if (glVer >= GR_GL_VER(3,0) && nullptr == fFunctions.fBindFragDataLocation) { RETURN_FALSE_INTERFACE } if (glVer >= GR_GL_VER(2,0) || fExtensions.has("GL_ARB_draw_buffers")) { if (nullptr == fFunctions.fDrawBuffers) { RETURN_FALSE_INTERFACE } } if (glVer >= GR_GL_VER(1,5) || fExtensions.has("GL_ARB_occlusion_query")) { if (nullptr == fFunctions.fGenQueries || nullptr == fFunctions.fDeleteQueries || nullptr == fFunctions.fBeginQuery || nullptr == fFunctions.fEndQuery || nullptr == fFunctions.fGetQueryiv || nullptr == fFunctions.fGetQueryObjectiv || nullptr == fFunctions.fGetQueryObjectuiv) { RETURN_FALSE_INTERFACE } } if (glVer >= GR_GL_VER(3,3) || fExtensions.has("GL_ARB_timer_query") || fExtensions.has("GL_EXT_timer_query")) { if (nullptr == fFunctions.fGetQueryObjecti64v || nullptr == fFunctions.fGetQueryObjectui64v) { RETURN_FALSE_INTERFACE } } if (glVer >= GR_GL_VER(3,3) || fExtensions.has("GL_ARB_timer_query")) { if (nullptr == fFunctions.fQueryCounter) { RETURN_FALSE_INTERFACE } } } // optional function on desktop before 1.3 if (kGL_GrGLStandard != fStandard || (glVer >= GR_GL_VER(1,3)) || fExtensions.has("GL_ARB_texture_compression")) { if (nullptr == fFunctions.fCompressedTexImage2D #if 0 || nullptr == fFunctions.fCompressedTexSubImage2D #endif ) { RETURN_FALSE_INTERFACE } } // part of desktop GL, but not ES if (kGL_GrGLStandard == fStandard && (nullptr == fFunctions.fGetTexLevelParameteriv || nullptr == fFunctions.fDrawBuffer || nullptr == fFunctions.fReadBuffer)) { RETURN_FALSE_INTERFACE } // GL_EXT_texture_storage is part of desktop 4.2 // There is a desktop ARB extension and an ES+desktop EXT extension if (kGL_GrGLStandard == fStandard) { if (glVer >= GR_GL_VER(4,2) || fExtensions.has("GL_ARB_texture_storage") || fExtensions.has("GL_EXT_texture_storage")) { if (nullptr == fFunctions.fTexStorage2D) { RETURN_FALSE_INTERFACE } } } else if (glVer >= GR_GL_VER(3,0) || fExtensions.has("GL_EXT_texture_storage")) { if (!ALLOW_MISSING_FUNCTIONS_FOR_INCOMPLETE_COMMAND_BUFFER_ES3) { if (nullptr == fFunctions.fTexStorage2D) { RETURN_FALSE_INTERFACE } } } // glTextureBarrier is part of desktop 4.5. There are also ARB and NV extensions. if (kGL_GrGLStandard == fStandard) { if (glVer >= GR_GL_VER(4,5) || fExtensions.has("GL_ARB_texture_barrier") || fExtensions.has("GL_NV_texture_barrier")) { if (nullptr == fFunctions.fTextureBarrier) { RETURN_FALSE_INTERFACE } } } else if (fExtensions.has("GL_NV_texture_barrier")) { if (nullptr == fFunctions.fTextureBarrier) { RETURN_FALSE_INTERFACE } } if (fExtensions.has("GL_KHR_blend_equation_advanced") || fExtensions.has("GL_NV_blend_equation_advanced")) { if (nullptr == fFunctions.fBlendBarrier) { RETURN_FALSE_INTERFACE } } if (fExtensions.has("GL_EXT_discard_framebuffer")) { // FIXME: Remove this once Chromium is updated to provide this function #if 0 if (nullptr == fFunctions.fDiscardFramebuffer) { RETURN_FALSE_INTERFACE } #endif } // FBO MSAA if (kGL_GrGLStandard == fStandard) { // GL 3.0 and the ARB extension have multisample + blit if (glVer >= GR_GL_VER(3,0) || fExtensions.has("GL_ARB_framebuffer_object")) { if (nullptr == fFunctions.fRenderbufferStorageMultisample || nullptr == fFunctions.fBlitFramebuffer) { RETURN_FALSE_INTERFACE } } else { if (fExtensions.has("GL_EXT_framebuffer_blit") && nullptr == fFunctions.fBlitFramebuffer) { RETURN_FALSE_INTERFACE } if (fExtensions.has("GL_EXT_framebuffer_multisample") && nullptr == fFunctions.fRenderbufferStorageMultisample) { RETURN_FALSE_INTERFACE } } } else { if (glVer >= GR_GL_VER(3,0) || fExtensions.has("GL_CHROMIUM_framebuffer_multisample")) { if (nullptr == fFunctions.fRenderbufferStorageMultisample || nullptr == fFunctions.fBlitFramebuffer) { RETURN_FALSE_INTERFACE } } if (fExtensions.has("GL_APPLE_framebuffer_multisample")) { if (nullptr == fFunctions.fRenderbufferStorageMultisampleES2APPLE || nullptr == fFunctions.fResolveMultisampleFramebuffer) { RETURN_FALSE_INTERFACE } } if (fExtensions.has("GL_IMG_multisampled_render_to_texture") || fExtensions.has("GL_EXT_multisampled_render_to_texture")) { if (nullptr == fFunctions.fRenderbufferStorageMultisampleES2EXT || nullptr == fFunctions.fFramebufferTexture2DMultisample) { RETURN_FALSE_INTERFACE } } } // On ES buffer mapping is an extension. On Desktop // buffer mapping was part of original VBO extension // which we require. if (kGL_GrGLStandard == fStandard || fExtensions.has("GL_OES_mapbuffer")) { if (nullptr == fFunctions.fMapBuffer || nullptr == fFunctions.fUnmapBuffer) { RETURN_FALSE_INTERFACE } } // Dual source blending if (kGL_GrGLStandard == fStandard) { if (glVer >= GR_GL_VER(3,3) || fExtensions.has("GL_ARB_blend_func_extended")) { if (nullptr == fFunctions.fBindFragDataLocationIndexed) { RETURN_FALSE_INTERFACE } } } else { if (glVer >= GR_GL_VER(3,0) && fExtensions.has("GL_EXT_blend_func_extended")) { if (nullptr == fFunctions.fBindFragDataLocation || nullptr == fFunctions.fBindFragDataLocationIndexed) { RETURN_FALSE_INTERFACE } } } // glGetStringi was added in version 3.0 of both desktop and ES. if (glVer >= GR_GL_VER(3, 0)) { if (nullptr == fFunctions.fGetStringi) { RETURN_FALSE_INTERFACE } } // glVertexAttribIPointer was added in version 3.0 of both desktop and ES. if (glVer >= GR_GL_VER(3, 0)) { if (NULL == fFunctions.fVertexAttribIPointer) { RETURN_FALSE_INTERFACE } } if (kGL_GrGLStandard == fStandard) { if (glVer >= GR_GL_VER(3, 0) || fExtensions.has("GL_ARB_vertex_array_object")) { if (nullptr == fFunctions.fBindVertexArray || nullptr == fFunctions.fDeleteVertexArrays || nullptr == fFunctions.fGenVertexArrays) { RETURN_FALSE_INTERFACE } } } else { if (glVer >= GR_GL_VER(3,0) || fExtensions.has("GL_OES_vertex_array_object")) { if (nullptr == fFunctions.fBindVertexArray || nullptr == fFunctions.fDeleteVertexArrays || nullptr == fFunctions.fGenVertexArrays) { RETURN_FALSE_INTERFACE } } } if (fExtensions.has("GL_EXT_debug_marker")) { if (nullptr == fFunctions.fInsertEventMarker || nullptr == fFunctions.fPushGroupMarker || nullptr == fFunctions.fPopGroupMarker) { RETURN_FALSE_INTERFACE } } if ((kGL_GrGLStandard == fStandard && glVer >= GR_GL_VER(4,3)) || fExtensions.has("GL_ARB_invalidate_subdata")) { if (nullptr == fFunctions.fInvalidateBufferData || nullptr == fFunctions.fInvalidateBufferSubData || nullptr == fFunctions.fInvalidateFramebuffer || nullptr == fFunctions.fInvalidateSubFramebuffer || nullptr == fFunctions.fInvalidateTexImage || nullptr == fFunctions.fInvalidateTexSubImage) { RETURN_FALSE_INTERFACE; } } else if (kGLES_GrGLStandard == fStandard && glVer >= GR_GL_VER(3,0)) { // ES 3.0 adds the framebuffer functions but not the others. if (nullptr == fFunctions.fInvalidateFramebuffer || nullptr == fFunctions.fInvalidateSubFramebuffer) { RETURN_FALSE_INTERFACE; } } if (kGLES_GrGLStandard == fStandard && fExtensions.has("GL_CHROMIUM_map_sub")) { if (nullptr == fFunctions.fMapBufferSubData || nullptr == fFunctions.fMapTexSubImage2D || nullptr == fFunctions.fUnmapBufferSubData || nullptr == fFunctions.fUnmapTexSubImage2D) { RETURN_FALSE_INTERFACE; } } // These functions are added to the 3.0 version of both GLES and GL. if (glVer >= GR_GL_VER(3,0) || (kGLES_GrGLStandard == fStandard && fExtensions.has("GL_EXT_map_buffer_range")) || (kGL_GrGLStandard == fStandard && fExtensions.has("GL_ARB_map_buffer_range"))) { if (!ALLOW_MISSING_FUNCTIONS_FOR_INCOMPLETE_COMMAND_BUFFER_ES3) { if (nullptr == fFunctions.fMapBufferRange || nullptr == fFunctions.fFlushMappedBufferRange) { RETURN_FALSE_INTERFACE; } } } if ((kGL_GrGLStandard == fStandard && (glVer >= GR_GL_VER(4,3) || fExtensions.has("GL_ARB_program_interface_query"))) || (kGLES_GrGLStandard == fStandard && glVer >= GR_GL_VER(3,1))) { if (nullptr == fFunctions.fGetProgramResourceLocation) { RETURN_FALSE_INTERFACE } } if (kGLES_GrGLStandard == fStandard || glVer >= GR_GL_VER(4,1) || fExtensions.has("GL_ARB_ES2_compatibility")) { #if 0 // Enable this once Chrome gives us the function ptr if (nullptr == fFunctions.fGetShaderPrecisionFormat) { RETURN_FALSE_INTERFACE } #endif } if (fExtensions.has("GL_NV_path_rendering") || fExtensions.has("GL_CHROMIUM_path_rendering")) { if (nullptr == fFunctions.fMatrixLoadf || nullptr == fFunctions.fMatrixLoadIdentity || nullptr == fFunctions.fPathCommands || nullptr == fFunctions.fPathParameteri || nullptr == fFunctions.fPathParameterf || nullptr == fFunctions.fGenPaths || nullptr == fFunctions.fDeletePaths || nullptr == fFunctions.fIsPath || nullptr == fFunctions.fPathStencilFunc || nullptr == fFunctions.fStencilFillPath || nullptr == fFunctions.fStencilStrokePath || nullptr == fFunctions.fStencilFillPathInstanced || nullptr == fFunctions.fStencilStrokePathInstanced || nullptr == fFunctions.fCoverFillPath || nullptr == fFunctions.fCoverStrokePath || nullptr == fFunctions.fCoverFillPathInstanced || nullptr == fFunctions.fCoverStrokePathInstanced #if 0 // List of functions that Skia uses, but which have been added since the initial release // of NV_path_rendering driver. We do not want to fail interface validation due to // missing features, we will just not use the extension. // Update this list -> update GrGLCaps::hasPathRenderingSupport too. || nullptr == fFunctions.fStencilThenCoverFillPath || nullptr == fFunctions.fStencilThenCoverStrokePath || nullptr == fFunctions.fStencilThenCoverFillPathInstanced || nullptr == fFunctions.fStencilThenCoverStrokePathInstanced || nullptr == fFunctions.fProgramPathFragmentInputGen #endif ) { RETURN_FALSE_INTERFACE } if (fExtensions.has("GL_CHROMIUM_path_rendering")) { if (nullptr == fFunctions.fBindFragmentInputLocation) { RETURN_FALSE_INTERFACE } } } if (fExtensions.has("GL_EXT_raster_multisample")) { if (nullptr == fFunctions.fRasterSamples) { RETURN_FALSE_INTERFACE } } if (fExtensions.has("GL_NV_framebuffer_mixed_samples") || fExtensions.has("GL_CHROMIUM_framebuffer_mixed_samples")) { if (nullptr == fFunctions.fCoverageModulation) { RETURN_FALSE_INTERFACE } } if (kGL_GrGLStandard == fStandard) { if (glVer >= GR_GL_VER(3,1) || fExtensions.has("GL_EXT_draw_instanced") || fExtensions.has("GL_ARB_draw_instanced")) { if (nullptr == fFunctions.fDrawArraysInstanced || nullptr == fFunctions.fDrawElementsInstanced) { RETURN_FALSE_INTERFACE } } } else if (kGLES_GrGLStandard == fStandard) { if (!ALLOW_MISSING_FUNCTIONS_FOR_INCOMPLETE_COMMAND_BUFFER_ES3) { if (glVer >= GR_GL_VER(3,0) || fExtensions.has("GL_EXT_draw_instanced")) { if (nullptr == fFunctions.fDrawArraysInstanced || nullptr == fFunctions.fDrawElementsInstanced) { RETURN_FALSE_INTERFACE } } } } if (kGL_GrGLStandard == fStandard) { if (glVer >= GR_GL_VER(3,2) || fExtensions.has("GL_ARB_instanced_arrays")) { if (nullptr == fFunctions.fVertexAttribDivisor) { RETURN_FALSE_INTERFACE } } } else if (kGLES_GrGLStandard == fStandard) { if (!ALLOW_MISSING_FUNCTIONS_FOR_INCOMPLETE_COMMAND_BUFFER_ES3) { if (glVer >= GR_GL_VER(3,0) || fExtensions.has("GL_EXT_instanced_arrays")) { if (nullptr == fFunctions.fVertexAttribDivisor) { RETURN_FALSE_INTERFACE } } } } if ((kGL_GrGLStandard == fStandard && glVer >= GR_GL_VER(4,3)) || (kGLES_GrGLStandard == fStandard && glVer >= GR_GL_VER(3,1))) { if (NULL == fFunctions.fDrawArraysIndirect || NULL == fFunctions.fDrawElementsIndirect) { RETURN_FALSE_INTERFACE } } if ((kGL_GrGLStandard == fStandard && glVer >= GR_GL_VER(4,3)) || (kGLES_GrGLStandard == fStandard && fExtensions.has("GL_EXT_multi_draw_indirect"))) { if (NULL == fFunctions.fMultiDrawArraysIndirect || NULL == fFunctions.fMultiDrawElementsIndirect) { RETURN_FALSE_INTERFACE } } if (fExtensions.has("GL_NV_bindless_texture")) { if (nullptr == fFunctions.fGetTextureHandle || nullptr == fFunctions.fGetTextureSamplerHandle || nullptr == fFunctions.fMakeTextureHandleResident || nullptr == fFunctions.fMakeTextureHandleNonResident || nullptr == fFunctions.fGetImageHandle || nullptr == fFunctions.fMakeImageHandleResident || nullptr == fFunctions.fMakeImageHandleNonResident || nullptr == fFunctions.fIsTextureHandleResident || nullptr == fFunctions.fIsImageHandleResident || nullptr == fFunctions.fUniformHandleui64 || nullptr == fFunctions.fUniformHandleui64v || nullptr == fFunctions.fProgramUniformHandleui64 || nullptr == fFunctions.fProgramUniformHandleui64v) { RETURN_FALSE_INTERFACE } } if (kGL_GrGLStandard == fStandard && fExtensions.has("GL_EXT_direct_state_access")) { if (nullptr == fFunctions.fTextureParameteri || nullptr == fFunctions.fTextureParameteriv || nullptr == fFunctions.fTextureParameterf || nullptr == fFunctions.fTextureParameterfv || nullptr == fFunctions.fTextureImage1D || nullptr == fFunctions.fTextureImage2D || nullptr == fFunctions.fTextureSubImage1D || nullptr == fFunctions.fTextureSubImage2D || nullptr == fFunctions.fCopyTextureImage1D || nullptr == fFunctions.fCopyTextureImage2D || nullptr == fFunctions.fCopyTextureSubImage1D || nullptr == fFunctions.fCopyTextureSubImage2D || nullptr == fFunctions.fGetTextureImage || nullptr == fFunctions.fGetTextureParameterfv || nullptr == fFunctions.fGetTextureParameteriv || nullptr == fFunctions.fGetTextureLevelParameterfv || nullptr == fFunctions.fGetTextureLevelParameteriv) { RETURN_FALSE_INTERFACE } if (glVer >= GR_GL_VER(1,2)) { if (nullptr == fFunctions.fTextureImage3D || nullptr == fFunctions.fTextureSubImage3D || nullptr == fFunctions.fCopyTextureSubImage3D || nullptr == fFunctions.fCompressedTextureImage3D || nullptr == fFunctions.fCompressedTextureImage2D || nullptr == fFunctions.fCompressedTextureImage1D || nullptr == fFunctions.fCompressedTextureSubImage3D || nullptr == fFunctions.fCompressedTextureSubImage2D || nullptr == fFunctions.fCompressedTextureSubImage1D || nullptr == fFunctions.fGetCompressedTextureImage) { RETURN_FALSE_INTERFACE } } if (glVer >= GR_GL_VER(1,5)) { if (nullptr == fFunctions.fNamedBufferData || nullptr == fFunctions.fNamedBufferSubData || nullptr == fFunctions.fMapNamedBuffer || nullptr == fFunctions.fUnmapNamedBuffer || nullptr == fFunctions.fGetNamedBufferParameteriv || nullptr == fFunctions.fGetNamedBufferPointerv || nullptr == fFunctions.fGetNamedBufferSubData) { RETURN_FALSE_INTERFACE } } if (glVer >= GR_GL_VER(2,0)) { if (nullptr == fFunctions.fProgramUniform1f || nullptr == fFunctions.fProgramUniform2f || nullptr == fFunctions.fProgramUniform3f || nullptr == fFunctions.fProgramUniform4f || nullptr == fFunctions.fProgramUniform1i || nullptr == fFunctions.fProgramUniform2i || nullptr == fFunctions.fProgramUniform3i || nullptr == fFunctions.fProgramUniform4i || nullptr == fFunctions.fProgramUniform1fv || nullptr == fFunctions.fProgramUniform2fv || nullptr == fFunctions.fProgramUniform3fv || nullptr == fFunctions.fProgramUniform4fv || nullptr == fFunctions.fProgramUniform1iv || nullptr == fFunctions.fProgramUniform2iv || nullptr == fFunctions.fProgramUniform3iv || nullptr == fFunctions.fProgramUniform4iv || nullptr == fFunctions.fProgramUniformMatrix2fv || nullptr == fFunctions.fProgramUniformMatrix3fv || nullptr == fFunctions.fProgramUniformMatrix4fv) { RETURN_FALSE_INTERFACE } } if (glVer >= GR_GL_VER(2,1)) { if (nullptr == fFunctions.fProgramUniformMatrix2x3fv || nullptr == fFunctions.fProgramUniformMatrix3x2fv || nullptr == fFunctions.fProgramUniformMatrix2x4fv || nullptr == fFunctions.fProgramUniformMatrix4x2fv || nullptr == fFunctions.fProgramUniformMatrix3x4fv || nullptr == fFunctions.fProgramUniformMatrix4x3fv) { RETURN_FALSE_INTERFACE } } if (glVer >= GR_GL_VER(3,0)) { if (nullptr == fFunctions.fNamedRenderbufferStorage || nullptr == fFunctions.fGetNamedRenderbufferParameteriv || nullptr == fFunctions.fNamedRenderbufferStorageMultisample || nullptr == fFunctions.fCheckNamedFramebufferStatus || nullptr == fFunctions.fNamedFramebufferTexture1D || nullptr == fFunctions.fNamedFramebufferTexture2D || nullptr == fFunctions.fNamedFramebufferTexture3D || nullptr == fFunctions.fNamedFramebufferRenderbuffer || nullptr == fFunctions.fGetNamedFramebufferAttachmentParameteriv || nullptr == fFunctions.fGenerateTextureMipmap || nullptr == fFunctions.fFramebufferDrawBuffer || nullptr == fFunctions.fFramebufferDrawBuffers || nullptr == fFunctions.fFramebufferReadBuffer || nullptr == fFunctions.fGetFramebufferParameteriv || nullptr == fFunctions.fNamedCopyBufferSubData || nullptr == fFunctions.fVertexArrayVertexOffset || nullptr == fFunctions.fVertexArrayColorOffset || nullptr == fFunctions.fVertexArrayEdgeFlagOffset || nullptr == fFunctions.fVertexArrayIndexOffset || nullptr == fFunctions.fVertexArrayNormalOffset || nullptr == fFunctions.fVertexArrayTexCoordOffset || nullptr == fFunctions.fVertexArrayMultiTexCoordOffset || nullptr == fFunctions.fVertexArrayFogCoordOffset || nullptr == fFunctions.fVertexArraySecondaryColorOffset || nullptr == fFunctions.fVertexArrayVertexAttribOffset || nullptr == fFunctions.fVertexArrayVertexAttribIOffset || nullptr == fFunctions.fEnableVertexArray || nullptr == fFunctions.fDisableVertexArray || nullptr == fFunctions.fEnableVertexArrayAttrib || nullptr == fFunctions.fDisableVertexArrayAttrib || nullptr == fFunctions.fGetVertexArrayIntegerv || nullptr == fFunctions.fGetVertexArrayPointerv || nullptr == fFunctions.fGetVertexArrayIntegeri_v || nullptr == fFunctions.fGetVertexArrayPointeri_v || nullptr == fFunctions.fMapNamedBufferRange || nullptr == fFunctions.fFlushMappedNamedBufferRange) { RETURN_FALSE_INTERFACE } } } if ((kGL_GrGLStandard == fStandard && glVer >= GR_GL_VER(4,3)) || fExtensions.has("GL_KHR_debug")) { if (nullptr == fFunctions.fDebugMessageControl || nullptr == fFunctions.fDebugMessageInsert || nullptr == fFunctions.fDebugMessageCallback || nullptr == fFunctions.fGetDebugMessageLog || nullptr == fFunctions.fPushDebugGroup || nullptr == fFunctions.fPopDebugGroup || nullptr == fFunctions.fObjectLabel) { RETURN_FALSE_INTERFACE } } if (fExtensions.has("EGL_KHR_image") || fExtensions.has("EGL_KHR_image_base")) { if (nullptr == fFunctions.fEGLCreateImage || nullptr == fFunctions.fEGLDestroyImage) { RETURN_FALSE_INTERFACE } } return true; }