mirror of
https://git.dev.opencascade.org/repos/occt.git
synced 2025-04-05 18:16:23 +03:00
0031477: Visualization, TKOpenGl - fetch/wrap getBufferSubData() function from WebGL 2.0
Added OpenGl_Context::GetBufferSubData() implementing getBufferSubData() based on capabilities of various APIs. Added OpenGl_VertexBuffer::GetSubData() similar to OpenGl_VertexBuffer::SubData().
This commit is contained in:
parent
872f98d9ef
commit
d4cefcc0da
@ -65,6 +65,7 @@ IMPLEMENT_STANDARD_RTTIEXT(OpenGl_Context,Standard_Transient)
|
|||||||
#endif
|
#endif
|
||||||
|
|
||||||
#ifdef __EMSCRIPTEN__
|
#ifdef __EMSCRIPTEN__
|
||||||
|
#include <emscripten.h>
|
||||||
#include <emscripten/html5.h>
|
#include <emscripten/html5.h>
|
||||||
|
|
||||||
//! Check if WebGL extension is available and activate it
|
//! Check if WebGL extension is available and activate it
|
||||||
@ -138,6 +139,7 @@ OpenGl_Context::OpenGl_Context (const Handle(OpenGl_Caps)& theCaps)
|
|||||||
core45 (NULL),
|
core45 (NULL),
|
||||||
core45back (NULL),
|
core45back (NULL),
|
||||||
caps (!theCaps.IsNull() ? theCaps : new OpenGl_Caps()),
|
caps (!theCaps.IsNull() ? theCaps : new OpenGl_Caps()),
|
||||||
|
hasGetBufferData (Standard_False),
|
||||||
#if defined(GL_ES_VERSION_2_0)
|
#if defined(GL_ES_VERSION_2_0)
|
||||||
hasHighp (Standard_False),
|
hasHighp (Standard_False),
|
||||||
hasUintIndex(Standard_False),
|
hasUintIndex(Standard_False),
|
||||||
@ -1677,6 +1679,7 @@ void OpenGl_Context::init (const Standard_Boolean theIsCoreProfile)
|
|||||||
{
|
{
|
||||||
core30 = (OpenGl_GlCore30* )(&(*myFuncs));
|
core30 = (OpenGl_GlCore30* )(&(*myFuncs));
|
||||||
core30fwd = (OpenGl_GlCore30Fwd* )(&(*myFuncs));
|
core30fwd = (OpenGl_GlCore30Fwd* )(&(*myFuncs));
|
||||||
|
hasGetBufferData = true;
|
||||||
}
|
}
|
||||||
|
|
||||||
// load OpenGL ES 3.1 new functions
|
// load OpenGL ES 3.1 new functions
|
||||||
@ -2183,6 +2186,7 @@ void OpenGl_Context::init (const Standard_Boolean theIsCoreProfile)
|
|||||||
core15 = (OpenGl_GlCore15* )(&(*myFuncs));
|
core15 = (OpenGl_GlCore15* )(&(*myFuncs));
|
||||||
}
|
}
|
||||||
core15fwd = (OpenGl_GlCore15Fwd* )(&(*myFuncs));
|
core15fwd = (OpenGl_GlCore15Fwd* )(&(*myFuncs));
|
||||||
|
hasGetBufferData = true;
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
@ -4612,6 +4616,36 @@ bool OpenGl_Context::SetSampleAlphaToCoverage (bool theToEnable)
|
|||||||
return anOldValue;
|
return anOldValue;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// =======================================================================
|
||||||
|
// function : GetBufferSubData
|
||||||
|
// purpose :
|
||||||
|
// =======================================================================
|
||||||
|
bool OpenGl_Context::GetBufferSubData (GLenum theTarget, GLintptr theOffset, GLsizeiptr theSize, void* theData)
|
||||||
|
{
|
||||||
|
if (!hasGetBufferData)
|
||||||
|
{
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
#ifdef __EMSCRIPTEN__
|
||||||
|
EM_ASM_(
|
||||||
|
{
|
||||||
|
Module.ctx.getBufferSubData($0, $1, HEAPU8.subarray($2, $2 + $3));
|
||||||
|
}, theTarget, theOffset, theData, theSize);
|
||||||
|
return true;
|
||||||
|
#elif defined(GL_ES_VERSION_2_0)
|
||||||
|
if (void* aData = core30fwd->glMapBufferRange (theTarget, theOffset, theSize, GL_MAP_READ_BIT))
|
||||||
|
{
|
||||||
|
memcpy (theData, aData, theSize);
|
||||||
|
core30fwd->glUnmapBuffer (theTarget);
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
#else
|
||||||
|
core15fwd->glGetBufferSubData (theTarget, theOffset, theSize, theData);
|
||||||
|
return true;
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
|
||||||
// =======================================================================
|
// =======================================================================
|
||||||
// function : DumpJson
|
// function : DumpJson
|
||||||
// purpose :
|
// purpose :
|
||||||
|
@ -987,6 +987,17 @@ public: //! @name methods to alter or retrieve current state
|
|||||||
//! Set line feater width.
|
//! Set line feater width.
|
||||||
void SetLineFeather(Standard_ShortReal theValue) { myLineFeather = theValue; }
|
void SetLineFeather(Standard_ShortReal theValue) { myLineFeather = theValue; }
|
||||||
|
|
||||||
|
//! Wrapper over glGetBufferSubData(), implemented as:
|
||||||
|
//! - OpenGL 1.5+ (desktop) via glGetBufferSubData();
|
||||||
|
//! - OpenGL ES 3.0+ via glMapBufferRange();
|
||||||
|
//! - WebGL 2.0+ via gl.getBufferSubData().
|
||||||
|
//! @param theTarget [in] target buffer to map
|
||||||
|
//! @param theOffset [in] offset to the beginning of sub-data
|
||||||
|
//! @param theSize [in] number of bytes to read
|
||||||
|
//! @param theData [out] destination pointer to fill
|
||||||
|
//! @return FALSE if functionality is unavailable
|
||||||
|
Standard_EXPORT bool GetBufferSubData (GLenum theTarget, GLintptr theOffset, GLsizeiptr theSize, void* theData);
|
||||||
|
|
||||||
//! Return Graphics Driver's vendor.
|
//! Return Graphics Driver's vendor.
|
||||||
const TCollection_AsciiString& Vendor() const { return myVendor; }
|
const TCollection_AsciiString& Vendor() const { return myVendor; }
|
||||||
|
|
||||||
@ -1044,6 +1055,7 @@ public: //! @name core profiles
|
|||||||
|
|
||||||
public: //! @name extensions
|
public: //! @name extensions
|
||||||
|
|
||||||
|
Standard_Boolean hasGetBufferData; //!< flag indicating if GetBufferSubData() is supported
|
||||||
Standard_Boolean hasHighp; //!< highp in GLSL ES fragment shader is supported
|
Standard_Boolean hasHighp; //!< highp in GLSL ES fragment shader is supported
|
||||||
Standard_Boolean hasUintIndex; //!< GLuint for index buffer is supported (always available on desktop; on OpenGL ES - since 3.0 or as extension GL_OES_element_index_uint)
|
Standard_Boolean hasUintIndex; //!< GLuint for index buffer is supported (always available on desktop; on OpenGL ES - since 3.0 or as extension GL_OES_element_index_uint)
|
||||||
Standard_Boolean hasTexRGBA8; //!< always available on desktop; on OpenGL ES - since 3.0 or as extension GL_OES_rgb8_rgba8
|
Standard_Boolean hasTexRGBA8; //!< always available on desktop; on OpenGL ES - since 3.0 or as extension GL_OES_rgb8_rgba8
|
||||||
|
@ -132,6 +132,18 @@ public: //! @name OpenGL 3.0 additives to 2.1
|
|||||||
using theBaseClass_t::glVertexAttribI4ubv;
|
using theBaseClass_t::glVertexAttribI4ubv;
|
||||||
using theBaseClass_t::glVertexAttribI4usv;
|
using theBaseClass_t::glVertexAttribI4usv;
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
#if defined(GL_ES_VERSION_2_0)
|
||||||
|
// the following functions from OpenGL 1.5 have been added only in OpenGL ES 3.0
|
||||||
|
using theBaseClass_t::glGenQueries;
|
||||||
|
using theBaseClass_t::glDeleteQueries;
|
||||||
|
using theBaseClass_t::glIsQuery;
|
||||||
|
using theBaseClass_t::glBeginQuery;
|
||||||
|
using theBaseClass_t::glEndQuery;
|
||||||
|
using theBaseClass_t::glGetQueryiv;
|
||||||
|
using theBaseClass_t::glGetQueryObjectuiv;
|
||||||
|
using theBaseClass_t::glUnmapBuffer;
|
||||||
|
#endif
|
||||||
};
|
};
|
||||||
|
|
||||||
//! OpenGL 3.0 core based on 2.1 version.
|
//! OpenGL 3.0 core based on 2.1 version.
|
||||||
|
@ -160,6 +160,33 @@ bool OpenGl_VertexBuffer::subData (const Handle(OpenGl_Context)& theGlCtx,
|
|||||||
return isDone;
|
return isDone;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// =======================================================================
|
||||||
|
// function : getSubData
|
||||||
|
// purpose :
|
||||||
|
// =======================================================================
|
||||||
|
bool OpenGl_VertexBuffer::getSubData (const Handle(OpenGl_Context)& theGlCtx,
|
||||||
|
const GLsizei theElemFrom,
|
||||||
|
const GLsizei theElemsNb,
|
||||||
|
void* theData,
|
||||||
|
const GLenum theDataType)
|
||||||
|
{
|
||||||
|
if (!IsValid() || myDataType != theDataType
|
||||||
|
|| theElemFrom < 0 || ((theElemFrom + theElemsNb) > myElemsNb)
|
||||||
|
|| !theGlCtx->hasGetBufferData)
|
||||||
|
{
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
Bind (theGlCtx);
|
||||||
|
const size_t aDataSize = sizeOfGlType (theDataType);
|
||||||
|
const GLintptr anOffset = GLintptr (theElemFrom) * GLintptr (myComponentsNb) * aDataSize;
|
||||||
|
const GLsizeiptr aSize = GLsizeiptr(theElemsNb) * GLsizeiptr(myComponentsNb) * aDataSize;
|
||||||
|
bool isDone = theGlCtx->GetBufferSubData (GetTarget(), anOffset, aSize, theData);
|
||||||
|
isDone = isDone && (glGetError() == GL_NO_ERROR);
|
||||||
|
Unbind (theGlCtx);
|
||||||
|
return isDone;
|
||||||
|
}
|
||||||
|
|
||||||
// =======================================================================
|
// =======================================================================
|
||||||
// function : BindVertexAttrib
|
// function : BindVertexAttrib
|
||||||
// purpose :
|
// purpose :
|
||||||
|
@ -154,6 +154,20 @@ public:
|
|||||||
return subData (theGlCtx, theElemFrom, theElemsNb, theData, GL_FLOAT);
|
return subData (theGlCtx, theElemFrom, theElemsNb, theData, GL_FLOAT);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
//! Read back buffer sub-range.
|
||||||
|
//! Notice that VBO will be unbound after this call.
|
||||||
|
//! Function reads portion of data from this VBO using glGetBufferSubData().
|
||||||
|
//! @param theElemFrom [in] element id from which replace buffer data (>=0);
|
||||||
|
//! @param theElemsNb [in] elements count (theElemFrom + theElemsNb <= GetElemsNb());
|
||||||
|
//! @param theData [out] destination pointer to GLfloat data.
|
||||||
|
bool GetSubData (const Handle(OpenGl_Context)& theGlCtx,
|
||||||
|
const GLsizei theElemFrom,
|
||||||
|
const GLsizei theElemsNb,
|
||||||
|
GLfloat* theData)
|
||||||
|
{
|
||||||
|
return getSubData (theGlCtx, theElemFrom, theElemsNb, theData, GL_FLOAT);
|
||||||
|
}
|
||||||
|
|
||||||
//! Notice that VBO will be unbound after this call.
|
//! Notice that VBO will be unbound after this call.
|
||||||
//! Function replaces portion of data within this VBO using glBufferSubData().
|
//! Function replaces portion of data within this VBO using glBufferSubData().
|
||||||
//! The VBO should be initialized before call.
|
//! The VBO should be initialized before call.
|
||||||
@ -168,6 +182,20 @@ public:
|
|||||||
return subData (theGlCtx, theElemFrom, theElemsNb, theData, GL_UNSIGNED_INT);
|
return subData (theGlCtx, theElemFrom, theElemsNb, theData, GL_UNSIGNED_INT);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
//! Read back buffer sub-range.
|
||||||
|
//! Notice that VBO will be unbound after this call.
|
||||||
|
//! Function reads portion of data from this VBO using glGetBufferSubData().
|
||||||
|
//! @param theElemFrom [in] element id from which replace buffer data (>=0);
|
||||||
|
//! @param theElemsNb [in] elements count (theElemFrom + theElemsNb <= GetElemsNb());
|
||||||
|
//! @param theData [out] destination pointer to GLuint data.
|
||||||
|
bool GetSubData (const Handle(OpenGl_Context)& theGlCtx,
|
||||||
|
const GLsizei theElemFrom,
|
||||||
|
const GLsizei theElemsNb,
|
||||||
|
GLuint* theData)
|
||||||
|
{
|
||||||
|
return getSubData (theGlCtx, theElemFrom, theElemsNb, theData, GL_UNSIGNED_INT);
|
||||||
|
}
|
||||||
|
|
||||||
//! Notice that VBO will be unbound after this call.
|
//! Notice that VBO will be unbound after this call.
|
||||||
//! Function replaces portion of data within this VBO using glBufferSubData().
|
//! Function replaces portion of data within this VBO using glBufferSubData().
|
||||||
//! The VBO should be initialized before call.
|
//! The VBO should be initialized before call.
|
||||||
@ -182,6 +210,20 @@ public:
|
|||||||
return subData (theGlCtx, theElemFrom, theElemsNb, theData, GL_UNSIGNED_SHORT);
|
return subData (theGlCtx, theElemFrom, theElemsNb, theData, GL_UNSIGNED_SHORT);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
//! Read back buffer sub-range.
|
||||||
|
//! Notice that VBO will be unbound after this call.
|
||||||
|
//! Function reads portion of data from this VBO using glGetBufferSubData().
|
||||||
|
//! @param theElemFrom [in] element id from which replace buffer data (>=0);
|
||||||
|
//! @param theElemsNb [in] elements count (theElemFrom + theElemsNb <= GetElemsNb());
|
||||||
|
//! @param theData [out] destination pointer to GLushort data.
|
||||||
|
bool GetSubData (const Handle(OpenGl_Context)& theGlCtx,
|
||||||
|
const GLsizei theElemFrom,
|
||||||
|
const GLsizei theElemsNb,
|
||||||
|
GLushort* theData)
|
||||||
|
{
|
||||||
|
return getSubData (theGlCtx, theElemFrom, theElemsNb, theData, GL_UNSIGNED_SHORT);
|
||||||
|
}
|
||||||
|
|
||||||
//! Notice that VBO will be unbound after this call.
|
//! Notice that VBO will be unbound after this call.
|
||||||
//! Function replaces portion of data within this VBO using glBufferSubData().
|
//! Function replaces portion of data within this VBO using glBufferSubData().
|
||||||
//! The VBO should be initialized before call.
|
//! The VBO should be initialized before call.
|
||||||
@ -196,6 +238,20 @@ public:
|
|||||||
return subData (theGlCtx, theElemFrom, theElemsNb, theData, GL_UNSIGNED_BYTE);
|
return subData (theGlCtx, theElemFrom, theElemsNb, theData, GL_UNSIGNED_BYTE);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
//! Read back buffer sub-range.
|
||||||
|
//! Notice that VBO will be unbound after this call.
|
||||||
|
//! Function reads portion of data from this VBO using glGetBufferSubData().
|
||||||
|
//! @param theElemFrom [in] element id from which replace buffer data (>=0);
|
||||||
|
//! @param theElemsNb [in] elements count (theElemFrom + theElemsNb <= GetElemsNb());
|
||||||
|
//! @param theData [out] destination pointer to GLubyte data.
|
||||||
|
bool GetSubData (const Handle(OpenGl_Context)& theGlCtx,
|
||||||
|
const GLsizei theElemFrom,
|
||||||
|
const GLsizei theElemsNb,
|
||||||
|
GLubyte* theData)
|
||||||
|
{
|
||||||
|
return getSubData (theGlCtx, theElemFrom, theElemsNb, theData, GL_UNSIGNED_BYTE);
|
||||||
|
}
|
||||||
|
|
||||||
//! Bind this VBO to active GLSL program.
|
//! Bind this VBO to active GLSL program.
|
||||||
Standard_EXPORT void BindVertexAttrib (const Handle(OpenGl_Context)& theGlCtx,
|
Standard_EXPORT void BindVertexAttrib (const Handle(OpenGl_Context)& theGlCtx,
|
||||||
const GLuint theAttribLoc) const;
|
const GLuint theAttribLoc) const;
|
||||||
@ -286,6 +342,13 @@ public: //! @name advanced methods
|
|||||||
const void* theData,
|
const void* theData,
|
||||||
const GLenum theDataType);
|
const GLenum theDataType);
|
||||||
|
|
||||||
|
//! Read back buffer sub-range.
|
||||||
|
Standard_EXPORT virtual bool getSubData (const Handle(OpenGl_Context)& theGlCtx,
|
||||||
|
const GLsizei theElemFrom,
|
||||||
|
const GLsizei theElemsNb,
|
||||||
|
void* theData,
|
||||||
|
const GLenum theDataType);
|
||||||
|
|
||||||
//! Setup array pointer - either for active GLSL program OpenGl_Context::ActiveProgram()
|
//! Setup array pointer - either for active GLSL program OpenGl_Context::ActiveProgram()
|
||||||
//! or for FFP using bindFixed() when no program bound.
|
//! or for FFP using bindFixed() when no program bound.
|
||||||
static void bindAttribute (const Handle(OpenGl_Context)& theGlCtx,
|
static void bindAttribute (const Handle(OpenGl_Context)& theGlCtx,
|
||||||
|
@ -175,3 +175,27 @@ bool OpenGl_VertexBufferCompat::subData (const Handle(OpenGl_Context)& ,
|
|||||||
memcpy (myData->ChangeData() + anOffset, theData, aNbBytes);
|
memcpy (myData->ChangeData() + anOffset, theData, aNbBytes);
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// =======================================================================
|
||||||
|
// function : getSubData
|
||||||
|
// purpose :
|
||||||
|
// =======================================================================
|
||||||
|
bool OpenGl_VertexBufferCompat::getSubData (const Handle(OpenGl_Context)& ,
|
||||||
|
const GLsizei theElemFrom,
|
||||||
|
const GLsizei theElemsNb,
|
||||||
|
void* theData,
|
||||||
|
const GLenum theDataType)
|
||||||
|
{
|
||||||
|
if (!IsValid() || myDataType != theDataType
|
||||||
|
|| theElemFrom < 0 || ((theElemFrom + theElemsNb) > myElemsNb)
|
||||||
|
|| theData == NULL)
|
||||||
|
{
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
const size_t aDataSize = sizeOfGlType (theDataType);
|
||||||
|
const size_t anOffset = size_t(theElemFrom) * size_t(myComponentsNb) * aDataSize;
|
||||||
|
const size_t aNbBytes = size_t(theElemsNb) * size_t(myComponentsNb) * aDataSize;
|
||||||
|
memcpy (theData, myData->Data() + anOffset, aNbBytes);
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
@ -82,6 +82,13 @@ public: //! @name advanced methods
|
|||||||
const void* theData,
|
const void* theData,
|
||||||
const GLenum theDataType) Standard_OVERRIDE;
|
const GLenum theDataType) Standard_OVERRIDE;
|
||||||
|
|
||||||
|
//! Read back buffer sub-range.
|
||||||
|
Standard_EXPORT virtual bool getSubData (const Handle(OpenGl_Context)& theGlCtx,
|
||||||
|
const GLsizei theElemFrom,
|
||||||
|
const GLsizei theElemsNb,
|
||||||
|
void* theData,
|
||||||
|
const GLenum theDataType) Standard_OVERRIDE;
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
|
|
||||||
Handle(NCollection_Buffer) myData; //!< buffer data
|
Handle(NCollection_Buffer) myData; //!< buffer data
|
||||||
|
Loading…
x
Reference in New Issue
Block a user