1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
|
/******************************************************************/
/* This file is part of the homework assignments for CSCI-427/527 */
/* at The College of William & Mary and authored by Pieter Peers. */
/* No part of this file, whether altered or in original form, can */
/* be distributed or used outside the context of CSCI-427/527 */
/* without consent of either the College of William & Mary or */
/* Pieter Peers. */
/******************************************************************/
#include <cmath>
#include "constants.h"
#include "diffuseBrdf.h"
//////////////////
// Constructors //
//////////////////
diffuseBrdf::diffuseBrdf(const color& albedo)
{
_albedo = albedo;
}
diffuseBrdf::diffuseBrdf(const diffuseBrdf& src)
{
_albedo = src._albedo;
}
///////////////
// Operators //
///////////////
diffuseBrdf& diffuseBrdf::operator=(const diffuseBrdf& src)
{
_assign(src);
return *this;
}
/////////////
// Methods //
/////////////
color diffuseBrdf::shade(const vec3d& in, const vec3d& out) const
{
if(in.z < 0.0f || out.z < 0.0f) return color();
else return _albedo * in.z;
}
color diffuseBrdf::reflectance(const vec3d& in, const vec3d& out) const
{
if(in.z < 0.0f || out.z < 0.0f) return color(0.0f);
return _albedo / PI;
}
brdfSample diffuseBrdf::sample(const vec3d& in, float r1, float r2) const
{
// check if below surface.
if(in.z < EPSILON) return brdfSample();
// sample hemisphere proportional to the cosine
vec3d out( cos(2.0f * PI * r1) * sqrt(std::max(1.0f - r2, 0.0f)),
sin(2.0f * PI * r1) * sqrt(std::max(1.0f - r2, 0.0f)),
sqrt(r2) );
// set pdf
float pdf = out.z / PI;
// Done.
return brdfSample(out,
out.z / PI, // PDF proportional to cosine
_albedo / PI // reflectance
);
}
bool diffuseBrdf::isSpecular(void) const
{
return false;
}
bool diffuseBrdf::isDiffuse(void) const
{
return true;
}
float diffuseBrdf::reflectivity(void) const
{
return (_albedo.r + _albedo.g + _albedo.b) / 3.0f;
}
/////////////////////
// Private Methods //
/////////////////////
void diffuseBrdf::_assign(const diffuseBrdf& src)
{
_albedo = src._albedo;
}
void diffuseBrdf::_swap(diffuseBrdf& src)
{
swap(_albedo, src._albedo);
}
void diffuseBrdf::_print(std::ostream& s) const
{
s << "Diffuse BRDF: albedo=" << _albedo;
}
|