Do that thing again!

Or

All sorts of pass properties


Here it comes

Other than defining a vertex and pixel program, your passes have all sorts of other functions they can perform. These properties are also called ”render states” or “pass attributes”. While each of these properties is potentially its own subject, they are hard to separate to individual chapters, and are better left tide to each other. Other than that, you can create multiple passes in each technique, allowing you to achieve more complex results.

In this chapter, we will overview some of the most commonly used properties and see how we can use them to enhance our rendering output.

The subjects

  1. Culling
  2. Color channels (color buffer)
  3. Blending (alpha and color)
  4. Z buffer (depth buffer)
  5. Multi-pass techniques and example usage (next chapter)


Most of the example usage is in the next chapter, so don’t be alarmed if you get confused; you might as well open the next chapter and read them simultaneously so you wont fall asleep while reading all the technical stuff.

Culling

Culling is the process of determining whether or not a polygon is visible or not when projected onto the screen before drawing it. In doing so, you can save GPU work and filter unwanted results while rendering.

When culling, you define which direction is “front” facing: clock-wise or counter clock-wise.
Whichever you choose, the other will be defined as the facing backwards. All polygons with that winding will fail the culling test, and thus will not be drawn.

We’ll take a sphere affected by Phong. If we choose that our cull mode is clock-wise, we will see what we usually would:
Image

What you can’t see, is that the polygons in the other side of the sphere were not drawn, since they have a counter clock-wise winding.

If we define cull mode as counter clock-wise, the result will be:
Image

Notice we can see the grid? This means that we are actually looking inside the sphere.

Code

As default culling is deactivated (By default cullMode = NONE)
If you want to activate it, add one of the following to your pass:

cullMode = cw;
//Or
cullMode = ccw;

In OGRE

Within your OGRE material script, you can set both hardware and software culling, read about them in the OGRE manual:

cull_hardware <clockwise | anticlockwise | none>
cull_software <clockwise | anticlockwise | none>


Color channels

This is an easy to understand, short and unsigned int... Wait no…Subject.
During a pass, we can control whether or not to write into the color buffer, and in which channel. Disabling color writing is usually not an operation you will do on regular basic, but at times, it can be of great use.

To control what channel we wish to write in, we call

colorWriteEnable = [value];


The values follow the color circle:
Image
Did you notice some of the diameters in the picture are not actually strait? I forgot I can draw the lines from side to side, and drew each radius alone. Looking at the bigger picture has never been my best quality.
Get it? Picture?

0 Off
1 Red
2 Green
3 Yellow
4 Blue
5 Purple
6 Light Blue
7 All on (white)

In OGRE

In an OGRE script, you can only command the colour_write on or off.
In fact, these are the only two values that you’ll ever use, but have a laugh, render a demon in pink in FXC.

colour_write <on|off>


And that’s it…
Usually, you will use this when you need to place things in the depth buffer but don’t wish to render at that particular pass.

Blending

What is?

Blending allows you to control how your pixel renders to the screen, in relation to pixels positioned behind it. As default, your pixel will overwrite the pixel behind it, which means it is completely opaque.

When blending, we use the following equation:

Color = sourcePixelColor x srcBlend + destPixelColor x DestBlend


The source is the pixel in the object we are blending with the scene (the one using this material)
The destination is the background it is blending with.
Both srcBlend and destBlend range from 1 (opaque) to 0 (transparent), normally, you will want the sum of srcBlend and destBlend to be 1 (each one will have a certain weight in the final color)

While in order to activate blending you use the “AlphaBlendEnable = true;” directive, you can blend either alpha channel or color channel.

Alpha blend vs color blend

So what is the difference between blending alpha and blending color? When you blend using alpha, the result will keep the original colors, but give each one a certain weight – effectively, mimicking transparency with different strengths. When you blend color, the result will be a different color.

If one of your pixels is green, and other is red, the resulting pixel will be yellow when color blending.

Values

Both source and destination can accept any of the following values:

ZERO Value of 0 – no weight in blending
ONE Value of 1 – the src/dest will be given full weight
SRCCOLOR The color channel of the source pixel
INVSRCCOLOR (1 – srcColor)
SRCALPHA Alpha channel of the source pixel
INVSRCALPHA Use ( 1 – srcAlpha)
DESTALPHA Alpha channel of the destination pixel
INVDESTALPHA ( 1 – destAlpha )
DESTCOLOR Color channel of the destination pixel
INVDESTCOLOR ( 1 – destColor)

To use blending in a pass, we need to add these 3 properties:

AlphaBlendEnable = true;
srcBlend = one;
destBlend = zero;



By default, the above example is how your pixel is rendered to the screen (in ‘as if’, not literal; by default no blending check occurs). When the source pixel receives 1 and the destination receives 0, the source pixel is completely opaque.
Image

Let’s use these values in the equation and see why:

Color 	= sourcePixelColor x srcBlend 	+ destPixelColor x DestBlend
Red	= Red 	 	    x 1		+ White		  X 0


Say our current pixel is red, and the destination pixel is white;

As you can see, white color from the destination pixel has no weight in the final output.

Let’s see other simple examples:

Completely transparent

If you swap the values of the default, your source will disappear completely from the scene, because it has no blending weight:

srcBlend = zero;
destBlend = one;

Image
It’s there, I swear it.

Blackness

If you give both src and dest zero value, the result will be a black silhouette of the src object, since if none of the two receive weight; the result will be 0, and 0 means black.

srcBlend = zero;
destBlend = zero;

Image
I can swallow your sun. Seriously

I want it all!

If you give both src and dest value of one, the result will be an additive blend.

srcBlend = one;
destBlend = one;

Image

By the way, I culled. If I hadn’t, I would have got:
Image

See the ugly Artifact in the bottom? Culling has a practical use you can actually see!

Common blending techniques

Now that we played a little with the values to get a better sense of what they mean, let’s blend like adults.

Pure alpha blending

In order to create an alpha blending we will give the source pixel its own alpha weight, and give the destination pixel the inverse of the source alpha.

srcBlend = srcAlpha;
destBlend = invSrcAlpha;

Image

In this example, my diffuse and specular both have alpha of one, making them opaque, while the ambient has an alpha of 0.5, making it half transparent in regions effected by ambient alone.

Why not give each one its own alpha value? Because we want the destination pixel to be obscured by the source pixel. If we were to give it its own weight, we would end up with a sort of multiplicative blend.
If we use this:

srcBlend = srcAlpha;
destBlend = destAlpha;

The result will be:
Image

By giving the destination pixel the opposite value of the source alpha, we make the two “complete” each other (unlike the second example), allowing us to control how opaque or transparent the source pixel is by changing its alpha. The more we lower the source’s alpha, the higher the destination’s become, and vice versa.

Pure color blending

Pure color blending works the same as alpha blending, but uses the color channels instead.

srcBlend = srcColor;
destBlend = invSrcColor;


Image
Pure color blend when ambient is complete black.

Image
…Same but with red ambient color. Notice how the color turns yellow in blended part.

Multiplicative and 2X multiplicative blending

One way multiplicative is defined as either:

srcBlend = zero;
destBlend = srcColor;

Or

srcBlend = destColor;
destBlend = zero;

The second one is the proper one, but the first one gives the same result.

In this method we use the ‘intensity’ of the source pixel, modulated by the destination pixel’s color.
Image

2X multiplicative is defined as:

srcBlend = destColor;
destBlend = srcColor;


In 2X we combine the intensity and the color of both source and destination.
Image

Of course, there are all sorts of different combinations, some completely illogical, others just not very useful. Play with the values; see what you can come up with.

If you use

srcblend = destalpha;
destblend = srcalpha;


The result will be one of my favorite stupidities:
Image

Blend in OGRE scripts

In OGRE, blend is also part of your pass. It works somewhat the same, but a few differences.

First thing: it’s called “Scene Blend”.

Your first options are the automated values:

scene_blend <add|modulate|alpha_blend|colour_blend>


These 4 options have an automated behavior for source and destination. 3 of them you already know (add, alpha_blend and colour_blend).
The fourth – modulate – is setup as follows:

srcblend = destColor;
destblend = zero;

And gives the following result:
Image

the OGRE manual wrote:

Generally colours and darkens the scene,
good for smoked glass,
semi-transparent objects
etc


Your second option is setting the equation manually, like we did in FXC:
scene_blend <src_factor> <dest_factor>

The values are spelled differently, but shouldn’t be foreign to you:

one
zero
dest_colour
src_colour
one_minus_dest_colour
one_minus_src_colour
dest_alpha
src_alpha
one_minus_dest_alpha
one_minus_src_alpha


If your logic organ is not operational, ‘one minus’ means inverse.

Depth buffer

What is?

The depth buffer is, in a simplified description, is where all pixels in the same spot in the view queue in line, for the renderer to pick the closest one to it. By default, all pixels both write themselves to the buffer, and test themselves against the rest of the queue when the frame is being prepared or rendered.

Control

The depth buffer has four control methods:
ZEnable = (true/false);
Whether or not the current pass will ignore the depth buffer or not, by default true. If false the pixel will always be rendered on top of all others (used in unit icons placed on billboards in many games). None of the other methods will take effect unless this method’s value is true.

ZWriteEnable = (true/false);
Whether or not the pixel will be written to the depth buffer for other pixels to test against
The result will usually not defer from ’ZEnable = false’ in first glance, but don’t be fooled, it’s very different.

DepthBias = (number);
This method provides the means to prevent Z fighting. Pixels affected by bias will “make way”. Used in most cases to ensure proper rendering of things such as shadows, projective decals or particle systems.


ZFunc = (options);
This method allows you to control in what manner the Z testing will behave. You can command it to which condition is the “win”: farther from the one you are testing, closer, or equal.

Less means closer to you
Greater farther away
Equal guess
NotEqual either less or greater…
Always I’m not answering you
Never the last statement did not change
LessEqual
GreaterEqual

It’s hard to explain their effect without examples, try each value, and see how it affects the scene. In multi-pass techniques (in the next chapter) we will see a practical implementation.

Depth in OGRE

The difference is almost negligible, text from the OGRE manual:

Methods

depth_check (on|off)
The ZEnable of OGRE. Actually, in most cases it is called z test or check.

depth_write (on|off)

depth_bias <constant_bias> <slopescale_bias>
The second one is optional. I’m not getting farther in for now, read about it in the manual if you like.

depth_func (func)
depth_func values, shouldn’t be foreign to you:

always_fail
always_pass
less
less_equal
equal
not_equal
greater_equal
greater