所以我正在关注转换反馈here上可以找到的一个最简单的例子,并将其转换为Android / Java,但Android似乎缺少OpenGL ES中可用的glGetBufferSubData
函数3.0。
我已经尝试搜索Android Source以查看在JNI下调用另一个方法时是否使用了此方法,但我不确定我是否正在查找正确的位置。
如果此功能不存在,我如何从变换反馈中读取数据?我查看了glGetTransformfeedbackVarying
,glGetBufferPointerv
和glReadBuffer
,但似乎没有做我想做的事。
这是我的代码:
import com.harmonicprocesses.penelopefree.openGL.MyGLRenderer;
import android.opengl.GLES30;
import android.util.Log;
import java.nio.ByteBuffer;
import java.nio.FloatBuffer;
/**
* Created by izzy on 6/24/15.
*/
public class TransformFeedback {
// Vertex shader
private final String vertexShaderSrc =
"in float inValue;\n" +
"out float outValue;\n" +
"void main() {\n" +
" outValue = sqrt(inValue);\n" +
"}\n";
private final int mProgram;
public TransformFeedback{
// Compile shader
int vertexShader = MyGLRenderer.loadShader(GLES30.GL_VERTEX_SHADER,
vertexShaderSrc);
// Create program and specify transform feedback variables
mProgram = GLES30.glCreateProgram();
GLES30.glAttachShader(mProgram, vertexShader);
//const GLchar* feedbackVaryings[] = { "outValue" };
GLES30.glTransformFeedbackVaryings(mProgram, 1, {"outValue"}, GLES30.GL_INTERLEAVED_ATTRIBS);
GLES30.glLinkProgram(mProgram);
GLES30.glUseProgram(mProgram);
// Create VAO
int[] vao = new int[1];
GLES30.glGenVertexArrays(1, vao, 0);
GLES30.glBindVertexArray(vao[0]);
// Create input VBO and vertex format
int bufferLength = 5 * 4; //5 floats 4 bytes each
ByteBuffer bb = ByteBuffer.allocateDirect(bufferLength);
FloatBuffer data = bb.asFloatBuffer();
float[] floatData = { 1.0f, 2.0f, 3.0f, 4.0f, 5.0f };
data.put(floatData);
data.position(0);
int[] vbo = new int[1];
GLES30.glGenBuffers(1, vbo, 0);
GLES30.glBindBuffer(GLES30.GL_ARRAY_BUFFER, vbo[0]);
GLES30.glBufferData(GLES30.GL_ARRAY_BUFFER, bufferLength, data, GLES30.GL_STATIC_DRAW);
int inputAttrib = GLES30.glGetAttribLocation(mProgram, "inValue");
GLES30.glEnableVertexAttribArray(inputAttrib);
GLES30.glVertexAttribPointer(inputAttrib, 1, GLES30.GL_FLOAT, false, 0, 0);
// Create transform feedback buffer
int[] tbo = new int[1];
GLES30.glGenBuffers(1, tbo, 0);
GLES30.glBindBuffer(GLES30.GL_ARRAY_BUFFER, tbo[0]);
GLES30.glBufferData(GLES30.GL_ARRAY_BUFFER, bufferLength, null, GLES30.GL_STATIC_READ);
// Perform feedback transform
GLES30.glEnable(GLES30.GL_RASTERIZER_DISCARD);
GLES30.glBindBufferBase(GLES30.GL_TRANSFORM_FEEDBACK_BUFFER, 0, tbo[0]);
GLES30.glBeginTransformFeedback(GLES30.GL_POINTS);
GLES30.glDrawArrays(GLES30.GL_POINTS, 0, 5);
GLES30.glEndTransformFeedback();
GLES30.glDisable(GLES30.GL_RASTERIZER_DISCARD);
GLES30.glFlush();
// Fetch and print results
float feedback[] = new float[5];
GLES30.glGetBufferSubData(GLES30.GL_TRANSFORM_FEEDBACK_BUFFER, 0, bufferLength, feedback);
Log.e("TransformFeedback", String.format("%f %f %f %f %f\n", feedback[0], feedback[1], feedback[2], feedback[3], feedback[4]));
}
我的最终实现不需要读取数据,只需在下一遍(反馈)的着色器中使用它。但这确实是第一步。
答案 0 :(得分:2)
glGetBufferSubData()
在ES 3.0或ES 3.1中不可用。因此,在这种情况下,Android Java绑定中没有任何遗漏。
用于在ES 3.0中读回缓冲区数据的函数是glMapBufferRange()
。在C / C ++绑定中,这将返回一个指向包含缓冲区内容的CPU可读内存的指针。完成阅读后别忘了给glUnmapBuffer()
打电话。
在Android Java bindings中,函数返回类型为Buffer
的对象。由于您无法直接从缓冲区读取,因此必须先将其转换为ByteBuffer
。这并没有真正记录,但我的答案解释了为什么我认为这种类型演员是安全的:Safe usage of glMapBufferRange() on Android/Java。