mirror of
https://github.com/PixarAnimationStudios/OpenSubdiv
synced 2025-01-03 13:41:06 +00:00
344 lines
12 KiB
HLSL
344 lines
12 KiB
HLSL
|
//
|
||
|
// Copyright (C) Pixar. All rights reserved.
|
||
|
//
|
||
|
// This license governs use of the accompanying software. If you
|
||
|
// use the software, you accept this license. If you do not accept
|
||
|
// the license, do not use the software.
|
||
|
//
|
||
|
// 1. Definitions
|
||
|
// The terms "reproduce," "reproduction," "derivative works," and
|
||
|
// "distribution" have the same meaning here as under U.S.
|
||
|
// copyright law. A "contribution" is the original software, or
|
||
|
// any additions or changes to the software.
|
||
|
// A "contributor" is any person or entity that distributes its
|
||
|
// contribution under this license.
|
||
|
// "Licensed patents" are a contributor's patent claims that read
|
||
|
// directly on its contribution.
|
||
|
//
|
||
|
// 2. Grant of Rights
|
||
|
// (A) Copyright Grant- Subject to the terms of this license,
|
||
|
// including the license conditions and limitations in section 3,
|
||
|
// each contributor grants you a non-exclusive, worldwide,
|
||
|
// royalty-free copyright license to reproduce its contribution,
|
||
|
// prepare derivative works of its contribution, and distribute
|
||
|
// its contribution or any derivative works that you create.
|
||
|
// (B) Patent Grant- Subject to the terms of this license,
|
||
|
// including the license conditions and limitations in section 3,
|
||
|
// each contributor grants you a non-exclusive, worldwide,
|
||
|
// royalty-free license under its licensed patents to make, have
|
||
|
// made, use, sell, offer for sale, import, and/or otherwise
|
||
|
// dispose of its contribution in the software or derivative works
|
||
|
// of the contribution in the software.
|
||
|
//
|
||
|
// 3. Conditions and Limitations
|
||
|
// (A) No Trademark License- This license does not grant you
|
||
|
// rights to use any contributor's name, logo, or trademarks.
|
||
|
// (B) If you bring a patent claim against any contributor over
|
||
|
// patents that you claim are infringed by the software, your
|
||
|
// patent license from such contributor to the software ends
|
||
|
// automatically.
|
||
|
// (C) If you distribute any portion of the software, you must
|
||
|
// retain all copyright, patent, trademark, and attribution
|
||
|
// notices that are present in the software.
|
||
|
// (D) If you distribute any portion of the software in source
|
||
|
// code form, you may do so only under this license by including a
|
||
|
// complete copy of this license with your distribution. If you
|
||
|
// distribute any portion of the software in compiled or object
|
||
|
// code form, you may only do so under a license that complies
|
||
|
// with this license.
|
||
|
// (E) The software is licensed "as-is." You bear the risk of
|
||
|
// using it. The contributors give no express warranties,
|
||
|
// guarantees or conditions. You may have additional consumer
|
||
|
// rights under your local laws which this license cannot change.
|
||
|
// To the extent permitted under your local laws, the contributors
|
||
|
// exclude the implied warranties of merchantability, fitness for
|
||
|
// a particular purpose and non-infringement.
|
||
|
//
|
||
|
|
||
|
struct OutputPointVertex {
|
||
|
float4 positionOut : SV_Position;
|
||
|
};
|
||
|
|
||
|
// ---------------------------------------------------------------------------
|
||
|
// Vertex Shader
|
||
|
// ---------------------------------------------------------------------------
|
||
|
|
||
|
void vs_main( in InputVertex input,
|
||
|
out OutputVertex output )
|
||
|
{
|
||
|
output.positionOut = mul(ModelViewProjectionMatrix, input.position);
|
||
|
output.position = mul(ModelViewMatrix, input.position);
|
||
|
output.normal = mul(ModelViewMatrix,float4(input.normal, 0)).xyz;
|
||
|
}
|
||
|
|
||
|
// ---------------------------------------------------------------------------
|
||
|
// Geometry Shader
|
||
|
// ---------------------------------------------------------------------------
|
||
|
|
||
|
OutputVertex
|
||
|
outputVertex(OutputVertex input, float3 normal)
|
||
|
{
|
||
|
OutputVertex v = input;
|
||
|
v.normal = normal;
|
||
|
return v;
|
||
|
}
|
||
|
|
||
|
#if defined(GEOMETRY_OUT_WIRE) || defined(GEOMETRY_OUT_LINE)
|
||
|
#ifdef PRIM_TRI
|
||
|
#define EDGE_VERTS 3
|
||
|
#endif
|
||
|
#ifdef PRIM_QUAD
|
||
|
#define EDGE_VERTS 4
|
||
|
#endif
|
||
|
|
||
|
static float VIEWPORT_SCALE = 1024.0; // XXXdyu
|
||
|
|
||
|
float edgeDistance(float2 p, float2 p0, float2 p1)
|
||
|
{
|
||
|
return VIEWPORT_SCALE *
|
||
|
abs((p.x - p0.x) * (p1.y - p0.y) -
|
||
|
(p.y - p0.y) * (p1.x - p0.x)) / length(p1.xy - p0.xy);
|
||
|
}
|
||
|
|
||
|
OutputVertex
|
||
|
outputWireVertex(OutputVertex input, float3 normal,
|
||
|
int index, float2 edgeVerts[EDGE_VERTS])
|
||
|
{
|
||
|
OutputVertex v = input;
|
||
|
v.normal = normal;
|
||
|
|
||
|
v.edgeDistance[0] =
|
||
|
edgeDistance(edgeVerts[index], edgeVerts[0], edgeVerts[1]);
|
||
|
v.edgeDistance[1] =
|
||
|
edgeDistance(edgeVerts[index], edgeVerts[1], edgeVerts[2]);
|
||
|
#ifdef PRIM_TRI
|
||
|
v.edgeDistance[2] =
|
||
|
edgeDistance(edgeVerts[index], edgeVerts[2], edgeVerts[0]);
|
||
|
#endif
|
||
|
#ifdef PRIM_QUAD
|
||
|
v.edgeDistance[2] =
|
||
|
edgeDistance(edgeVerts[index], edgeVerts[2], edgeVerts[3]);
|
||
|
v.edgeDistance[3] =
|
||
|
edgeDistance(edgeVerts[index], edgeVerts[3], edgeVerts[0]);
|
||
|
#endif
|
||
|
|
||
|
return v;
|
||
|
}
|
||
|
#endif
|
||
|
|
||
|
[maxvertexcount(6)]
|
||
|
void gs_quad( lineadj OutputVertex input[4],
|
||
|
inout TriangleStream<OutputVertex> triStream )
|
||
|
{
|
||
|
float3 A = (input[0].position - input[1].position).xyz;
|
||
|
float3 B = (input[3].position - input[1].position).xyz;
|
||
|
float3 C = (input[2].position - input[1].position).xyz;
|
||
|
|
||
|
float3 n0 = normalize(cross(B, A));
|
||
|
|
||
|
triStream.Append(outputVertex(input[0], n0));
|
||
|
triStream.Append(outputVertex(input[1], n0));
|
||
|
triStream.Append(outputVertex(input[3], n0));
|
||
|
triStream.RestartStrip();
|
||
|
triStream.Append(outputVertex(input[3], n0));
|
||
|
triStream.Append(outputVertex(input[1], n0));
|
||
|
triStream.Append(outputVertex(input[2], n0));
|
||
|
triStream.RestartStrip();
|
||
|
}
|
||
|
|
||
|
#if defined(GEOMETRY_OUT_WIRE) || defined(GEOMETRY_OUT_LINE)
|
||
|
#ifdef PRIM_QUAD
|
||
|
[maxvertexcount(6)]
|
||
|
void gs_quad_wire( lineadj OutputVertex input[4],
|
||
|
inout TriangleStream<OutputVertex> triStream )
|
||
|
{
|
||
|
float3 A = (input[0].position - input[1].position).xyz;
|
||
|
float3 B = (input[3].position - input[1].position).xyz;
|
||
|
float3 C = (input[2].position - input[1].position).xyz;
|
||
|
|
||
|
float3 n0 = normalize(cross(B, A));
|
||
|
|
||
|
float2 edgeVerts[4];
|
||
|
edgeVerts[0] = input[0].positionOut.xy / input[0].positionOut.w;
|
||
|
edgeVerts[1] = input[1].positionOut.xy / input[1].positionOut.w;
|
||
|
edgeVerts[2] = input[2].positionOut.xy / input[2].positionOut.w;
|
||
|
edgeVerts[3] = input[3].positionOut.xy / input[3].positionOut.w;
|
||
|
|
||
|
triStream.Append(outputWireVertex(input[0], n0, 0, edgeVerts));
|
||
|
triStream.Append(outputWireVertex(input[1], n0, 1, edgeVerts));
|
||
|
triStream.Append(outputWireVertex(input[3], n0, 3, edgeVerts));
|
||
|
triStream.RestartStrip();
|
||
|
triStream.Append(outputWireVertex(input[3], n0, 3, edgeVerts));
|
||
|
triStream.Append(outputWireVertex(input[1], n0, 1, edgeVerts));
|
||
|
triStream.Append(outputWireVertex(input[2], n0, 2, edgeVerts));
|
||
|
triStream.RestartStrip();
|
||
|
}
|
||
|
#endif
|
||
|
#endif
|
||
|
|
||
|
[maxvertexcount(3)]
|
||
|
void gs_triangle( triangle OutputVertex input[3],
|
||
|
inout TriangleStream<OutputVertex> triStream )
|
||
|
{
|
||
|
float3 A = (input[0].position - input[1].position).xyz;
|
||
|
float3 B = (input[2].position - input[1].position).xyz;
|
||
|
|
||
|
float3 n0 = normalize(cross(B, A));
|
||
|
|
||
|
triStream.Append(outputVertex(input[0], n0));
|
||
|
triStream.Append(outputVertex(input[1], n0));
|
||
|
triStream.Append(outputVertex(input[2], n0));
|
||
|
}
|
||
|
|
||
|
[maxvertexcount(3)]
|
||
|
void gs_triangle_smooth( triangle OutputVertex input[3],
|
||
|
inout TriangleStream<OutputVertex> triStream )
|
||
|
{
|
||
|
triStream.Append(outputVertex(input[0], input[0].normal));
|
||
|
triStream.Append(outputVertex(input[1], input[1].normal));
|
||
|
triStream.Append(outputVertex(input[2], input[2].normal));
|
||
|
}
|
||
|
|
||
|
#if defined(GEOMETRY_OUT_WIRE) || defined(GEOMETRY_OUT_LINE)
|
||
|
#ifdef PRIM_TRI
|
||
|
[maxvertexcount(3)]
|
||
|
void gs_triangle_wire( triangle OutputVertex input[3],
|
||
|
inout TriangleStream<OutputVertex> triStream )
|
||
|
{
|
||
|
float3 A = (input[0].position - input[1].position).xyz;
|
||
|
float3 B = (input[2].position - input[1].position).xyz;
|
||
|
|
||
|
float3 n0 = normalize(cross(B, A));
|
||
|
|
||
|
float2 edgeVerts[3];
|
||
|
edgeVerts[0] = input[0].positionOut.xy / input[0].positionOut.w;
|
||
|
edgeVerts[1] = input[1].positionOut.xy / input[1].positionOut.w;
|
||
|
edgeVerts[2] = input[2].positionOut.xy / input[2].positionOut.w;
|
||
|
|
||
|
triStream.Append(outputWireVertex(input[0], n0, 0, edgeVerts));
|
||
|
triStream.Append(outputWireVertex(input[1], n0, 1, edgeVerts));
|
||
|
triStream.Append(outputWireVertex(input[2], n0, 2, edgeVerts));
|
||
|
}
|
||
|
|
||
|
[maxvertexcount(3)]
|
||
|
void gs_triangle_smooth_wire( triangle OutputVertex input[3],
|
||
|
inout TriangleStream<OutputVertex> triStream )
|
||
|
{
|
||
|
float2 edgeVerts[3];
|
||
|
edgeVerts[0] = input[0].positionOut.xy / input[0].positionOut.w;
|
||
|
edgeVerts[1] = input[1].positionOut.xy / input[1].positionOut.w;
|
||
|
edgeVerts[2] = input[2].positionOut.xy / input[2].positionOut.w;
|
||
|
|
||
|
triStream.Append(outputWireVertex(input[0], input[0].normal, 0, edgeVerts));
|
||
|
triStream.Append(outputWireVertex(input[1], input[1].normal, 1, edgeVerts));
|
||
|
triStream.Append(outputWireVertex(input[2], input[2].normal, 2, edgeVerts));
|
||
|
}
|
||
|
#endif
|
||
|
#endif
|
||
|
|
||
|
[maxvertexcount(1)]
|
||
|
void gs_point( point OutputVertex input[1],
|
||
|
inout PointStream<OutputPointVertex> pointStream )
|
||
|
{
|
||
|
OutputPointVertex v0;
|
||
|
v0.positionOut = input[0].positionOut;
|
||
|
|
||
|
pointStream.Append(v0);
|
||
|
}
|
||
|
|
||
|
// ---------------------------------------------------------------------------
|
||
|
// Lighting
|
||
|
// ---------------------------------------------------------------------------
|
||
|
|
||
|
#define NUM_LIGHTS 2
|
||
|
|
||
|
struct LightSource {
|
||
|
float4 position;
|
||
|
float4 ambient;
|
||
|
float4 diffuse;
|
||
|
float4 specular;
|
||
|
};
|
||
|
|
||
|
cbuffer Lighting : register( b2 ) {
|
||
|
LightSource lightSource[NUM_LIGHTS];
|
||
|
};
|
||
|
|
||
|
float4
|
||
|
lighting(float3 Peye, float3 Neye)
|
||
|
{
|
||
|
float4 color = float4(0.0, 0.0, 0.0, 0.0);
|
||
|
//float4 material = float4(0.4, 0.4, 0.8, 1);
|
||
|
float4 material = float4(0.13, 0.13, 0.61, 1); // sRGB (gamma 2.2)
|
||
|
|
||
|
for (int i = 0; i < NUM_LIGHTS; ++i) {
|
||
|
|
||
|
float4 Plight = lightSource[i].position;
|
||
|
|
||
|
float3 l = (Plight.w == 0.0)
|
||
|
? normalize(Plight.xyz) : normalize(Plight.xyz - Peye);
|
||
|
|
||
|
float3 n = normalize(Neye);
|
||
|
float3 h = normalize(l + float3(0,0,1)); // directional viewer
|
||
|
|
||
|
float d = max(0.0, dot(n, l));
|
||
|
float s = pow(max(0.0, dot(n, h)), 500.0f);
|
||
|
|
||
|
color += lightSource[i].ambient * material
|
||
|
+ d * lightSource[i].diffuse * material
|
||
|
+ s * lightSource[i].specular;
|
||
|
}
|
||
|
|
||
|
|
||
|
color.a = 1.0;
|
||
|
return color;
|
||
|
}
|
||
|
|
||
|
// ---------------------------------------------------------------------------
|
||
|
// Pixel Shader
|
||
|
// ---------------------------------------------------------------------------
|
||
|
|
||
|
float4
|
||
|
edgeColor(float4 Cfill, float4 edgeDistance)
|
||
|
{
|
||
|
#if defined(GEOMETRY_OUT_WIRE) || defined(GEOMETRY_OUT_LINE)
|
||
|
#ifdef PRIM_TRI
|
||
|
float d =
|
||
|
min(edgeDistance[0], min(edgeDistance[1], edgeDistance[2]));
|
||
|
#endif
|
||
|
#ifdef PRIM_QUAD
|
||
|
float d =
|
||
|
min(min(edgeDistance[0], edgeDistance[1]),
|
||
|
min(edgeDistance[2], edgeDistance[3]));
|
||
|
#endif
|
||
|
float4 Cedge = float4(1.0, 1.0, 0.0, 1.0);
|
||
|
float p = exp2(-2 * d * d);
|
||
|
|
||
|
#if defined(GEOMETRY_OUT_WIRE)
|
||
|
if (p < 0.25) discard;
|
||
|
#endif
|
||
|
|
||
|
Cfill.rgb = lerp(Cfill.rgb, Cedge.rgb, p);
|
||
|
#endif
|
||
|
return Cfill;
|
||
|
}
|
||
|
|
||
|
// ---------------------------------------------------------------------------
|
||
|
// Pixel Shader
|
||
|
// ---------------------------------------------------------------------------
|
||
|
|
||
|
void
|
||
|
ps_main( in OutputVertex input,
|
||
|
bool isFrontFacing : SV_IsFrontFace,
|
||
|
out float4 colorOut : SV_Target )
|
||
|
{
|
||
|
float3 N = (isFrontFacing ? input.normal : -input.normal);
|
||
|
colorOut = edgeColor(lighting(input.position.xyz, N), input.edgeDistance);
|
||
|
}
|
||
|
|
||
|
void
|
||
|
ps_main_point( in OutputPointVertex input,
|
||
|
out float4 colorOut : SV_Target )
|
||
|
{
|
||
|
colorOut = float4(1, 1, 1, 1);
|
||
|
}
|