Shaders

You may need to read the introduction article before this one.


The GPU is in charge of processing graphics-related code, so we need some way of providing code to the GPU. We supply this code in the form of functions called shaders. In WebGL, there are two types of shaders, each with a specific purpose:

Shader Programs

One vertex shader and one fragment shader can be linked together to create a shader program, which can be used to render primitives. The exact process for rendering a primitive with a shader program is as follows:

  1. The vertex shader runs a certain amount of times depending on what type of primitive is being rendered. Each time it runs, it computes the position of a vertex in clip space, which is a coordinate system that represents the canvas in the range [1,1][-1, 1] on each axis. The negative boundaries for each direction are left, down, and near, respectively. Clip space coordinates are also sometimes called normalized device coordinates.
  2. Once enough vertices have had their positions computed to assemble a primitive, the primitive assembly stage does so.
  3. In the rasterization stage, the primitive is mapped to fragments. Fragments that are outside of clip space are clipped (discarded).
  4. The fragment shader runs once for each remaining fragment. Each time it runs, it computes the color of the fragment in color space, which organizes each of the red, green, blue, and alpha values of the color in the range [0,1][0, 1].
  5. Various tests are performed on the fragments, such as the depth and stencil tests, as well as blending. More on all of those in a later article.

In other versions of OpenGL, there is another type of shader called a geometry shader that runs between the vertex shader and the primitive assembly stage. Although geometry shaders do not exist in WebGL, their functionality can be emulated somewhat with the gl_VertexID variable in the vertex shader.

GLSL

WebGL shaders are written in a strictly-typed language called GLSL (OpenGL Shading Language). The specification for GLSL ES 3.00 (the version used by WebGL2) can be found here. In order to specify that a shader should use GLSL ES 3.00, its first line must be #version 300 es. If this is not the first line in a WebGL2 shader, it will not compile.

Every shader defines a function called main that is executed when the shader is executed. This function is responsible for setting the return value of the shader. Both types of shaders return vec4 values, which are vectors that contain four floating-point values each. In order, the four values are referred to as either xyzw, rgba (for colors), or stpq (for textures).

For vertex shaders, the return value is set with a special variable called gl_Position.

#version 300 es

void main() {
	gl_Position = vec4(0, 0, 0, 1);
}

Fragment shaders can have multiple return values, so output variables must be defined using the out keyword. The fragments returned by a fragment shader constitute a bitmap image that is stored in a portion of contiguous memory called a framebuffer. A framebuffer can have multiple regions that hold color data called color buffers. Each color buffer corresponds to an output variable in a fragment shader. The draw buffer is the color buffer that contains the fragments that are rendered to the canvas. By default, the draw buffer is the first color buffer, so the first output variable in the fragment shader will contain the color that is applied to the canvas.

#version 300 es

precision mediump float;

out vec4 outColor;

void main() {
	outColor = vec4(0, 0, 0, 1);
}

The line precision mediump float; in the fragment shader sets the precision of floating-point values. It is not necessary in the vertex shader because the vertex shader has a default precision for floating-point values.


The WebGL API can be used to compile shaders. First, a WebGLShader object must be created with createShader. Then, it must be assigned source code with shaderSource and compiled with compileShader. Optionally, the shader's compile status can be checked with getShaderParameter and logged with getShaderInfoLog.

const vertexShaderSource = `\
#version 300 es

void main() {
	gl_Position = vec4(0, 0, 0, 1);
}`;

const vertexShader = gl.createShader(gl.VERTEX_SHADER);
gl.shaderSource(vertexShader, vertexShaderSource);
gl.compileShader(vertexShader);
if (!gl.getShaderParameter(vertexShader, gl.COMPILE_STATUS)) {
	throw new Error(gl.getShaderInfoLog(vertexShader));
}

The process for creating a shader program with the WebGL API is very similar to the process for creating a shader. First, a WebGLProgram must be created with createProgram. Then, it must be assigned a vertex shader and a fragment shader with attachShader and linked together with linkProgram. Optionally, the shader program's link status can be checked with getProgramParameter and logged with getProgramInfoLog.

const program = gl.createProgram();
gl.attachShader(program, vertexShader);
gl.attachShader(program, fragmentShader);
gl.linkProgram(program);
if (!gl.getProgramParameter(program, gl.LINK_STATUS)) {
	throw new Error(gl.getProgramInfoLog(program));
}

The next article is about program structure.