forked from githole/edubpt
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathrender.h
229 lines (192 loc) · 9.05 KB
/
render.h
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
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
#ifndef _RENDER_H_
#define _RENDER_H_
#include "reference_pathtracing.h"
#include "pathtracing.h"
#include "lighttracing.h"
#include "bidirectional_pathtracing.h"
#include "ppm.h"
#define USE_OPENMP
#ifdef USE_OPENMP
#include <omp.h>
#endif
#include <assert.h>
#include <iostream>
namespace edubpt {
//
// 各種レンダリングアルゴリズムを使って実際に画像をレンダリングする関数群
//
int render_by_refernce_pathtracing(const Camera &camera, const int num_threads) {
const int width = camera.image_width_px;
const int height = camera.image_height_px;
const int spp = camera.samples_per_pixel;
std::vector<Color> image_buffer(width * height);
#ifdef USE_OPENMP
omp_set_num_threads(num_threads);
#endif
for (int sample = 0; sample < spp; ++sample) {
std::cerr << "Rendering (" << (sample + 1) << " samples/pixel) " << (100.0 * (sample + 1) / spp) << "%" << std::endl;
#pragma omp parallel for schedule(dynamic, 1)
for (int y = 0; y < height; ++y) {
Random random(y * spp + sample);
for (int x = 0; x < width; ++x) {
Vec position_on_imagesensor, position_on_objectplane, position_on_lens;
double P_Image, P_lens;
camera.sample_points(x, y, &random, &position_on_imagesensor, &position_on_objectplane, &position_on_lens, &P_Image, &P_lens);
// 方向
const Vec dir = normalize(position_on_objectplane - position_on_lens);
// L(x0 <- x1) (= L(xI <- x0) = L(x0 <- xV))
const Color L = radiance_no_recursion(Ray(position_on_lens, dir), &random);
if (!is_valid_value(L)) // 異常値は除外する
continue;
const Vec x0_xI = position_on_imagesensor - position_on_lens;
const double coefficient = pow(dot(normalize(-1.0 * camera.imagesensor_dir), normalize(x0_xI)), 2.0) / x0_xI.length_squared(); // センサ上の点の半球積分を、レンズ上の点の積分に変数変換した時に導入される係数
image_buffer[y * width + x] = image_buffer[y * width + x] +
camera.W * L * coefficient / (P_Image * P_lens);
}
}
}
// サンプル数で割る + 左右反転
std::vector<Color> finale_buffer(width * height);
for (int y = 0; y < height; ++y)
for (int x = 0; x < width; ++x)
finale_buffer[y * width + x] = image_buffer[y * width + (width - x - 1)] / spp;
// 出力
save_ppm_file("image_ref_pt.ppm", &finale_buffer[0], width, height);
return 0;
}
int render_by_pathtracing(const Camera &camera, const int num_threads) {
const int width = camera.image_width_px;
const int height = camera.image_height_px;
const int spp = camera.samples_per_pixel;
std::vector<Color> image_buffer(width * height);
#ifdef USE_OPENMP
omp_set_num_threads(num_threads);
#endif
for (int sample = 0; sample < spp; ++sample) {
std::cerr << "Rendering (" << (sample + 1) << " samples/pixel) " << (100.0 * (sample + 1) / spp) << "%" << std::endl;
#pragma omp parallel for schedule(dynamic, 1)
for (int y = 0; y < height; ++y) {
Random random(y * spp + sample);
for (int x = 0; x < width; ++x) {
std::vector<Vertex> vertices;
const PathtracingResult result = generate_vertices_by_pathtracing(camera, x, y, &random, vertices);
if (result.is_light_hit) {
if (is_valid_value(result.value)) {
image_buffer[y * width + x] = image_buffer[y * width + x] + result.value;
}
}
}
}
}
// サンプル数で割る + 左右反転
std::vector<Color> finale_buffer(width * height);
for (int y = 0; y < height; ++y)
for (int x = 0; x < width; ++x)
finale_buffer[y * width + x] = image_buffer[y * width + (width - x - 1)] / spp;
// 出力
save_ppm_file("image_pt.ppm", &finale_buffer[0], width, height);
return 0;
}
void render_by_lighttracing(const Camera &camera, const int num_threads) {
const int width = camera.image_width_px;
const int height = camera.image_height_px;
const int spp = camera.samples_per_pixel;
std::vector<Color> image_buffer(width * height * num_threads);
#ifdef USE_OPENMP
omp_set_num_threads(num_threads);
#endif
const long long samples_per_thread = (long long)spp * width * height / num_threads;
std::cerr << "Average samples/pixel: " << (samples_per_thread * num_threads / (width * height)) << std::endl;
int thread_id = 0;
#pragma omp parallel private(thread_id)
{
#ifdef USE_OPENMP
thread_id = omp_get_thread_num();
#endif
Random random(thread_id * 32);
for (long long sample = 0; sample < samples_per_thread; ++sample) {
if (thread_id == 0 && sample % (width * height) == 0)
std::cerr << "Rendering (" << ((double)(num_threads * (sample + 1)) / (width * height)) << " samples/pixel) "
<< (100.0 * (double)((sample + 1)) / samples_per_thread) << "%" << std::endl;
std::vector<Vertex> vertices;
LighttracingResult result = generate_vertices_by_lighttracing(camera, &random, vertices);
if (result.is_lens_hit) {
if (is_valid_value(result.value)) {
const int x = result.imagebuffer_x, y = result.imagebuffer_y;
image_buffer[(thread_id * width * height) + y * width + x] = image_buffer[(thread_id * width * height) + y * width + x] + result.value;
}
}
}
}
// サンプル数で割る + 左右反転
// Lighttracingでは、光源からパスを生成した回数=各画素におけるモンテカルロ積分のサンプル数、になるため、
// 光源からパスを生成した回数で割ることになる。
std::vector<Color> final_buffer(width * height);
for (int t = 0; t < num_threads; ++t)
for (int y = 0; y < height; ++y)
for (int x = 0; x < width; ++x) {
final_buffer[y * width + x] = final_buffer[y * width + x] +
image_buffer[(t * width * height) + y * width + (width - x - 1)] / ((double)samples_per_thread * num_threads);
}
// 出力
save_ppm_file("image_lt.ppm", &final_buffer[0], width, height);
}
void render_by_bidirectional_pathtracing(const Camera &camera, const int num_threads) {
const int width = camera.image_width_px;
const int height = camera.image_height_px;
const int spp = camera.samples_per_pixel;
std::vector<Color> image_buffer(width * height * num_threads);
#ifdef USE_OPENMP
omp_set_num_threads(num_threads);
#endif
const long long iteration_per_thread = spp / num_threads;
std::cerr << "Average bidirectional sampling/pixel: " << (iteration_per_thread * num_threads) << std::endl;
int thread_id = 0;
#pragma omp parallel private(thread_id)
{
#ifdef USE_OPENMP
thread_id = omp_get_thread_num();
#endif
Random random(thread_id * 32);
for (int iteration = 0; iteration < iteration_per_thread; ++iteration) {
if (thread_id == 0)
std::cerr << "Rendering (" << ((double)(num_threads * (iteration + 1))) << " bidirectional sampling/pixel) "
<< (100.0 * (double)((iteration + 1)) / iteration_per_thread) << "%" << std::endl;
for (int y = 0; y < height; ++y) {
for (int x = 0; x < width; ++x) {
BidirectionalPathtracingResult bpt_result = bidirectional_pathtracing(camera, x, y, &random);
for (int i = 0; i < bpt_result.samples.size(); ++i) {
const int ix = bpt_result.samples[i].imagebuffer_x;
const int iy = bpt_result.samples[i].imagebuffer_y;
const int idx = (thread_id * width * height) + (iy * width + ix);
if (is_valid_value(bpt_result.samples[i].value)) {
// 得られたサンプルについて、サンプルが現在の画素(x,y)から発射されたeyeサブパスを含むものだった場合、
// Ixy のモンテカルロ推定値はsamples[i].valueそのものなので、そのまま足す。その後、下の画像出力時に発射された回数の総計(iteration_per_thread * num_threads)で割る。
//
// 得られたサンプルについて、現在の画素から発射されたeyeサブパスを含むものではなかった場合(lightサブパスが別の画素(x',y')に到達した場合)は
// Ix'y' のモンテカルロ推定値を新しく得たわけだが、この場合、画像全体に対して光源側からサンプルを生成し、たまたまx'y'にヒットしたと考えるため、
// このようなサンプルについては最終的に光源から発射した回数の総計(width * height * iteration_per_thread * num_threads)で割って、画素への寄与とする必要がある。
// iteration_per_thread * num_threadsの分は上と共通なので、width * heightで割ってからimage_bufferに足すことで、最終的な画像出力時に帳尻があり、正確な結果になる。
if (bpt_result.samples[i].start_from_pixel)
image_buffer[idx] = image_buffer[idx] + bpt_result.samples[i].value;
else
image_buffer[idx] = image_buffer[idx] + bpt_result.samples[i].value / ((double)width * height);
}
}
}
}
}
}
// サンプル数で割る + 左右反転
std::vector<Color> final_buffer(width * height);
for (int t = 0; t < num_threads; ++t)
for (int y = 0; y < height; ++y)
for (int x = 0; x < width; ++x) {
final_buffer[y * width + x] = final_buffer[y * width + x] + image_buffer[(t * width * height) + y * width + (width - x - 1)] /
((double)iteration_per_thread * num_threads);
}
// 出力
save_ppm_file("image_bpt.ppm", &final_buffer[0], width, height);
}
};
#endif