Hxsl

    Vertex shaders are used for transforming and projecting each geometry point into 2D space and set up “variables” that will be interpolated on a per-pixel basis. This can be used by the pixel shader.

    Fragment shaders are used for blending different textures and colors into a single pixel color that will be written to screen.

    Example:

    This shader defines:

    • an variable (tagged with @input metadata) which lists the vertex properties that will be accessible in the shader (vertex attributes in GLSL)
    • several @param variables which are per-shader-instance values that will be set for all vertexes (uniforms in GLSL)
    • several variables which can be read/written in shader, here we use output as a name to store each shader stage result and transformedNormal as the value of the vertex normal in absolute world space.
    • @input are vertex attributes input values
    • @global are defined for multiple shaders
    • are constants that will produce a different shader output if changed
    • @var are varying variables that are written in vertex shader and read in pixel shader (optional, hxsl can infer this)
    • @borrow(path.to.other.Shader) are foreign uniform references that were declared in another shader which are usually not accessible outside of it (note that borrowed shader has to be added to the shader list, otherwise runtime shader compilation error will occur).
    • normal untagged variables are “pipeline variables” which can be either shader local vars, varying our output variables depending on the pipeline analysis (see below)

    Unlike other shader models (GLSL, HLSL, etc.), HXSL allows several shaders to be linked together at runtime, allowing you to split your shader into several distinct effects that can be enabled/disabled easily, without having to maintain a single “Uber Shader” with all the possible combinations.

    For instance, this is an shader that will affect all normals based on the previous shader:

    1. class MyEffect extends hxsl.Shader {
    2. static var SRC = {
    3. var transformedNormal : Vec3;
    4. transformedNormal.x *= -1.;
    5. };
    6. }

    The effect can be added to each material list of shaders.

    HXSL will then compile this list into a single shader instance, based on a desired list of outputs.

    After we know the list of the shaders and the desired output, HXSL will be linking the shaders together, ensuring that all variables read have been previously written (for instance MyEffect requires a transformedNormal variable that has been written by another shader in the pipeline).

    Once linking is finished, HXSL will be optimizing out all the unused parts, such as:

    • branches implying variables are optimized out when they are unreachable
    • unused variables, unused textures, operations not performing any side effect that affects one of the outputs

    The final resulting shader will then be compiled to the target platform native shader language (HXSL currently supports GLSL for OpenGL, HLSL for DirectX 11+, AGAL for AdobeAir, and PSSL for Sony PS4).

    Shader examples

    There are several shader examples available in the h3d.shader package.

    • for all 2D display
    • BaseMesh for all 3D display