Shader stages

The function for each stage has a predefined name, so we recommend you don't change it.

  • HSMain for hull shader
  • HSConstantMain for patch constant function
  • DSMain for domain shader
  • VSMain for vertex shader (takes no arguments)
  • GSMain for geometry shader
  • PSMain for pixel shader (takes no arguments)
  • CSMain for compute shader (takes no arguments)These are all void methods.

The geometry and tessellation shaders need some kind of predefined structure as input and output. However, since Xenko shaders are generic, it's impossible to know beforehand what the structure will be. As a result, these shaders use Input and Output structures that are automatically generated to fit the final shader.

Vertex shader

A vertex shader has to set the variable with the semantic SV_Position. In ShaderBase, it's ShadingPosition.

For example:

  1. override stage void VSMain()
  2. {
  3. ...
  4. streams.ShadingPosition = ...;
  5. ...
  6. }

Pixel shader

A pixel shader has to set the variable with the semantic SV_Target. In ShaderBase, it is ColorTarget.

For example:

  1. override stage void PSMain()
  2. {
  3. ...
  4. streams.ColorTarget = ...;
  5. ...
  6. }

Geometry shader

An example of geometry shader:

  1. [maxvertexcount(1)]
  2. void GSMain(triangle Input input[3], inout PointStream<Output> pointStream)
  3. {
  4. ...
  5. // fill the streams object
  6. streams = input[0];
  7. ...
  8. // always append streams
  9. pointStream.Append(streams);
  10. ...
  11. }

Input can be used in the method body. It behaves like the streams object and contains the same members.

Output is only used in the declaration of the method. You should append the streams object to your geometry shader output stream.

Tessellation shader

An example of a tessellation shader:

  1. [domain("tri")]
  2. [partitioning("fractional_odd")]
  3. [outputtopology("triangle_cw")]
  4. [outputcontrolpoints(3)]
  5. [patchconstantfunc("HSConstantMain")]
  6. [maxtessfactor(48.0)]
  7. void HSMain(InputPatch<Input, 3> input, out Output output, uint uCPID : SV_OutputControlPointID)
  8. {
  9. ...
  10. output = streams;
  11. }
  12. void HSConstantMain(InputPatch<Input, 3> input, const OutputPatch<Input2, 3> output, out Constants constants)
  13. {
  14. ...
  15. output = streams;
  16. ...
  17. }
  18. [domain("tri")]
  19. void DSMain(const OutputPatch<Input, 3> input, out Output output, in Constants constants, float3 f3BarycentricCoords : SV_DomainLocation)
  20. {
  21. ...
  22. output = streams;
  23. ...
  24. }

Input and Input2 both behave like streams.

Note

Don't forget to assign output to streams at the end of your stage.

Compute shader

An example of a compute shader:

  1. [numthreads(2, 3, 5)]
  2. void CSMain()
  3. {
  4. ...
  5. }

You can inherit from ComputeShaderBase and override the Compute method.

See also