Shading language
Introduction
Godot uses a simplified shader language (almost a subset of GLSL). Shaders can be used for:
- Materials
- Post-Processing
- 2D
and are divided in Vertex, Fragment and Light sections.
Language
Typing
The language is statically type and supports only a few operations. Arrays, classes, structures, etc are not supported. Several built-in datatypes are provided:
Data types
DataType | Description |
---|---|
void | Void |
bool | boolean (true or false) |
float | floating point |
vec2 | 2-component vector, float subindices (x,y or r,g ) |
vec3 | 3-component vector, float subindices (x,y,z or r,g,b ) |
vec4, color | 4-component vector, float subindices (x,y,z,w or r,g,b,a ) |
mat2 | 2x2 matrix, vec3 subindices (x,y) |
mat3 | 3x3 matrix, vec3 subindices (x,y,z) |
mat4 | 4x4 matrix, vec4 subindices (x,y,z,w) |
texture | texture sampler, can only be used as uniform |
cubemap | cubemap sampler, can only be used as uniform |
Syntax
The syntax is similar to C, with statements ending with ;
and comments as //
and /* */
. Example:
float a = 3; vec3 b; b.x = a;
Swizzling
It is possible to use swizzling to reassigning subindices or groups of subindices, in order:
vec3 a = vec3(1,2,3); vec3 b = a.zyx; // b will contain vec3(3,2,1) vec2 c = a.xy; // c will contain vec2(1,2) vec4 d = a.xyzz; // d will contain vec4(1,2,3,3)
Constructors
Constructors take the regular amount of elements, but can also accept less if the element has more subindices, for example:
vec3 a = vec3(1,vec2(2,3)); vec3 b = vec3(a); vec3 c = vec3(vec2(2,3),1); vec4 d = vec4(a,5); mat3 m = mat3(a,b,c);
Conditionals
For now, only the if
conditional is supported. Example:
if (a < b) { c = b; }
Uniforms
A variable can be declared as uniform. In this case, its value will come from outside the shader (it will be the responsibility of the material or whatever using the shader to provide it).
uniform vec3 direction; uniform color tint; vec3 result = tint.rgb * direction;
Functions
Simple support for functions is provided. Functions can’t access uniforms or other shader variables.
vec3 addtwo(vec3 a, vec3 b) { return a+b; } vec3 c = addtwo(vec3(1,1,1), vec3(2,2,2));
Built-in functions
Several built-in functions are provided for convenience, listed as follows:
Function | Description |
---|---|
float sin ( float ) | Sine |
float cos ( float ) | Cosine |
float tan ( float ) | Tangent |
float asin ( float ) | arc-Sine |
float acos ( float ) | arc-Cosine |
float atan ( float ) | arc-Tangent |
vec_type pow ( vec_type, float ) | Power |
vec_type pow ( vec_type, vec_type ) | Power (Vec. Exponent) |
vec_type exp ( vec_type ) | Base-e Exponential |
vec_type log ( vec_type ) | Natural Logarithm |
vec_type sqrt ( vec_type ) | Square Root |
vec_type abs ( vec_type ) | Absolute |
vec_type sign ( vec_type ) | Sign |
vec_type floor ( vec_type ) | Floor |
vec_type trunc ( vec_type ) | Trunc |
vec_type ceil ( vec_type ) | Ceiling |
vec_type fract ( vec_type ) | Fractional |
vec_type mod ( vec_type,vec_type ) | Remainder |
vec_type min ( vec_type,vec_type ) | Minimum |
vec_type max ( vec_type,vec_type ) | Maximum |
vec_type clamp ( vec_type value,vec_type min, vec_type max ) | Clamp to Min-Max |
vec_type mix ( vec_type a,vec_type b, float c ) | Linear Interpolate |
vec_type mix ( vec_type a,vec_type b, vec_type c ) | Linear Interpolate (Vector Coef.) |
vec_type step ( vec_type a,vec_type b) | ` a[i] < b[i] ? 0.0 : 1.0` |
vec_type smoothstep ( vec_type a,vec_type b,vec_type c) | |
float length ( vec_type ) | Vector Length |
float distance ( vec_type, vec_type ) | Distance between vector. |
float dot ( vec_type, vec_type ) | Dot Product |
vec3 cross ( vec3, vec3 ) | Cross Product |
vec_type normalize ( vec_type ) | Normalize to unit length |
vec3 reflect ( vec3, vec3 ) | Reflect |
color tex ( texture, vec2 ) | Read from a texture in normalized coords |
color texcube ( texture, vec3 ) | Read from a cubemap |
vec3 texscreen ( vec2 ) | Read RGB from screen (generates a copy) |
Built-in variables
Depending on the shader type, several built-in variables are available, listed as follows:
Material (3D) - VertexShader
Variable | Description |
---|---|
const vec3 SRC_VERTEX | Model-Space Vertex |
const vec3 SRC_NORMAL | Model-Space Normal |
const vec3 SRC_TANGENT | Model-Space Tangent |
const float SRC_BINORMALF | Direction to Compute Binormal |
vec3 VERTEX | View-Space Vertex |
vec3 NORMAL | View-Space Normal |
vec3 TANGENT | View-Space Tangent |
vec3 BINORMAL | View-Space Binormal |
vec2 UV | UV |
vec2 UV2 | UV2 |
color COLOR | Vertex Color |
out vec4 VAR1 | Varying 1 Output |
out vec4 VAR2 | Varying 2 Output |
out float SPEC_EXP | Specular Exponent (for Vertex Lighting) |
out float POINT_SIZE | Point Size (for points) |
const mat4 WORLD_MATRIX | Object World Matrix |
const mat4 INV_CAMERA_MATRIX | Inverse Camera Matrix |
const mat4 PROJECTION_MATRIX | Projection Matrix |
const mat4 MODELVIEW_MATRIX | (InvCamera * Projection) |
const float INSTANCE_ID | Instance ID (for multimesh) |
const float TIME | Time (in seconds) |
Material (3D) - FragmentShader
Variable | Description |
---|---|
const vec3 VERTEX | View-Space vertex |
const vec4 POSITION | View-Space Position |
const vec3 NORMAL | View-Space Normal |
const vec3 TANGENT | View-Space Tangent |
const vec3 BINORMAL | View-Space Binormal |
const vec3 NORMALMAP | Alternative to NORMAL, use for normal texture output. |
const vec3 NORMALMAP_DEPTH | Complementary to the above, allows changing depth of normalmap. |
const vec2 UV | UV |
const vec2 UV2 | UV2 |
const color COLOR | Vertex Color |
const vec4 VAR1 | Varying 1 |
const vec4 VAR2 | Varying 2 |
const vec2 SCREEN_UV | Screen Texture Coordinate (for using with texscreen) |
const float TIME | Time (in seconds) |
const vec2 POINT_COORD | UV for point, when drawing point sprites. |
out vec3 DIFFUSE | Diffuse Color |
out vec4 DIFFUSE_ALPHA | Diffuse Color with Alpha (using this sends geometry to alpha pipeline) |
out vec3 SPECULAR | Specular Color |
out vec3 EMISSION | Emission Color |
out float SPEC_EXP | Specular Exponent (Fragment Version) |
out float GLOW | Glow |
out mat4 INV_CAMERA_MATRIX | Inverse camera matrix, can be used to obtain world coords (see example below). |
Material (3D) - LightShader
Variable | Description |
---|---|
const vec3 NORMAL | View-Space normal |
const vec3 LIGHT_DIR | View-Space Light Direction |
const vec3 EYE_VEC | View-Space Eye-Point Vector |
const vec3 DIFFUSE | Material Diffuse Color |
const vec3 LIGHT_DIFFUSE | Light Diffuse Color |
const vec3 SPECULAR | Material Specular Color |
const vec3 LIGHT_SPECULAR | Light Specular Color |
const float SPECULAR_EXP | Specular Exponent |
const vec1 SHADE_PARAM | Generic Shade Parameter |
const vec2 POINT_COORD | Current UV for Point Sprite |
out vec2 LIGHT | Resulting Light |
const float TIME | Time (in seconds) |
CanvasItem (2D) - VertexShader
Variable | Description |
---|---|
const vec2 SRC_VERTEX | CanvasItem space vertex. |
vec2 UV | UV |
out vec2 VERTEX | Output LocalSpace vertex. |
out vec2 WORLD_VERTEX | Output WorldSpace vertex. (use this or the one above) |
color COLOR | Vertex Color |
out vec4 VAR1 | Varying 1 Output |
out vec4 VAR2 | Varying 2 Output |
out float POINT_SIZE | Point Size (for points) |
const mat4 WORLD_MATRIX | Object World Matrix |
const mat4 EXTRA_MATRIX | Extra (user supplied) matrix via CanvasItem.draw_set_transform(). Identity by default. |
const mat4 PROJECTION_MATRIX | Projection Matrix (model coords to screen). |
const float TIME | Time (in seconds) |
const bool AT_LIGHT_PASS | Whether the shader is being run for a lighting pass (happens per affecting light) |
CanvasItem (2D) - FragmentShader
Variable | Description |
---|---|
const vec4 SRC_COLOR | Vertex color |
const vec4 POSITION | Screen Position |
vec2 UV | UV |
out color COLOR | Output Color |
out vec3 NORMAL | Optional Normal (used for 2D Lighting) |
out vec3 NORMALMAP | Optional Normal in standard normalmap format (flipped y and Z from 0 to 1) |
out float NORMALMAP_DEPTH | Depth option for above normalmap output, default value is 1.0 |
const texture TEXTURE | Current texture in use for CanvasItem |
const vec2 TEXTURE_PIXEL_SIZE | Pixel size for current 2D texture |
in vec4 VAR1 | Varying 1 Output |
in vec4 VAR2 | Varying 2 Output |
const vec2 SCREEN_UV | Screen Texture Coordinate (for using with texscreen) |
const vec2 POINT_COORD | Current UV for Point Sprite |
const float TIME | Time (in seconds) |
const bool AT_LIGHT_PASS | Whether the shader is being run for a lighting pass (happens per affecting light) |
CanvasItem (2D) - LightShader
Variable | Description |
---|---|
const vec4 POSITION | Screen Position |
in vec3 NORMAL | Input Normal |
in vec2 UV | UV |
in color COLOR | Input Color |
const texture TEXTURE | Current texture in use for CanvasItem |
const vec2 TEXTURE_PIXEL_SIZE | Pixel size for current 2D texture |
in vec4 VAR1 | Varying 1 Output |
in vec4 VAR2 | Varying 2 Output |
const vec2 SCREEN_UV | Screen Texture Coordinate (for using with texscreen) |
const vec2 POINT_COORD | Current UV for Point Sprite |
const float TIME | Time (in seconds) |
vec2 LIGHT_VEC | Vector from light to fragment, can be modified to alter shadow computation. |
const float LIGHT_HEIGHT | Height of Light |
const color LIGHT_COLOR | Color of Light |
const color LIGHT_SHADOW_COLOR | Color of Light shadow |
vec2 LIGHT_UV | UV for light image |
color SHADOW | Light shadow color override |
out vec4 LIGHT | Light Output (shader is ignored if this is not used) |
Examples
Material that reads a texture, a color and multiples them, fragment program:
uniform color modulate; uniform texture source; DIFFUSE = modulate.rgb * tex(source, UV).rgb;
Material that glows from red to white:
DIFFUSE = vec3(1,0,0) + vec3(1,1,1) * mod(TIME, 1.0);
Standard Blinn Lighting Shader
float NdotL = max(0.0, dot(NORMAL, LIGHT_DIR)); vec3 half_vec = normalize(LIGHT_DIR + EYE_VEC); float eye_light = max(dot(NORMAL, half_vec), 0.0); LIGHT = LIGHT_DIFFUSE + DIFFUSE + NdotL; if (NdotL > 0.0) { LIGHT += LIGHT_SPECULAR + SPECULAR + pow(eye_light, SPECULAR_EXP); };
Obtaining world-space normal and position in material fragment program:
// Use reverse multiply because INV_CAMERA_MATRIX is world2cam vec4 invcamx = INV_CAMERA_MATRIX.x; vec4 invcamy = INV_CAMERA_MATRIX.y; vec4 invcamz = INV_CAMERA_MATRIX.z; vec4 invcamw = INV_CAMERA_MATRIX.w; mat3 invcam = mat3(invcamx.xyz, invcamy.xyz, invcamz.xyz); vec3 world_normal = NORMAL * invcam; vec3 world_pos = (VERTEX - invcamw.xyz) * invcam;
Notes
- Do not use DIFFUSE_ALPHA unless you really intend to use transparency. Transparent materials must be sorted by depth and slow down the rendering pipeline. For opaque materials, just use DIFFUSE.
- Do not use DISCARD unless you really need it. Discard makes rendering slower, specially on mobile devices.
- TIME may reset after a while (may last an hour or so), it’s meant for effects that vary over time.
- In general, every built-in variable not used results in less shader code generated, so writing a single giant shader with a lot of code and optional scenarios is often not a good idea.
© 2014–2020 Juan Linietsky, Ariel Manzur and the Godot community
Licensed under the Creative Commons Attribution Unported License v3.0.
https://docs.godotengine.org/en/2.1/learning/features/shading/shading_language.html