[英]“shader program is not linked” when running QT hello opengl
我已经从http://releases.qt-project.org/learning/developerguides/qtopengltutorial/OpenGLTutorial.pdf下载了一个 QT OpenGL 教程,并试图运行他们在那里的最简单的例子,它只是在上面画一个三角形黑色背景。
我设法编译了这个例子,但是当我运行它时,我只得到一个黑色窗口,控制台报告以下内容:
启动 /home/minas/Desktop/hello-opengl-build-Desktop_Qt_5_0_1_GCC_64bit-Debug/hello-opengl... QGLShaderProgram::uniformLocation( mvpMatrix ): 着色器程序未链接 QGLShaderProgram::uniformLocation( color ): 着色器程序未链接QGLShaderProgram::attributeLocation( vertex ): 着色器程序未链接 QGLShaderProgram::attributeLocation( vertex ): 着色器程序未链接 QGLShaderProgram::attributeLocation( vertex ): 着色器程序未链接 QGLShaderProgram::uniformLocation( mvpMatrix ): 着色器程序是未链接 QGLShaderProgram::uniformLocation( color ): 着色器程序未链接 QGLShaderProgram::attributeLocation( vertex ): 着色器程序未链接 QGLShaderProgram::attributeLocation( vertex ): 着色器程序未链接 QGLShaderProgram::attributeLocation( vertex ): 着色器程序未链接 QGLShaderProgram::uniformLocation( mvpMatrix ): 着色器程序未链接 QGLShaderProgram::uniformLocation( color ): 着色器程序未链接 ed QGLShaderProgram::attributeLocation( vertex ): 着色器程序未链接 QGLShaderProgram::attributeLocation( vertex ): 着色器程序未链接 QGLShaderProgram::attributeLocation( vertex ): 着色器程序未链接 QGLShaderProgram::uniformLocation( mvpMatrix ): 着色器程序未链接 QGLShaderProgram::uniformLocation( color ): 着色器程序未链接 QGLShaderProgram::attributeLocation( vertex ): 着色器程序未链接 QGLShaderProgram::attributeLocation( vertex ): 着色器程序未链接 QGLShaderProgram::attributeLocation( vertex ):着色器程序未链接
.pro 文件看起来像这样(“unix:!mac”部分是我添加的,所以它会找到 OpenGL .so 文件)。
QT += core gui opengl
TARGET = hello-opengl
TEMPLATE = app
SOURCES += main.cpp\
glwidget.cpp
HEADERS += glwidget.h
OTHER_FILES += fragmentShader.fsh\
vertexShader.vsh
RESOURCES += resources.qrc
unix:!mac{
QMAKE_LFLAGS += -Wl,--rpath=/usr/lib64/nvidia-current
QMAKE_LFLAGS += -L/usr/lib64/nvidia-current
}
“glxinfo | grep -i opengl”的输出是
OpenGL vendor string: NVIDIA Corporation
OpenGL renderer string: GeForce 9800 GT/PCIe/SSE2
OpenGL version string: 2.1.2 NVIDIA 304.64
OpenGL shading language version string: (null)
OpenGL extensions:
进行实际渲染的代码部分是
void GlWidget::paintGL()
{
glClear(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT);
QMatrix4x4 mMatrix;
QMatrix4x4 vMatrix;
// shaderProgram initialized in initializeGL()
shaderProgram.bind();
// "mvpMatrix", "color" and "vertex" do exist in the shaders
shaderProgram.setUniformValue("mvpMatrix", pMatrix * vMatrix * mMatrix);
shaderProgram.setUniformValue("color", QColor(Qt::white));
shaderProgram.setAttributeArray("vertex", vertices.constData());
shaderProgram.enableAttributeArray("vertex");
glDrawArrays(GL_TRIANGLES, 0, vertices.size());
shaderProgram.disableAttributeArray("vertex");
shaderProgram.release();
}
另外,作为一个附带问题,为什么“glxinfo | grep -i opengl”的输出包含该行
OpenGL 着色语言版本字符串:(空)
(似乎与我有关......是吗?)
shaderProgram
在我的QGLWidget
子类中声明,我称之为GLWidget
并在initializeGL
中initializeGL
如下:
void GlWidget::initializeGL()
{
glEnable(GL_DEPTH_TEST);
glEnable(GL_CULL_FACE);
qglClearColor(QColor(Qt::black));
shaderProgram.addShaderFromSourceFile(QGLShader::Vertex, ":/vertexShader.vsh");
shaderProgram.addShaderFromSourceFile(QGLShader::Fragment, ":/fragmentShader.fsh");
shaderProgram.link();
vertices << QVector3D(1, 0, -2) << QVector3D(0, 1, -2) << QVector3D(-1, 0, -2);
}
这是我的 initializeGL() 方法:
void GlWidget::initializeGL()
{
glEnable(GL_DEPTH_TEST);
glEnable(GL_CULL_FACE);
qglClearColor(QColor(Qt::black));
shaderProgram.addShaderFromSourceFile(QGLShader::Vertex, ":/vertexShader.vsh");
shaderProgram.addShaderFromSourceFile(QGLShader::Fragment, ":/fragmentShader.fsh");
shaderProgram.link();
vertices << QVector3D(1, 0, -2) << QVector3D(0, 1, -2) << QVector3D(-1, 0, -2);
}
和把
qDebug() << shaderProgram.log();
exit(0);
在调用上面的 link() 之后立即不会产生任何输出。
刚刚看到这个。 很奇怪,因为link()
返回 true,但它声称程序没有链接?
结果我需要在我的link()
调用之前移动两个bindAttributeLocation()
语句。 显然,如果在程序链接后使用bindAttributeLocation()
,它将导致程序不再链接。
错误的:
program.link();
program.bindAttributeLocation("position", 0);
program.bindAttributeLocation("texcoord", 1);
对:
program.bindAttributeLocation("position", 0);
program.bindAttributeLocation("texcoord", 1);
program.link();
// 在类构造函数中初始化了 shaderProgram
如果您在类构造函数中编译和链接着色器源文件,那么这很可能是罪魁祸首。 您应该覆盖 QGLWidget 的 initializeGL 成员并在那里对 OpenGL 对象进行初始化。 由于尚未建立 OpenGL 上下文,因此您无法更早地执行此操作,因此您无法创建或管理任何 OpenGL 对象。
我也发现了这个问题,我只是使用FromSourceCode
而不是FromSourceFile
像这样:
shaderProgram.addShaderFromSourceCode(QGLShader::Fragment, "uniform vec4 color;\n"
"out vec4 fragColor;\n"
"void main(void)\n"
"{\n"
" fragColor = color;\n"
"}");
声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.