Pixel Bender Shaders

The following document contains the source code for various Pixel Bender shaders I wrote. I learned most of these from Programming Vertex & Pixel Shaders (2004).

The shaders were initially published in a compiled form (as PBJ files - nice pun) sometime in Q1 2010 but their source code (*.PBK) is now here for archival purposes. The source code below can serve as a good starting point for learning pixel shader writing.

Brightness

<languageVersion : 1.0;>

kernel Brightness
<   namespace : "com.goldenxp";
    vendor : "goldenxp";
    version : 1;
    description : "Basic Brightness filter";
>
{
    input image4 src;
    output pixel4 dst;
    parameter float scale 
    <
        minValue:float(0);
        maxValue:float(2);
        defaultValue:float(1);
    >;

    void
    evaluatePixel()
    {
        float4x4 mBright = float4x4(
        scale, 0, 0, 0,
        0, scale, 0, 0,
        0, 0, scale, 0,
        0, 0, 0, 1        
        );
        float4 rgba = sampleNearest(src,outCoord());
        dst = mBright * rgba;
    }
}

Invert

<languageVersion : 1.0;>

kernel Invert
<   namespace : "com.goldenxp";
    vendor : "goldenxp";
    version : 1;
    description : "Basic invert filter";
>
{
    input image4 src;
    output pixel4 dst;

    void
    evaluatePixel()
    {
        float4 rgba = sampleNearest(src,outCoord());
        dst.rgb = float3(1.0,1.0,1.0)- rgba.rgb;
        dst.a = rgba.a;
    }
}

Edge Detect

<languageVersion : 1.0;>

kernel EdgeDetect
<   namespace : "com.goldenxp";
    vendor : "goldenxp";
    version : 1;
    description : "Edge Detection (Laplacian) filter";
>
{
    input image4 src;
    output pixel4 dst;

    void
    evaluatePixel()
    {        
        float2 px = pixelSize(src);
        float4 color = float4(0,0,0,0);
        float4 rgba = sampleNearest(src, outCoord());
        
        color += sampleNearest(src, outCoord() + float2( 0, px.y ) );
        color += sampleNearest(src, outCoord() + float2( 0, -px.y ) );
        color += sampleNearest(src, outCoord() + float2( px.x, 0 ) );
        color += sampleNearest(src, outCoord() + float2( -px.x, 0 ) );
        color -= sampleNearest(src, outCoord());
        color -= sampleNearest(src, outCoord());
        color -= sampleNearest(src, outCoord());
        color -= sampleNearest(src, outCoord());
     
        color.a = rgba.a;
        
        dst = color;
    }
}

Sharpen

<languageVersion : 1.0;>

kernel Sharpen
<   namespace : "com.goldenxp";
    vendor : "goldenxp";
    version : 1;
    description : "Sharpen filter";
>
{
    input image4 src;
    output pixel4 dst;

    void
    evaluatePixel()
    {        
        float2 px = pixelSize(src);        
        float4 color = float4(0,0,0,0);
        float4 rgba = sampleNearest(src, outCoord());
        
        color += sampleNearest(src, outCoord());
        color += sampleNearest(src, outCoord());
        color += sampleNearest(src, outCoord());
        color += sampleNearest(src, outCoord());
        color += sampleNearest(src, outCoord());
        color -= sampleNearest(src, outCoord() + float2( 0, px.y ) );
        color -= sampleNearest(src, outCoord() + float2( 0, -px.y ) );
        color -= sampleNearest(src, outCoord() + float2( px.x, 0 ) );
        color -= sampleNearest(src, outCoord() + float2( -px.x, 0 ) );
        color.rgb = clamp(float3(0),float3(1),color.rgb);
     
        color.a = rgba.a;
        
        dst = color;
    }
}

Desaturate

<languageVersion : 1.0;>

kernel Desaturation
<   namespace : "com.goldenxp";
    vendor : "goldenxp";
    version : 1;
    description : "Basic desaturation filter";
>
{
    input image4 src;
    output pixel4 dst;

    void
    evaluatePixel()
    {
        float4x4 mDesat = float4x4(
        0.33, 0.33, 0.33, 0,
        0.33, 0.33, 0.33, 0,
        0.33, 0.33, 0.33, 0,
        0, 0, 0, 1
        );
        float4 rgba = sampleNearest(src,outCoord());
        dst = mDesat * rgba;
    }
}

Saturate

<languageVersion : 1.0;>

kernel Saturate
<   namespace : "com.goldenxp";
    vendor : "goldenxp";
    version : 1;
    description : "Saturation control";
>
{
    input image4 src;
    output pixel4 dst;
    parameter float saturate
    <
        minValue:float(-1);
        maxValue:float(10);
        defaultValue:float(2);
    >;

    void
    evaluatePixel()
    {
        float red = 0.3086;
        float green = 0.6094;
        float blue = 0.0820;
        float s = saturate;
        
        float4x4 mSaturate = float4x4(
            (1.0-s) * red + s, (1.0-s)*red, (1.0-s)*red, 0,
            (1.0-s) * green, (1.0-s) * green + s, (1.0-s) * green, 0,
            (1.0-s) * blue, (1.0-s) * blue, (1.0-s) * blue + s, 0,
            0, 0, 0, 1
        );
        
        float4 rgba = sampleNearest(src,outCoord());
        dst = mSaturate * rgba;
    }
}

Edge Normals

<languageVersion : 1.0;>

kernel NormalMap
<   namespace : "com.goldenxp";
    vendor : "goldenxp";
    version : 1;
    description : "Edge Detection from Normals pass";
>
{
    input image4 normalmap;
    output pixel4 dst;

    void
    evaluatePixel()
    {
        float4 rgba = sampleNearest(normalmap,outCoord());
        rgba.rgb = 2.0 * rgba.rgb - 1.0;
        float2 px = pixelSize(normalmap);
        float outline = 0.0; 
        outline = clamp(1.0-dot(rgba.rgb, 2.0*sampleNearest(normalmap, outCoord() + float2(px.x,0)).rgb-1.0), 0.0, 1.0);
        outline += clamp(1.0-dot(rgba.rgb, 2.0*sampleNearest(normalmap, outCoord() + float2(-px.x,0)).rgb-1.0 ), 0.0, 1.0);
        outline += clamp(1.0-dot(rgba.rgb, 2.0*sampleNearest(normalmap, outCoord() + float2(0,px.y)).rgb-1.0), 0.0, 1.0);
        outline += clamp(1.0-dot(rgba.rgb, 2.0*sampleNearest(normalmap, outCoord() + float2(0,-px.y)).rgb-1.0), 0.0, 1.0);
        
        dst.rgb = float3(outline);
        dst.a = rgba.a;
    }
}

Normal Map

<languageVersion : 1.0;>

kernel NormalMap
<   namespace : "com.goldenxp";
    vendor : "goldenxp";
    version : 1;
    description : "Diffuse lighting from Normals pass";
>
{
    input image4 normalmap;
    output pixel4 dst;
    parameter float3 lightDirection
    <
    minValue:float3(-1);maxValue:float3(1);
    defaultValue:float3(0.4,0.5,0);
    >;
    parameter pixel3 dark<defaultValue:float3(0);>;
    parameter pixel3 light<defaultValue:float3(1);>;
    parameter float specular<minValue:0.0;maxValue:100.0;defaultValue:0.0;>;
    parameter float specular_intensity<minValue:0.0;maxValue:1.0;defaultValue:1.0;>;

    void
    evaluatePixel()
    {
        float4 rgba = sampleNearest(normalmap,outCoord());
        rgba.rgb = 2.0 * rgba.rgb - 1.0;
        float3 color = mix(dark,light,dot(lightDirection, rgba.rgb));
        if (specular != 0.0) {
            float spec = dot(lightDirection+float3(0,0,1),rgba.rgb);
            color +=  float3( specular_intensity * pow( spec , specular ) );
        }
        
        dst.rgb = color;
        dst.a = rgba.a;
    }
}

Color Remap

<languageVersion : 1.0;>

kernel ColorRemap
<   namespace : "com.goldenxp";
    vendor : "goldenxp";
    version : 1;
    description : "Color remapping filter";
>
{
    input image4 src;
    output pixel4 dst;
    parameter pixel3 color1< defaultValue:float3(0.3,0.2,0.1); >;
    parameter pixel3 color2< defaultValue:float3(1.0,0.5,0.3); >;

    void
    evaluatePixel()
    {
        float4 rgba = sampleNearest(src,outCoord());
       
        float3x3 mLuminance = float3x3(
        0.3086, 0.3086, 0.3086, 
        0.6094, 0.6094, 0.6094, 
        0.0820, 0.0820, 0.0820
        );        
        
        float3 col = mLuminance * rgba.rgb;
               
        dst.rgb = mix(color1, color2, col.r) ;
        dst.a = rgba.a;
    }
}

UV Remaps

<languageVersion : 1.0;>

kernel UVMap
<   namespace : "com.goldenxp";
    vendor : "goldenxp";
    version : 1;
    description : "UV Mapping (crude - no samples)";
>
{
    input image4 uv_pass;
    input image4 texture;
    parameter float sizeX <defaultValue:128.0; minValue:2.0; maxValue:2048.0;>;
    parameter float sizeY <defaultValue:128.0; minValue:2.0; maxValue:2048.0;>;
    output pixel4 dst;

    void
    evaluatePixel()
    {       
        float4 rgba = sampleNearest(texture, outCoord() );
        float2 px = pixelSize(uv_pass);
        float4 tex = sampleNearest(uv_pass, float2(sizeX*rgba.r, sizeY*rgba.g));
        dst.rgb = tex.rgb;
        dst.a = rgba.a;
    }
}

Anaglyph

<languageVersion : 1.0;>

kernel Anaglyph
<   namespace : "com.goldenxp";
    vendor : "goldenxp";
    version : 1;
    description : "Anaglyph Red/Cyan Stereo filter";
>
{
    input image4 leftImage;
    input image4 rightImage;
    output pixel4 dst;

    void
    evaluatePixel()
    {
        float4 left = sampleNearest(leftImage,outCoord());
        float4 right = sampleNearest(rightImage,outCoord());
        
        left = float4(left.r,0,0,left.a);
        right = float4(0,right.g,right.b,right.a);
        
        dst.rgb = left.rgb + right.rgb;
        dst.a = left.a;
    }
}

Template

This has never been published before since it’s just a template. BONUS!

<languageVersion : 1.0;>

kernel Nameless
<   namespace : "com.goldenxp";
    vendor : "goldenxp";
    version : 1;
    description : "nameless";
>
{
    input image4 leftImage;

    output pixel4 dst;

    void
    evaluatePixel()
    {
        float4 rgba = sampleNearest(leftImage,outCoord());
        

        dst = rgba;
    }
}