# PBR: Real-time Implementation
[Site du cours](https://davidpeicho.github.io/teaching/)
# Before we start
*Comment on genere une image ?*
- On a vu le raytracing
- On a vu la rasterization
:::success
On va se focus sur le temps reel avec de la rasterization
:::
# Old Times
## Lambert
data:image/s3,"s3://crabby-images/d68a3/d68a333fda94b08fa4f9f9ae55320f1b0ff33124" alt=""
On a tous fait un Lambert model
- Plus l'angle est eleve entre la normal et la lumiere, moins il n'y a d'energie
:::warning
Il n'y a pas de modele $100\%$ diffus
:::
:::success
En une seule operation on a notre BRDF
:::
> Il existe d'autres modeles mais la difference visuelle n'est pas assez bonne pour etre utilises
## Phong
data:image/s3,"s3://crabby-images/62bfa/62bfa6045949381c0cd39bde107cd551853a59af" alt=""
- Approximation pas tres bonne
- MAIS precurseur a son epoque ($'70s$)
Pas de conservation d'energie:
data:image/s3,"s3://crabby-images/1a5ac/1a5ac7cdcdb05f027da85dd1ef11dfe375fb3eed" alt=""
## Pseudocode
### Lambert
```glsl
void main()
{
vec3 diffuse = kD * dot(normal, lightDirection) * color;
gl_FragColor.rgba = vec4(diffuse, 1.0);
}
```
### Phong
```glsl
void main()
{
vec3 r = reflect(- viewDirection, normal);
vec3 diffuse = kD * dot(normal, lightDirection) * color;
vec3 specular = kS * pow(max(dot(lightDirection, r)), exponent);
gl_FragColor.rgba = vec4(diffuse + specular, 1.0);
}
```
# What and why
## Introduction
:::warning
Non-physical model requires a lot of tweaking
:::
> Si on a un artiste qui fait une scene en exterieur puis on lui dit qu'on doit aller dans un tunnel en voiture, l'artiste pleure
> Il doit *tweaker* les materiaux pour que ca ait l'air joli en fonction de la lumiere
:::success
C'est pour ca qu'il y a eu l'avenement du **Real Time Rendering** vers 2013
:::
:::warning
C'est dur de definir le PBR
:::
:::info
**Definition: PBR**
Modele mathematiques et approximations que nous allons tous suivre pour decrire les interactions entre la lumiere et la matiere
:::
## What is PBR ?
data:image/s3,"s3://crabby-images/1b203/1b20373693825a97e85ac551d66135b8119e508d" alt=""
*Pourquoi c'est populaire ?*
> Decrit le monde plus precisement, donne des rendus realistes
> Tout le monde utilise plus ou moins les memes inputs
> Moins de tweaking
:::success
Win-win pour les ingenieurs et artistes
:::
# Microfacets Theory
data:image/s3,"s3://crabby-images/e353d/e353df46f9bbe82e953521b6eeec3906fda743eb" alt=""
> Ce modele approxime ce qu'il se passe dans la vraie vie
:::info
On dit que tous les materiaux sont composes de miroirs plus ou moins alignes
:::
data:image/s3,"s3://crabby-images/f4cd1/f4cd1659186d47cf8508fc59684a0677a37ca370" alt=""
data:image/s3,"s3://crabby-images/0b081/0b08148256f27e3d0f138459fc195941db6ee963" alt=""
*C'est quoi la difference entre un miroir et un plastique ?*
> Notre premier cas sera un miroir
> Le second est un materiaux super diffus
# Dielectrics vs Conductors
data:image/s3,"s3://crabby-images/8c750/8c750a3d9e1f0b79fa1481df1a49c2c269ede122" alt=""
## Conductors
data:image/s3,"s3://crabby-images/8cf04/8cf04c4daf8a1a475c4229be2e0e7c9c0fa578f3" alt=""
La couleur diffuse serait une approximation du sub-surface scattering
data:image/s3,"s3://crabby-images/579f8/579f8ebc4328ef2a44b2119704c620518a6fec74" alt=""
- Les conducteurs reflete $0-20\%$ de la lumiere
:::warning
Les metaux n'ont pas de sub-surface scattering
:::
data:image/s3,"s3://crabby-images/d4913/d4913d8d059365b0ce15c6fc98e429895a2e1d78" alt=""
- Les conducteurs refletent $60-90\%$ de la lumiere
- Certains conducteurs ont leur couleur propre due aux longueurs d'ondes absorbees
# BRDF
## BRDF Simplification
$$
f_r(p,\omega_0,\omega_i)=f_d(p,\omega_0,\omega_i) + f_s(p,\omega_0, \omega_i)
$$
data:image/s3,"s3://crabby-images/64796/647961f4f75a7cdf61724eaaeffcb78c7b94c0db" alt=""
:::success
Notre BRDF devient pulg & play
:::
> On peut remplacer par ce qu'on veut du moment que $\int\le1$
## Implementation notes
$$
f_r(p,\omega_0,\omega_i)=k_df_d(p,\omega_0,\omega_i) + k_sf_s(p,\omega_0, \omega_i)\\
k_d+k_d\le1
$$
## Diffuse Lobe
$$
f_d(p,\omega_0,\omega_i) = \frac{\rho}{\pi}
$$
- $\rho$: reflectance spectrum
data:image/s3,"s3://crabby-images/60f4f/60f4f0d763c3826d6a6c8cf628fec98a03b2e3f0" alt=""
## Specular Lobe
$$
f_s(p,\omega_0,\omega_i)=\frac{D(\omega_0,\omega_i)F(\omega_0,\omega_i)G(\omega_0,\omega_i)}{4(\omega_0,\omega_i)(\omega_i\times n)}
$$
data:image/s3,"s3://crabby-images/e8c67/e8c67b9d0606e385460bb94de0b052f57aa2e8d7" alt=""
### Specular BRDF
$$
D_{GGX}(n,h,a)=\frac{\alpha^2}{\pi((n\times h)^2(\alpha^2-1)+1)^2}\\
\vec h=\frac{\vec v+\vec L}{\Vert\vec v+\vec L\Vert}
$$
- Normal distribution function $D(\omega_0, \omega_i)$
- Estimates the area of microfacets aligned to give perfect specular
- As usual, lots of different NDF equations...
- To be consistent, let's implement the Trowbridge-Reitz equation
- Low roughness means few samples contributing a lot to specular
data:image/s3,"s3://crabby-images/aba5b/aba5b02c772c092be3174aa921cbe4d365743247" alt=""
### Shadowing term $G(\omega_0,\omega_i)$
data:image/s3,"s3://crabby-images/39d1b/39d1b54db6661c0251451c470496f93a3e816de0" alt=""
$$
G(n,v,l,k)=\underbrace{G_{SchlickGGX}(n,v,k)}_{Obstruction}\underbrace{G_{Schlik}(n,l,k)}_{Shadowing}\\
G_{SchlickGGX}(n,v,k)=\frac{n\times v}{(n\times v)(1-k)+k}
$$
- On va approximer $k=\alpha$
- Approximation de l'occlusion
:::warning
L'orientation des facettes peut *pieger* la lumiere
:::
data:image/s3,"s3://crabby-images/37f33/37f33ff40e966969b3e6d16060a3d5529adfd562" alt=""
### Effet Fresnel
data:image/s3,"s3://crabby-images/4f981/4f9810fbe8a6943d0baf10e88441adbbbe896458" alt=""
> On a un joli coucher de soleil sur la mer (ou ocean)
> L'eau est un miroir modulo les vagues
:::info
Pour tout materiaux, la reflectance va etre maximale aux **angles rasants**
:::
:::danger
L'effet Fresnel c'est le poids du *specular lobe* $k_s$
:::
$$
F_{Schlik}(v,h,f_0,f_{90}) = f_0+(f_{90}-f_0)(1-v\times h)^5\\
F_{Schlik}(v,h,f_0) = f_0+(1-f_0)(1-v\times h)^5\\
F_0(ior)=\frac{(1-ior)^2}{(ior+1)^2}
$$
- $f_0$: base reflectivity at normal incidence
- $f_{90}$: base reflectivity at grazing angle
- Almost always 1 for conductors
data:image/s3,"s3://crabby-images/fcad0/fcad0868dd993bf6593bacb7bdff0bd707dad164" alt=""
data:image/s3,"s3://crabby-images/abecd/abecd622d36d08d1304efe152fd7da09b64cfb97" alt=""
> Fresnel reflectance for common materials
- For dialectics, $f_0$ is often approximated with $0.04$
- Some materials $f_0$ are tainted (gold, copper)
- Implementation note:
- For dielectics, pick $0.04$ $f_0$
- For conductors, store $f_0$ in albedo texture
- Use *metallic* input to lerp between the 2
## Demo !
data:image/s3,"s3://crabby-images/01a3c/01a3cffd44e9fe92bde1d1e5602af554d3e6547d" alt=""
### Direct-Lightning pseudocode
```glsl
vec3 radiance = vec3(0.0);
for(int i = 0; i < NB_LIGHTS; ++i)
{
vec3 w_i = lights[i].direction;
vec3 kS = FresnelShlick(f0, wi, w_o);
vec3 specularBRDFEval = kS * f_s(p, w_i, w_o);
vec3 diffuseBRDFEval = (1.0 - kS) * f_d(p, w_i, w_o);
radiance += (diffuseBRDFEval + specularBRDFEval) * sampleLight(lights[i], p, w_i) * dot(normal, w_i);
}
```
## Textures
data:image/s3,"s3://crabby-images/d601f/d601fd78b97dd62866d4211e361876516510b010" alt=""
> Les artistes font plusieurs textures
## To remember !
- Diffuse is an approximation of sub-surface scattering
- La plupart des moteurs connus vont avoir des *metallics workflow*
- Ca simplifie beaucoup la vie
# Ponctual light
# Point light
- Infinitely small
- Isotropic
- Describe only by a position
- Simple to code and fast to sample
- Power unit should be set using **Lumens**
- How to select a proper value ?
- Not as accurate as **Area Light**
$$
L_i(p,\omega_i)=\frac{\phi}{4\pi r^2}n\times\omega_i
$$
data:image/s3,"s3://crabby-images/8019d/8019da15084d17749f8ad164756d1a8b524c8d39" alt=""
:::warning
Cette lumiere n'existe pas dans la vraie vie
:::
### Note
- On ne va pas parler de directionnal light (deja fait)
- Si on utilise une directionnal light, il faudra tweaker les parametres
- Ce n'est pas aussi fidele que les **Area lights**
# Image Based Lightning
data:image/s3,"s3://crabby-images/e7209/e7209b7888ab4a415db3489278783a75fcdc43ee" alt=""
- 4 points lights
data:image/s3,"s3://crabby-images/ebcb0/ebcb0dce3dc43db3d35c8449c273ff17c42341f9" alt=""
- Avec environnement
$$
L_0(p,\omega_0)=\int_{\Omega}(f_d(p,\omega_0,\omega_i) + f_s(p,\omega_0,\omega_i))L_i(p,\omega_i)n\times w_i\\
L_0(p,\omega_0)=\int_{\Omega}f_d(p,\omega_0,\omega_i)L_i(p,\omega_i)m\times\omega_i+\int_{\Omega}f_s(p,\omega_0,\omega_i)L_i(p,\omega_i)m\times\omega_i
$$
## IBL Diffuse
$$
\int_{\Omega}f_d(p,\omega_0,\omega_i)L_i(p,\omega_i)m\times\omega_i
$$
data:image/s3,"s3://crabby-images/c94a1/c94a1c67c1924496d20a671083330733f22bbafd" alt=""
*Mais c'est juste un flou gaussien ?*
> C'est pas si faux que ca, c'est assez proche
$$
L_0(p,n)=\int_{\Omega}\frac{\rho}{\pi}L_i(p,\omega_i)n\times\omega_id\omega_i\\
L_0(p,n)=\frac{\rho}{\pi}\int_{\Omega}L_i(p,\omega_i)n\times\omega_id\omega_i\\
$$
Il faut faire une integration par angle solide, et c'est complique.
data:image/s3,"s3://crabby-images/6eed9/6eed99036df0219d3eede644b9735ec8f9778b93" alt=""
- Utilisation des coordonnees spheriques pour l'integration
- Discretiser l'integrale avec la somme de Riemann
- Calculer pour chaque texel, avec la direction $N$ du centre
## IBL Specular
$$
\int_{\Omega}f_s(p,\omega_0,\omega_i)L_i(p,\omega_i)m\times\omega_i
$$
data:image/s3,"s3://crabby-images/4402f/4402fe0cf7e409165f30d52816d30c9566649890" alt=""
$$
L_0(p,\omega_0)=\int_{\Omega}L_i(p,\omega_i)d\omega_i\times\int_{\Omega}f_r(p,\omega_0,\omega_i)n\times\omega_i d\omega_i
$$
data:image/s3,"s3://crabby-images/a8460/a84606c19adf73b9f602f778f20dbfc3eb38a92d" alt=""
> Ca a ete teste et ca marche: c'est ca la 3D
Changer le niveau de roughness c'est faire du downsampling, pourquoi par appliquer la roughness en faisant des images de plus en plus petites
data:image/s3,"s3://crabby-images/de973/de9731902c14f66af1ada297c83e0689da551fe0" alt=""
### Pre-computed BRDF
$$
\begin{aligned}
\int_{\Omega}f_r(p,\omega_0,\omega_i)n\times\omega_id\omega_i &= F_0\int_{\Omega}f_r(p,\omega_0,\omega_i)(1-(1-\omega_0\times h)^5)n\times\omega_id\omega_i\\
&+\int_{\Omega}f_r(p,\omega_0,\omega_i)(1-\omega_0\times h)^5n\times\omega_id\omega_i
\end{aligned}
$$
> Obtained bu substituting Fresnel Shlick
> Only 2 inputs left: roughness, viewing angle
data:image/s3,"s3://crabby-images/53454/53454f465964a876931c09ccef418f4bb5d75060" alt=""
At runtime:
1. Fetch pre-integrated BRDF texture
2. Fetch convoluted environment
3. Apply the above equation to get the full specular component
### Specular: compisistion
```glsl
c2 brdf = GetIntegratedBRDF(NdotV, roughness);
vec3 prefilteredSpecular = GetPrefilteredSpecular((NdotV, roughness);
vec3 specular = prefilteredSpecular * (F * brdf.x + brdf.y);
```
- `F`: Fresnel term
## To remember
- C'est juste du pre-filtering
# Colorspace and color precision
data:image/s3,"s3://crabby-images/94e99/94e994c4f450998ce996e3ef5df8925551380bd0" alt=""
- sRGB vs Linear
- Monitors apply pow function to luminance
- Toute l'industrie a du se base sur les ecran qui font ca donc ils ont cree le $sRGB$
- Sur photoshop, une image sera encodee en sRGB pour retrouver les couleurs imaginees
:::warning
On va eviter de faire nos calculs en sRGB
:::
- Soit on fait tout en sRGB
- Soit on fait tout en lineaire $\Rightarrow$ **OUI**
- On applique a la fin la fonction sRGB pour convertir en lineaire
## HDR vs LDR
data:image/s3,"s3://crabby-images/d9bc0/d9bc0338d5105454ed11f06ec968d42ea497ac9f" alt=""
:::info
HDR: High Dynamic Range
:::
Reinhard Tonemapping:
$$
color_{final}=\frac{c}{c+1}
$$
- HDR has larger range of values
- Units will create radiance color outside the $0\dots1$ range
- Perform computation in HDR, tonemap to LDR is required
- HDR is required to get correct PBR result
- Especially important for IBL
# Going further
## Advanced materials
data:image/s3,"s3://crabby-images/10be5/10be56e5448873ba09500f382498217fef5611ba" alt=""
> Examples: hair, skin, cloud, etc.