2014-09-18 02:08:41 +04:00
|
|
|
/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
|
|
|
|
/* This Source Code Form is subject to the terms of the Mozilla Public
|
|
|
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
|
|
|
* file, You can obtain one at http://mozilla.org/MPL/2.0/. */
|
|
|
|
|
|
|
|
#include "WebGL2Context.h"
|
2014-11-26 05:00:06 +03:00
|
|
|
#include "WebGLActiveInfo.h"
|
|
|
|
#include "WebGLProgram.h"
|
|
|
|
#include "WebGLTransformFeedback.h"
|
2014-09-18 02:08:41 +04:00
|
|
|
#include "GLContext.h"
|
|
|
|
|
|
|
|
using namespace mozilla;
|
|
|
|
using namespace mozilla::dom;
|
|
|
|
|
|
|
|
// -------------------------------------------------------------------------
|
|
|
|
// Transform Feedback
|
|
|
|
|
|
|
|
already_AddRefed<WebGLTransformFeedback>
|
|
|
|
WebGL2Context::CreateTransformFeedback()
|
|
|
|
{
|
2014-11-26 05:00:06 +03:00
|
|
|
if (IsContextLost())
|
|
|
|
return nullptr;
|
|
|
|
|
|
|
|
GLuint tf = 0;
|
|
|
|
MakeContextCurrent();
|
|
|
|
gl->fGenTransformFeedbacks(1, &tf);
|
|
|
|
|
|
|
|
nsRefPtr<WebGLTransformFeedback> globj = new WebGLTransformFeedback(this, tf);
|
|
|
|
return globj.forget();
|
2014-09-18 02:08:41 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
WebGL2Context::DeleteTransformFeedback(WebGLTransformFeedback* tf)
|
|
|
|
{
|
2014-11-26 05:00:06 +03:00
|
|
|
if (IsContextLost())
|
|
|
|
return;
|
|
|
|
|
|
|
|
if (!ValidateObjectAllowDeletedOrNull("deleteTransformFeedback", tf))
|
|
|
|
return;
|
|
|
|
|
|
|
|
if (!tf || tf->IsDeleted())
|
|
|
|
return;
|
|
|
|
|
|
|
|
if (mBoundTransformFeedback == tf)
|
|
|
|
BindTransformFeedback(LOCAL_GL_TRANSFORM_FEEDBACK, tf);
|
|
|
|
|
|
|
|
tf->RequestDelete();
|
2014-09-18 02:08:41 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
bool
|
|
|
|
WebGL2Context::IsTransformFeedback(WebGLTransformFeedback* tf)
|
|
|
|
{
|
2014-11-26 05:00:06 +03:00
|
|
|
if (IsContextLost())
|
|
|
|
return false;
|
|
|
|
|
|
|
|
if (!ValidateObjectAllowDeleted("isTransformFeedback", tf))
|
|
|
|
return false;
|
|
|
|
|
|
|
|
if (tf->IsDeleted())
|
|
|
|
return false;
|
|
|
|
|
|
|
|
MakeContextCurrent();
|
|
|
|
return gl->fIsTransformFeedback(tf->GLName());
|
2014-09-18 02:08:41 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
2014-11-26 05:00:06 +03:00
|
|
|
WebGL2Context::BindTransformFeedback(GLenum target, WebGLTransformFeedback* tf)
|
2014-09-18 02:08:41 +04:00
|
|
|
{
|
2014-11-26 05:00:06 +03:00
|
|
|
if (IsContextLost())
|
|
|
|
return;
|
|
|
|
|
|
|
|
if (!ValidateObjectAllowDeletedOrNull("bindTransformFeedback", tf))
|
|
|
|
return;
|
|
|
|
|
|
|
|
if (target != LOCAL_GL_TRANSFORM_FEEDBACK)
|
|
|
|
return ErrorInvalidEnum("bindTransformFeedback: target must be TRANSFORM_FEEDBACK");
|
|
|
|
|
|
|
|
WebGLRefPtr<WebGLTransformFeedback> currentTF = mBoundTransformFeedback;
|
|
|
|
if (currentTF && currentTF->mIsActive && !currentTF->mIsPaused) {
|
|
|
|
return ErrorInvalidOperation("bindTransformFeedback: Currently bound transform "
|
|
|
|
"feedback is active and not paused");
|
|
|
|
}
|
|
|
|
|
|
|
|
if (tf && tf->IsDeleted())
|
|
|
|
return ErrorInvalidOperation("bindTransformFeedback: Attempt to bind deleted id");
|
|
|
|
|
|
|
|
if (tf)
|
|
|
|
tf->BindTo(LOCAL_GL_TRANSFORM_FEEDBACK);
|
|
|
|
|
|
|
|
MakeContextCurrent();
|
|
|
|
gl->fBindTransformFeedback(target, tf ? tf->GLName() : 0);
|
|
|
|
if (tf)
|
|
|
|
mBoundTransformFeedback = tf;
|
|
|
|
else
|
|
|
|
mBoundTransformFeedback = mDefaultTransformFeedback;
|
2014-09-18 02:08:41 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
WebGL2Context::BeginTransformFeedback(GLenum primitiveMode)
|
|
|
|
{
|
2014-11-26 05:00:06 +03:00
|
|
|
if (IsContextLost())
|
|
|
|
return;
|
|
|
|
|
|
|
|
WebGLTransformFeedback* tf = mBoundTransformFeedback;
|
|
|
|
MOZ_ASSERT(tf);
|
|
|
|
if (!tf)
|
|
|
|
return;
|
|
|
|
|
|
|
|
if (tf->mIsActive)
|
|
|
|
return ErrorInvalidOperation("beginTransformFeedback: transform feedback is active");
|
|
|
|
|
|
|
|
const GLenum mode = tf->mMode;
|
|
|
|
if (mode != LOCAL_GL_POINTS && mode != LOCAL_GL_LINES && mode != LOCAL_GL_TRIANGLES)
|
|
|
|
return ErrorInvalidEnum("beginTransformFeedback: primitive must be one of POINTS, LINES, or TRIANGLES");
|
|
|
|
|
|
|
|
// TODO:
|
|
|
|
// GL_INVALID_OPERATION is generated by glBeginTransformFeedback
|
|
|
|
// if any binding point used in transform feedback mode does not
|
|
|
|
// have a buffer object bound. In interleaved mode, only the first
|
|
|
|
// buffer object binding point is ever written to.
|
|
|
|
|
|
|
|
// GL_INVALID_OPERATION is generated by glBeginTransformFeedback
|
|
|
|
// if no binding points would be used, either because no program
|
|
|
|
// object is active of because the active program object has
|
|
|
|
// specified no varying variables to record.
|
|
|
|
if (!mCurrentProgram)
|
|
|
|
return ErrorInvalidOperation("beginTransformFeedback: no program is active");
|
|
|
|
|
|
|
|
MakeContextCurrent();
|
|
|
|
gl->fBeginTransformFeedback(primitiveMode);
|
|
|
|
tf->mIsActive = true;
|
|
|
|
tf->mIsPaused = false;
|
2014-09-18 02:08:41 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
WebGL2Context::EndTransformFeedback()
|
|
|
|
{
|
2014-11-26 05:00:06 +03:00
|
|
|
if (IsContextLost())
|
|
|
|
return;
|
|
|
|
|
|
|
|
WebGLTransformFeedback* tf = mBoundTransformFeedback;
|
|
|
|
MOZ_ASSERT(tf);
|
|
|
|
|
|
|
|
if (!tf)
|
|
|
|
return;
|
|
|
|
|
|
|
|
if (!tf->mIsActive)
|
|
|
|
return ErrorInvalidOperation("%s: transform feedback in not active",
|
|
|
|
"endTransformFeedback");
|
|
|
|
|
|
|
|
MakeContextCurrent();
|
|
|
|
gl->fEndTransformFeedback();
|
|
|
|
tf->mIsActive = false;
|
|
|
|
tf->mIsPaused = false;
|
2014-09-18 02:08:41 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
2014-11-26 05:00:06 +03:00
|
|
|
WebGL2Context::PauseTransformFeedback()
|
2014-09-18 02:08:41 +04:00
|
|
|
{
|
2014-11-26 05:00:06 +03:00
|
|
|
if (IsContextLost())
|
|
|
|
return;
|
2014-09-18 02:08:41 +04:00
|
|
|
|
2014-11-26 05:00:06 +03:00
|
|
|
WebGLTransformFeedback* tf = mBoundTransformFeedback;
|
|
|
|
MOZ_ASSERT(tf);
|
|
|
|
if (!tf)
|
|
|
|
return;
|
2014-09-18 02:08:41 +04:00
|
|
|
|
2014-11-26 05:00:06 +03:00
|
|
|
if (!tf->mIsActive || tf->mIsPaused) {
|
|
|
|
return ErrorInvalidOperation("%s: transform feedback is not active or is paused",
|
|
|
|
"pauseTransformFeedback");
|
|
|
|
}
|
|
|
|
|
|
|
|
MakeContextCurrent();
|
|
|
|
gl->fPauseTransformFeedback();
|
|
|
|
tf->mIsPaused = true;
|
2014-09-18 02:08:41 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
2014-11-26 05:00:06 +03:00
|
|
|
WebGL2Context::ResumeTransformFeedback()
|
2014-09-18 02:08:41 +04:00
|
|
|
{
|
2014-11-26 05:00:06 +03:00
|
|
|
if (IsContextLost())
|
|
|
|
return;
|
|
|
|
|
|
|
|
WebGLTransformFeedback* tf = mBoundTransformFeedback;
|
|
|
|
MOZ_ASSERT(tf);
|
|
|
|
if (!tf)
|
|
|
|
return;
|
|
|
|
|
|
|
|
if (!tf->mIsActive || !tf->mIsPaused)
|
|
|
|
return ErrorInvalidOperation("resumeTransformFeedback: transform feedback is not active or is not paused");
|
|
|
|
|
|
|
|
MakeContextCurrent();
|
|
|
|
gl->fResumeTransformFeedback();
|
|
|
|
tf->mIsPaused = false;
|
2014-09-18 02:08:41 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
2014-11-26 05:00:06 +03:00
|
|
|
WebGL2Context::TransformFeedbackVaryings(WebGLProgram* program,
|
|
|
|
const dom::Sequence<nsString>& varyings,
|
|
|
|
GLenum bufferMode)
|
2014-09-18 02:08:41 +04:00
|
|
|
{
|
2014-11-26 05:00:06 +03:00
|
|
|
if (IsContextLost())
|
|
|
|
return;
|
|
|
|
|
|
|
|
if (!ValidateObject("transformFeedbackVaryings: program", program))
|
|
|
|
return;
|
|
|
|
|
|
|
|
GLsizei count = varyings.Length();
|
|
|
|
GLchar** tmpVaryings = (GLchar**) nsMemory::Alloc(count * sizeof(GLchar*));
|
|
|
|
|
|
|
|
for (GLsizei n = 0; n < count; n++) {
|
|
|
|
tmpVaryings[n] = (GLchar*) ToNewCString(varyings[n]);
|
|
|
|
}
|
|
|
|
|
2015-01-16 02:40:39 +03:00
|
|
|
GLuint progname = program->mGLName;
|
2014-11-26 05:00:06 +03:00
|
|
|
MakeContextCurrent();
|
|
|
|
gl->fTransformFeedbackVaryings(progname, count, tmpVaryings, bufferMode);
|
|
|
|
|
|
|
|
NS_FREE_XPCOM_ALLOCATED_POINTER_ARRAY(count, tmpVaryings);
|
|
|
|
}
|
|
|
|
|
|
|
|
already_AddRefed<WebGLActiveInfo>
|
|
|
|
WebGL2Context::GetTransformFeedbackVarying(WebGLProgram* program, GLuint index)
|
|
|
|
{
|
|
|
|
if (IsContextLost())
|
|
|
|
return nullptr;
|
|
|
|
|
|
|
|
if (!ValidateObject("getTransformFeedbackVarying: program", program))
|
|
|
|
return nullptr;
|
|
|
|
|
|
|
|
MakeContextCurrent();
|
|
|
|
|
|
|
|
GLint len = 0;
|
2015-01-16 02:40:39 +03:00
|
|
|
GLuint progname = program->mGLName;
|
2014-11-26 05:00:06 +03:00
|
|
|
gl->fGetProgramiv(progname, LOCAL_GL_TRANSFORM_FEEDBACK_VARYING_MAX_LENGTH, &len);
|
|
|
|
if (!len)
|
|
|
|
return nullptr;
|
|
|
|
|
|
|
|
UniquePtr<char[]> name(new char[len]);
|
|
|
|
GLint tfsize = 0;
|
|
|
|
GLuint tftype = 0;
|
|
|
|
|
|
|
|
gl->fGetTransformFeedbackVarying(progname, index, len, &len, &tfsize, &tftype, name.get());
|
|
|
|
if (len == 0 || tfsize == 0 || tftype == 0)
|
|
|
|
return nullptr;
|
|
|
|
|
2015-01-16 02:40:39 +03:00
|
|
|
MOZ_CRASH("todo");
|
|
|
|
/*
|
|
|
|
// Reverse lookup of name
|
|
|
|
nsCString reverseMappedName;
|
|
|
|
prog->ReverveMapIdentifier(nsDependentCString(name), &reverseMappedName);
|
2014-11-26 05:00:06 +03:00
|
|
|
|
|
|
|
nsRefPtr<WebGLActiveInfo> result = new WebGLActiveInfo(tfsize, tftype, nsDependentCString(name.get()));
|
|
|
|
return result.forget();
|
2015-01-16 02:40:39 +03:00
|
|
|
*/
|
2014-09-18 02:08:41 +04:00
|
|
|
}
|