-
Notifications
You must be signed in to change notification settings - Fork 125
/
beerlambert_color.c
151 lines (132 loc) · 5.12 KB
/
beerlambert_color.c
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
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
#include "svpng.inc"
#include <math.h> // fabsf(), fminf(), fmaxf(), sinf(), cosf(), sqrt()
#include <stdlib.h> // rand(), RAND_MAX
#define TWO_PI 6.28318530718f
#define W 512
#define H 512
#define N 256
#define MAX_STEP 64
#define MAX_DISTANCE 5.0f
#define EPSILON 1e-6f
#define BIAS 1e-4f
#define MAX_DEPTH 5
#define BLACK { 0.0f, 0.0f, 0.0f }
typedef struct { float r, g, b; } Color;
typedef struct { float sd, reflectivity, eta; Color emissive, absorption; } Result;
unsigned char img[W * H * 3];
Color colorAdd(Color a, Color b) {
Color c = { a.r + b.r, a.g + b.g, a.b + b.b };
return c;
}
Color colorMultiply(Color a, Color b) {
Color c = { a.r * b.r, a.g * b.g, a.b * b.b };
return c;
}
Color colorScale(Color a, float s) {
Color c = { a.r * s, a.g * s, a.b * s };
return c;
}
float circleSDF(float x, float y, float cx, float cy, float r) {
float ux = x - cx, uy = y - cy;
return sqrtf(ux * ux + uy * uy) - r;
}
float planeSDF(float x, float y, float px, float py, float nx, float ny) {
return (x - px) * nx + (y - py) * ny;
}
float ngonSDF(float x, float y, float cx, float cy, float r, float n) {
float ux = x - cx, uy = y - cy, a = TWO_PI / n;
float t = fmodf(atan2f(uy, ux) + TWO_PI, a), s = sqrtf(ux * ux + uy * uy);
return planeSDF(s * cosf(t), s * sinf(t), r, 0.0f, cosf(a * 0.5f), sinf(a * 0.5f));
}
Result unionOp(Result a, Result b) {
return a.sd < b.sd ? a : b;
}
Result scene(float x, float y) {
Result a = { circleSDF(x, y, 0.5f, -0.2f, 0.1f), 0.0f, 0.0f, { 10.0f, 10.0f, 10.0f }, BLACK };
Result b = { ngonSDF(x, y, 0.5f, 0.5f, 0.25f, 5.0f), 0.0f, 1.5f, BLACK, { 4.0f, 4.0f, 1.0f} };
return unionOp(a, b);
}
void gradient(float x, float y, float* nx, float* ny) {
*nx = (scene(x + EPSILON, y).sd - scene(x - EPSILON, y).sd) * (0.5f / EPSILON);
*ny = (scene(x, y + EPSILON).sd - scene(x, y - EPSILON).sd) * (0.5f / EPSILON);
}
void reflect(float ix, float iy, float nx, float ny, float* rx, float* ry) {
float idotn2 = (ix * nx + iy * ny) * 2.0f;
*rx = ix - idotn2 * nx;
*ry = iy - idotn2 * ny;
}
int refract(float ix, float iy, float nx, float ny, float eta, float* rx, float* ry) {
float idotn = ix * nx + iy * ny;
float k = 1.0f - eta * eta * (1.0f - idotn * idotn);
if (k < 0.0f)
return 0; // Total internal reflection
float a = eta * idotn + sqrtf(k);
*rx = eta * ix - a * nx;
*ry = eta * iy - a * ny;
return 1;
}
float fresnel(float cosi, float cost, float etai, float etat) {
float rs = (etat * cosi - etai * cost) / (etat * cosi + etai * cost);
float rp = (etai * cosi - etat * cost) / (etai * cosi + etat * cost);
return (rs * rs + rp * rp) * 0.5f;
}
Color beerLambert(Color a, float d) {
Color c = { expf(-a.r * d), expf(-a.g * d), expf(-a.b * d) };
return c;
}
Color trace(float ox, float oy, float dx, float dy, int depth) {
float t = 1e-3f;
float sign = scene(ox, oy).sd > 0.0f ? 1.0f : -1.0f;
for (int i = 0; i < MAX_STEP && t < MAX_DISTANCE; i++) {
float x = ox + dx * t, y = oy + dy * t;
Result r = scene(x, y);
if (r.sd * sign < EPSILON) {
Color sum = r.emissive;
if (depth < MAX_DEPTH && r.eta > 0.0f) {
float nx, ny, rx, ry, refl = r.reflectivity;
gradient(x, y, &nx, &ny);
float s = 1.0f / (nx * nx + ny * ny);
nx *= sign * s;
ny *= sign * s;
if (r.eta > 0.0f) {
if (refract(dx, dy, nx, ny, sign < 0.0f ? r.eta : 1.0f / r.eta, &rx, &ry)) {
float cosi = -(dx * nx + dy * ny);
float cost = -(rx * nx + ry * ny);
refl = sign < 0.0f ? fresnel(cosi, cost, r.eta, 1.0f) : fresnel(cosi, cost, 1.0f, r.eta);
refl = fmaxf(fminf(refl, 1.0f), 0.0f);
sum = colorAdd(sum, colorScale(trace(x - nx * BIAS, y - ny * BIAS, rx, ry, depth + 1), 1.0f - refl));
}
else
refl = 1.0f; // Total internal reflection
}
if (refl > 0.0f) {
reflect(dx, dy, nx, ny, &rx, &ry);
sum = colorAdd(sum, colorScale(trace(x + nx * BIAS, y + ny * BIAS, rx, ry, depth + 1), refl));
}
}
return colorMultiply(sum, beerLambert(r.absorption, t));
}
t += r.sd * sign;
}
Color black = BLACK;
return black;
}
Color sample(float x, float y) {
Color sum = BLACK;
for (int i = 0; i < N; i++) {
float a = TWO_PI * (i + (float)rand() / RAND_MAX) / N;
sum = colorAdd(sum, trace(x, y, cosf(a), sinf(a), 0));
}
return colorScale(sum, 1.0f / N);
}
int main() {
unsigned char* p = img;
for (int y = 0; y < H; y++)
for (int x = 0; x < W; x++, p += 3) {
Color c = sample((float)x / W, (float)y / H);
p[0] = (int)(fminf(c.r * 255.0f, 255.0f));
p[1] = (int)(fminf(c.g * 255.0f, 255.0f));
p[2] = (int)(fminf(c.b * 255.0f, 255.0f));
}
svpng(fopen("beerlambert_color.png", "wb"), W, H, img, 0);
}