Shaders - Demo 20

Objective

Learn what “shaders” are, with a brief introduction to how to use them in OpenGL 2.1.

How to Execute

Load src/modelviewprojection/demo20/demo20.py in Spyder and hit the play button.

Move the Paddles using the Keyboard

Keyboard Input

Action

w

Move Left Paddle Up

s

Move Left Paddle Down

k

Move Right Paddle Down

i

Move Right Paddle Up

d

Increase Left Paddle’s Rotation

a

Decrease Left Paddle’s Rotation

l

Increase Right Paddle’s Rotation

j

Decrease Right Paddle’s Rotation

UP

Move the camera up, moving the objects down

DOWN

Move the camera down, moving the objects up

LEFT

Move the camera left, moving the objects right

RIGHT

Move the camera right, moving the objects left

q

Rotate the square around its center

e

Rotate the square around paddle 1’s center

Description

In the previous demo, in the footnote, we found that “glVertex” did a lot behind the scenes, sneakily grabbing data from the matrix stacks and the current color (and lots of other things we haven’t covered in this book, like texturing and lighting).

What did the graphics driver do with the data? This older style of graphics programming is called “fixed-function” graphics programming, in that we can tweak some values, but the graphics card and its driver are going to do whatever they were programmed to, and we can’t control it. Kind of like having a graphing calculator, but not having the ability to program it - OpenGL is in control at this point, and we can just parameterize it.

Programmers wanted more control, first over how lighting works (which we don’t cover in this book). Rather than having 3 or so lighting models to choose from, the programmers wanted to be able to specify the math of how their custom lighting worked. So OpenGL 2.1 allowed the programmer to create “shaders”, which I believe were called such because they allow the programmer to change the “shade” of a fragment [1].

In this demo, we keep most of the code from the previous demo the same, we just add in programmable shaders. So the calls to “glVertex” are no longer a black box, in which something happens of the graphics card; instead, before sending modelspace data to the graphics card, we program a sequence of transformations that happen on the graphics card per vector, which a driver could implement in parallel.

We have some new imports, “glUseProgram”, “GL_VERTEX_SHADER”, “GL_FRAGMENT_SHADER”

src/modelviewprojection/demo20/demo20.py
28import OpenGL.GL as GL
29import OpenGL.GL.shaders as shaders
30import OpenGL.GLU as GLU
31
32import modelviewprojection.colorutils as colorutils
33

We now ask the OpenGL driver to accept OpenGL 2.1 function calls.

src/modelviewprojection/demo20/demo20.py
41glfw.window_hint(glfw.CONTEXT_VERSION_MAJOR, 2)
42glfw.window_hint(glfw.CONTEXT_VERSION_MINOR, 1)

Before the event loop starts, we send two mini programs to the GPU, the vertex shader and the fragment shader. The purpose of the vertex shader is to transform the modelspace data to clip-space (x,y,z,w), where to get NDC you would divide by the w; i.e. (x/w,y/w,z/w,1)

I like to think of a matrix multiplication to a vector as a “function-application”, in the same way we did in the lambda stack.

src/modelviewprojection/demo20/demo20.py
202# compile shaders
203
204# initialize shaders
205pwd = os.path.dirname(os.path.abspath(__file__))
206
207with open(os.path.join(pwd, "triangle.vert"), "r") as f:
208    vs = shaders.compileShader(f.read(), GL.GL_VERTEX_SHADER)
209
210with open(os.path.join(pwd, "triangle.frag"), "r") as f:
211    fs = shaders.compileShader(f.read(), GL.GL_FRAGMENT_SHADER)
212
213shader = shaders.compileProgram(vs, fs)
214GL.glUseProgram(shader)

In the vertex shader, there are a lot of predefined variable names with values provided to us. “gl_Modelview_matrix” is the matrix at the top of the modelview matrix stack at the time “glVertex” is called, “glProjectionMatrix” is the top of the projection matrix stack (although there is typically only one). “glColor” is the color that was defined for this vector by calling “glColor3f”.

In OpenGL 2.1, the output of the vertex shader that we use here is “glFrontColor”, which is a variable name that we must use to get the data to the fragment shader, which we haven’t covered. (Actually there are a bunch of other predefined variable outputs, including “glBackColor”, for the case that we are looking at the back face of the geometry.

// Copyright (c) 2018-2025 William Emerison Six
//
// This program is free software; you can redistribute it and/or
// modify it under the terms of the GNU General Public License
// as published by the Free Software Foundation; either version 2
// of the License, or (at your option) any later version.
//
// This program is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
// GNU General Public License for more details.
//
// You should have received a copy of the GNU General Public License
// along with this program; if not, write to the Free Software
// Foundation, Inc., 59 Temple Place - Suite 330,
// Boston, MA 02111-1307, USA.

#version 120

void main(void)
{
    // gl_Vertex is modelspace data

    vec4 camera_space = gl_ModelViewMatrix * gl_Vertex;
    // in camera space, the frustum is on the negative z axis
    vec4 clip_space = gl_ProjectionMatrix * camera_space;
    vec3 ndc_space = clip_space.xyz / clip_space.w;
    // in ndc space, x y and z need to be divided by w

    // normal MVP transform
    gl_Position = gl_ModelViewProjectionMatrix * gl_Vertex;

    // Copy the primary color
    gl_FrontColor = gl_Color;
}

This is the fragment shader. Unlike the vertex shader, which took an vector in (x,y,z) and outputted clip-space (NDC), the fragment shader for a given pixel is determining the color [2]. Is “glColor” in the fragment shader the same “glColor” from the vertex shader, or is it the value of the vertex shader’s output “glFrontColor”? Honestly, the author doesn’t know, but it is set to the output of the fragment shader, and such questions are a moot point once we get to OpenGL 3.3 Core profile in the next section.

// Copyright (c) 2018-2025 William Emerison Six
//
// This program is free software; you can redistribute it and/or
// modify it under the terms of the GNU General Public License
// as published by the Free Software Foundation; either version 2
// of the License, or (at your option) any later version.
//
// This program is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
// GNU General Public License for more details.
//
// You should have received a copy of the GNU General Public License
// along with this program; if not, write to the Free Software
// Foundation, Inc., 59 Temple Place - Suite 330,
// Boston, MA 02111-1307, USA.

#version 120

void main(void)
{
    gl_FragColor = gl_Color;
}

The explicit mapping of variable names to values in OpenGL 3.3 Core Profile will make the flow of data from the CPU program, to the GPU vertex shader, to the fragment shader much more clear, albeit at the expense of verbosity.