diff --git a/DX11Starter.vcxproj b/DX11Starter.vcxproj
index 5ab53c1..89cca91 100644
--- a/DX11Starter.vcxproj
+++ b/DX11Starter.vcxproj
@@ -182,12 +182,24 @@
5.0
5.0
+
+ Pixel
+ Pixel
+ Pixel
+ Pixel
+
Pixel
Pixel
Pixel
Pixel
+
+ Vertex
+ Vertex
+ Vertex
+ Vertex
+
Pixel
Pixel
@@ -321,7 +333,9 @@
+
+
diff --git a/DX11Starter.vcxproj.filters b/DX11Starter.vcxproj.filters
index ad68647..1005a28 100644
--- a/DX11Starter.vcxproj.filters
+++ b/DX11Starter.vcxproj.filters
@@ -133,6 +133,12 @@
Shaders
+
+ Shaders
+
+
+ Shaders
+
@@ -381,5 +387,11 @@
Shaders
+
+ Shaders
+
+
+ Shaders
+
\ No newline at end of file
diff --git a/Helpers.hlsli b/Helpers.hlsli
index 4e9a634..ff263fa 100644
--- a/Helpers.hlsli
+++ b/Helpers.hlsli
@@ -4,6 +4,12 @@
// from environment map demo
static const float F0_NON_METAL = 0.04f;
+// Minimum roughness for when spec distribution function denominator goes to zero
+static const float MIN_ROUGHNESS = 0.0000001f; // 6 zeros after decimal
+
+// Handy to have this as a constant
+static const float PI = 3.14159265359f;
+
// gets view vector, needed once per shader
float3 getView(float3 cameraPosition, float3 pixelWorldPosition)
{
diff --git a/HelpersPBR.hlsli b/HelpersPBR.hlsli
new file mode 100644
index 0000000..da7acfe
--- /dev/null
+++ b/HelpersPBR.hlsli
@@ -0,0 +1,104 @@
+#ifndef __SHADER_HELPERS_PBR__
+#define __SHADER_HELPERS_PBR__
+
+// Lambert diffuse BRDF - Same as the basic lighting diffuse calculation!
+// - NOTE: this function assumes the vectors are already NORMALIZED!
+float DiffusePBR(float3 normal, float3 dirToLight)
+{
+ return saturate(dot(normal, dirToLight));
+}
+
+// Calculates diffuse amount based on energy conservation
+//
+// diffuse - Diffuse amount
+// specular - Specular color (including light color)
+// metalness - surface metalness amount
+//
+// Metals should have an albedo of (0,0,0)...mostly
+// See slide 65: http://blog.selfshadow.com/publications/s2014-shading-course/hoffman/s2014_pbs_physics_math_slides.pdf
+float3 DiffuseEnergyConserve(float3 diffuse, float3 specular, float metalness)
+{
+ return diffuse * ((1 - saturate(specular)) * (1 - metalness));
+}
+
+// GGX (Trowbridge-Reitz)
+//
+// a - Roughness
+// h - Half vector
+// n - Normal
+//
+// D(h, n) = a^2 / pi * ((n dot h)^2 * (a^2 - 1) + 1)^2
+float SpecDistribution(float3 n, float3 h, float roughness)
+{
+ // Pre-calculations
+ float NdotH = saturate(dot(n, h));
+ float NdotH2 = NdotH * NdotH;
+ float a = roughness * roughness;
+ float a2 = max(a * a, MIN_ROUGHNESS); // Applied after remap!
+
+ // ((n dot h)^2 * (a^2 - 1) + 1)
+ float denomToSquare = NdotH2 * (a2 - 1) + 1;
+ // Can go to zero if roughness is 0 and NdotH is 1; MIN_ROUGHNESS helps here
+
+ // Final value
+ return a2 / (PI * denomToSquare * denomToSquare);
+}
+
+// Fresnel term - Schlick approx.
+//
+// v - View vector
+// h - Half vector
+// f0 - Value when l = n (full specular color)
+//
+// F(v,h,f0) = f0 + (1-f0)(1 - (v dot h))^5
+float3 Fresnel(float3 v, float3 h, float3 f0)
+{
+ // Pre-calculations
+ float VdotH = saturate(dot(v, h));
+
+ // Final value
+ return f0 + (1 - f0) * pow(1 - VdotH, 5);
+}
+
+// Geometric Shadowing - Schlick-GGX (based on Schlick-Beckmann)
+// - k is remapped to a / 2, roughness remapped to (r+1)/2
+//
+// n - Normal
+// v - View vector
+//
+// G(l,v)
+float GeometricShadowing(float3 n, float3 v, float roughness)
+{
+ // End result of remapping:
+ float k = pow(roughness + 1, 2) / 8.0f;
+ float NdotV = saturate(dot(n, v));
+
+ // Final value
+ return NdotV / (NdotV * (1 - k) + k);
+}
+
+// Microfacet BRDF (Specular)
+//
+// f(l,v) = D(h)F(v,h)G(l,v,h) / 4(n dot l)(n dot v)
+// - part of the denominator are canceled out by numerator (see below)
+//
+// D() - Spec Dist - Trowbridge-Reitz (GGX)
+// F() - Fresnel - Schlick approx
+// G() - Geometric Shadowing - Schlick-GGX
+float3 MicrofacetBRDF(float3 n, float3 l, float3 v, float roughness, float3 specColor)
+{
+ // Other vectors
+ float3 h = normalize(v + l);
+
+ // Grab various functions
+ float D = SpecDistribution(n, h, roughness);
+ float3 F = Fresnel(v, h, specColor);
+ float G = GeometricShadowing(n, v, roughness) * GeometricShadowing(n, l, roughness);
+
+ // Final formula
+ // Denominator dot products partially canceled by G()!
+ // See page 16: http://blog.selfshadow.com/publications/s2012-shading-course/hoffman/s2012_pbs_physics_math_notes.pdf
+ return (D * F * G) / (4 * max(dot(n, v), dot(n, l)));
+}
+
+#endif
\ No newline at end of file
diff --git a/LightsPBR.hlsli b/LightsPBR.hlsli
new file mode 100644
index 0000000..5acd89c
--- /dev/null
+++ b/LightsPBR.hlsli
@@ -0,0 +1,29 @@
+#ifndef __SHADER_LIGHTS_PBR__
+#define __SHADER_LIGHTS_PBR__
+
+// Gets the RGB value of a pixel with a directional light
+float3 directionalLightPBR(Light light, float3 normal, float3 view, float roughness, float metalness, float3 surfaceColor, float3 specularColor)
+{
+ float3 lightDirection = normalize(light.Direction);
+ float diffuse = DiffusePBR(normal, -lightDirection);
+ float3 specular = MicrofacetBRDF(normal, lightDirection, view, roughness, specularColor);
+
+ float3 balancedDiff = DiffuseEnergyConserve(diffuse, specular, metalness);
+
+ return (balancedDiff * surfaceColor + specular) * light.Intensity * light.Color;
+}
+
+// Gets the RGB value of a pixel with a point light
+float3 pointLightPBR(Light light, float3 normal, float3 view, float roughness, float metalness, float3 surfaceColor, float3 specularColor, float3 worldPosition)
+{
+ float3 lightDirection = normalize(light.Position - worldPosition);
+ float attenuation = getAttenuation(light.Position, worldPosition, light.Range);
+ float diffuse = DiffusePBR(normal, lightDirection);
+ float3 specular = MicrofacetBRDF(normal, lightDirection, view, roughness, specularColor);
+
+ float3 balancedDiff = DiffuseEnergyConserve(diffuse, specular, metalness);
+
+ return (balancedDiff * surfaceColor + specular) * attenuation * light.Intensity * light.Color;
+}
+
+#endif
diff --git a/SimplePixelPBR.hlsl b/SimplePixelPBR.hlsl
new file mode 100644
index 0000000..913f2fd
--- /dev/null
+++ b/SimplePixelPBR.hlsl
@@ -0,0 +1,63 @@
+#include "Defines.hlsli"
+#include "Helpers.hlsli"
+#include "HelpersPBR.hlsli"
+#include "Lights.hlsli"
+#include "LightsPBR.hlsli"
+
+#define MAX_LIGHTS 128
+
+cbuffer ExternalData : register(b0)
+{
+ float2 offset;
+ float2 scale;
+
+ float3 cameraPosition;
+ float lightCount;
+
+ Light lights[MAX_LIGHTS];
+}
+
+Texture2D Albedo : register(t0);
+Texture2D Normal : register(t1);
+Texture2D Roughness : register(t2);
+Texture2D Metalness : register(t3);
+TextureCube Reflection : register(t4);
+SamplerState Sampler : register(s0);
+
+float4 main(VertexToPixel input) : SV_TARGET
+{
+ input.normal = normalize(input.normal);
+ input.tangent = normalize(input.tangent);
+
+ float3 view = normalize(cameraPosition - input.worldPosition);
+
+ float4 albedo = pow(Albedo.Sample(Sampler, input.uv), 2.2f);
+
+ float3 N = Normal.Sample(Sampler, input.uv).rgb * 2 - 1;
+ float3 T = normalize(input.tangent - input.normal * dot(input.tangent, input.normal));
+ float3 B = cross(T, input.normal);
+ float3x3 TBN = float3x3(T, B, input.normal);
+ input.normal = mul(N, TBN);
+
+ float roughness = Roughness.Sample(Sampler, input.uv).r;
+ float metalness = Metalness.Sample(Sampler, input.uv).r;
+ float3 specular = lerp(F0_NON_METAL.rrr, albedo.rgb, metalness);
+
+ float3 light = albedo;
+ for (int i = 0; i < lightCount; i++)
+ {
+ switch (lights[i].Type)
+ {
+ case LIGHT_TYPE_DIRECTIONAL:
+ light += directionalLightPBR(lights[i], input.normal, view, roughness, metalness, albedo, specular);
+ break;
+ case LIGHT_TYPE_POINT:
+ light += pointLightPBR(lights[i], input.normal, view, roughness, metalness, albedo, specular, input.worldPosition);
+ break;
+ }
+ }
+
+ float3 final = light;
+
+ return float4(pow(final, 1.0f / 2.2f), albedo.a);
+}
\ No newline at end of file
diff --git a/SimpleVertexPBR.hlsl b/SimpleVertexPBR.hlsl
new file mode 100644
index 0000000..f1c97c0
--- /dev/null
+++ b/SimpleVertexPBR.hlsl
@@ -0,0 +1,24 @@
+#include "Defines.hlsli"
+
+cbuffer ExternalData : register(b0)
+{
+ matrix world;
+ matrix worldInvTranspose;
+ matrix view;
+ matrix projection;
+}
+
+VertexToPixel main(VertexShaderInput input)
+{
+ VertexToPixel output;
+
+ matrix worldViewProjection = mul(projection, mul(view, world));
+ output.screenPosition = mul(worldViewProjection, float4(input.localPosition, 1.0f));
+
+ output.uv = input.uv;
+ output.normal = normalize(mul((float3x3)worldInvTranspose, input.normal));
+ output.tangent = normalize(mul((float3x3)worldInvTranspose, input.tangent));
+ output.worldPosition = mul(world, float4(input.localPosition, 1)).xyz;
+
+ return output;
+}