Compare commits
2 Commits
085a9fcce8
...
0f172ef987
Author | SHA1 | Date | |
---|---|---|---|
|
0f172ef987 | ||
|
7c4872c4e2 |
4
Makefile
4
Makefile
@ -2,8 +2,8 @@ CXX = g++
|
|||||||
|
|
||||||
CXXFLAGS = -std=c++20 -Wall -Wextra -Wno-unused-parameter -march=native -O3 -flto -fopenmp
|
CXXFLAGS = -std=c++20 -Wall -Wextra -Wno-unused-parameter -march=native -O3 -flto -fopenmp
|
||||||
|
|
||||||
DEPS = util.h vec3.h color.h ray.h camera.h hittable.h hittable_list.h sphere.h material.h lodepng.h moving_sphere.h bvh.h aabb.h texture.h aarect.h
|
DEPS = util.h vec3.h color.h ray.h camera.h hittable.h hittable_list.h sphere.h material.h lodepng.h moving_sphere.h bvh.h aabb.h texture.h aarect.h box.h
|
||||||
OBJ = main.o material.o vec3.o lodepng.o sphere.o moving_sphere.o bvh.o aarect.o hittable_list.o
|
OBJ = main.o material.o vec3.o lodepng.o sphere.o moving_sphere.o bvh.o aarect.o box.o hittable_list.o hittable.o
|
||||||
|
|
||||||
TARGET = toytracer
|
TARGET = toytracer
|
||||||
|
|
||||||
|
25
box.cpp
Normal file
25
box.cpp
Normal file
@ -0,0 +1,25 @@
|
|||||||
|
#include "box.h"
|
||||||
|
#include "aarect.h"
|
||||||
|
|
||||||
|
Box::Box(const Point3& p0, const Point3& p1, std::shared_ptr<Material> ptr) {
|
||||||
|
box_min = p0;
|
||||||
|
box_max = p1;
|
||||||
|
|
||||||
|
sides.add(std::make_shared<Xy_rect>(p0.x(), p1.x(), p0.y(), p1.y(), p1.z(), ptr));
|
||||||
|
sides.add(std::make_shared<Flip_face>(std::make_shared<Xy_rect>(p0.x(), p1.x(), p0.y(), p1.y(), p0.z(), ptr)));
|
||||||
|
|
||||||
|
sides.add(std::make_shared<Xz_rect>(p0.x(), p1.x(), p0.z(), p1.z(), p1.y(), ptr));
|
||||||
|
sides.add(std::make_shared<Flip_face>(std::make_shared<Xz_rect>(p0.x(), p1.x(), p0.z(), p1.z(), p0.y(), ptr)));
|
||||||
|
|
||||||
|
sides.add(std::make_shared<Yz_rect>(p0.y(), p1.y(), p0.z(), p1.z(), p1.x(), ptr));
|
||||||
|
sides.add(std::make_shared<Flip_face>(std::make_shared<Yz_rect>(p0.y(), p1.y(), p0.z(), p1.z(), p0.x(), ptr)));
|
||||||
|
}
|
||||||
|
|
||||||
|
bool Box::hit(const Ray& r, double t0, double t1, hit_record& rec) const {
|
||||||
|
return sides.hit(r, t0, t1, rec);
|
||||||
|
}
|
||||||
|
|
||||||
|
bool Box::bounding_box(double t0, double t1, Aabb& output_box) const {
|
||||||
|
output_box = Aabb(box_min, box_max);
|
||||||
|
return true;
|
||||||
|
}
|
24
box.h
Normal file
24
box.h
Normal file
@ -0,0 +1,24 @@
|
|||||||
|
#ifndef BOX_H
|
||||||
|
#define BOX_H
|
||||||
|
|
||||||
|
#include "vec3.h"
|
||||||
|
#include "hittable.h"
|
||||||
|
#include "hittable_list.h"
|
||||||
|
|
||||||
|
class Box : public Hittable {
|
||||||
|
public:
|
||||||
|
Box() {}
|
||||||
|
|
||||||
|
Box(const Point3& p0, const Point3& p1, std::shared_ptr<Material> ptr);
|
||||||
|
|
||||||
|
virtual bool hit(const Ray& r, double t0, double t1, hit_record& rec) const;
|
||||||
|
|
||||||
|
virtual bool bounding_box(double t0, double t1, Aabb& output_box) const;
|
||||||
|
|
||||||
|
private:
|
||||||
|
Point3 box_min;
|
||||||
|
Point3 box_max;
|
||||||
|
Hittable_list sides;
|
||||||
|
};
|
||||||
|
|
||||||
|
#endif // BOX_H
|
93
hittable.cpp
Normal file
93
hittable.cpp
Normal file
@ -0,0 +1,93 @@
|
|||||||
|
#include "hittable.h"
|
||||||
|
|
||||||
|
#include <cmath>
|
||||||
|
|
||||||
|
bool Flip_face::hit(const Ray& r, double tmin, double tmax, hit_record& rec) const {
|
||||||
|
if (!ptr->hit(r, tmin, tmax, rec))
|
||||||
|
return false;
|
||||||
|
|
||||||
|
rec.front_face = !rec.front_face;
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool Translate::hit(const Ray& r, double tmin, double tmax, hit_record& rec) const {
|
||||||
|
Ray moved_r(r.origin() - offset, r.direction(), r.time());
|
||||||
|
if (!ptr->hit(moved_r, tmin, tmax, rec))
|
||||||
|
return false;
|
||||||
|
|
||||||
|
rec.p += offset;
|
||||||
|
rec.set_face_normal(moved_r, rec.normal);
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool Translate::bounding_box(double t0, double t1, Aabb& output_box) const {
|
||||||
|
if (!ptr->bounding_box(t0, t1, output_box))
|
||||||
|
return false;
|
||||||
|
|
||||||
|
output_box = Aabb(output_box.min() + offset, output_box.max() + offset);
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
Rotate_y::Rotate_y(std::shared_ptr<Hittable> p, double angle) : ptr(p) {
|
||||||
|
auto radians = degrees_to_radians(angle);
|
||||||
|
sin_theta = std::sin(radians);
|
||||||
|
cos_theta = std::cos(radians);
|
||||||
|
|
||||||
|
hasbox = ptr->bounding_box(0, 1, bbox);
|
||||||
|
|
||||||
|
Point3 min( infinity, infinity, infinity);
|
||||||
|
Point3 max(-infinity, -infinity, -infinity);
|
||||||
|
|
||||||
|
for (int i = 0; i < 2; i++) {
|
||||||
|
for (int j = 0; j < 2; j++) {
|
||||||
|
for (int k = 0; k < 2; k++) {
|
||||||
|
auto x = i*bbox.max().x() + (1-i)*bbox.min().x();
|
||||||
|
auto y = j*bbox.max().y() + (1-j)*bbox.min().y();
|
||||||
|
auto z = k*bbox.max().z() + (1-k)*bbox.min().z();
|
||||||
|
|
||||||
|
auto newx = cos_theta*x + sin_theta*z;
|
||||||
|
auto newz = -sin_theta*x + cos_theta*z;
|
||||||
|
|
||||||
|
Vec3 tester(newx, y, newz);
|
||||||
|
|
||||||
|
for (int c = 0; c < 3; c++) {
|
||||||
|
min[c] = std::fmin(min[c], tester[c]);
|
||||||
|
max[c] = std::fmin(max[c], tester[c]);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
bbox = Aabb(min, max);
|
||||||
|
}
|
||||||
|
|
||||||
|
bool Rotate_y::hit(const Ray& r, double tmin, double tmax, hit_record& rec) const {
|
||||||
|
auto origin = r.origin();
|
||||||
|
auto direction = r.direction();
|
||||||
|
|
||||||
|
origin[0] = cos_theta*r.origin()[0] - sin_theta*r.origin()[2];
|
||||||
|
origin[2] = sin_theta*r.origin()[0] + cos_theta*r.origin()[2];
|
||||||
|
|
||||||
|
direction[0] = cos_theta*r.direction()[0] - sin_theta*r.direction()[2];
|
||||||
|
direction[2] = sin_theta*r.direction()[0] + cos_theta*r.direction()[2];
|
||||||
|
|
||||||
|
Ray rotated_r(origin, direction, r.time());
|
||||||
|
|
||||||
|
if (!ptr->hit(rotated_r, tmin, tmax, rec))
|
||||||
|
return false;
|
||||||
|
|
||||||
|
auto p = rec.p;
|
||||||
|
auto normal = rec.normal;
|
||||||
|
|
||||||
|
p[0] = cos_theta*rec.p[0] + sin_theta*rec.p[2];
|
||||||
|
p[2] = -sin_theta*rec.p[0] + cos_theta*rec.p[2];
|
||||||
|
|
||||||
|
normal[0] = cos_theta*rec.normal[0] + sin_theta*rec.normal[2];
|
||||||
|
normal[2] = -sin_theta*rec.normal[0] + cos_theta*rec.normal[2];
|
||||||
|
|
||||||
|
rec.p = p;
|
||||||
|
rec.set_face_normal(rotated_r, normal);
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
38
hittable.h
38
hittable.h
@ -34,13 +34,7 @@ class Flip_face : public Hittable {
|
|||||||
public:
|
public:
|
||||||
Flip_face(std::shared_ptr<Hittable> p) : ptr(p) {}
|
Flip_face(std::shared_ptr<Hittable> p) : ptr(p) {}
|
||||||
|
|
||||||
virtual bool hit(const Ray &r, double tmin, double tmax, hit_record &rec) const {
|
virtual bool hit(const Ray &r, double tmin, double tmax, hit_record &rec) const;
|
||||||
if (!ptr->hit(r, tmin, tmax, rec))
|
|
||||||
return false;
|
|
||||||
|
|
||||||
rec.front_face = !rec.front_face;
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
virtual bool bounding_box(double t0, double t1, Aabb& output_box) const {
|
virtual bool bounding_box(double t0, double t1, Aabb& output_box) const {
|
||||||
return ptr->bounding_box(t0, t1, output_box);
|
return ptr->bounding_box(t0, t1, output_box);
|
||||||
@ -50,4 +44,34 @@ class Flip_face : public Hittable {
|
|||||||
std::shared_ptr<Hittable> ptr;
|
std::shared_ptr<Hittable> ptr;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
class Translate : public Hittable {
|
||||||
|
public:
|
||||||
|
Translate(std::shared_ptr<Hittable> p, const Vec3& displacement) : ptr(p), offset(displacement) {}
|
||||||
|
|
||||||
|
virtual bool hit(const Ray &r, double tmin, double tmax, hit_record &rec) const;
|
||||||
|
virtual bool bounding_box(double t0, double t1, Aabb& output_box) const;
|
||||||
|
|
||||||
|
private:
|
||||||
|
std::shared_ptr<Hittable> ptr;
|
||||||
|
Vec3 offset;
|
||||||
|
};
|
||||||
|
|
||||||
|
class Rotate_y : public Hittable {
|
||||||
|
public:
|
||||||
|
Rotate_y(std::shared_ptr<Hittable> p, double angle);
|
||||||
|
|
||||||
|
virtual bool hit(const Ray& r, double tmin, double tmax, hit_record& rec) const;
|
||||||
|
virtual bool bounding_box(double t0, double t1, Aabb& output_box) const {
|
||||||
|
output_box = bbox;
|
||||||
|
return hasbox;
|
||||||
|
}
|
||||||
|
|
||||||
|
private:
|
||||||
|
std::shared_ptr<Hittable> ptr;
|
||||||
|
double sin_theta;
|
||||||
|
double cos_theta;
|
||||||
|
bool hasbox;
|
||||||
|
Aabb bbox;
|
||||||
|
};
|
||||||
|
|
||||||
#endif // HITTABLE_H
|
#endif // HITTABLE_H
|
||||||
|
23
main.cpp
23
main.cpp
@ -19,6 +19,7 @@
|
|||||||
#include "moving_sphere.h"
|
#include "moving_sphere.h"
|
||||||
#include "camera.h"
|
#include "camera.h"
|
||||||
#include "aarect.h"
|
#include "aarect.h"
|
||||||
|
#include "box.h"
|
||||||
|
|
||||||
Color ray_color(const Ray& r, const Color& background, const Hittable& world, int depth) {
|
Color ray_color(const Ray& r, const Color& background, const Hittable& world, int depth) {
|
||||||
hit_record rec;
|
hit_record rec;
|
||||||
@ -153,7 +154,9 @@ std::pair<Hittable_list, Camera> cornell_box(const double aspect_ratio) {
|
|||||||
auto white = std::make_shared<Lambertian>(std::make_shared<Solid_color>(0.73, 0.73, 0.73));
|
auto white = std::make_shared<Lambertian>(std::make_shared<Solid_color>(0.73, 0.73, 0.73));
|
||||||
auto green = std::make_shared<Lambertian>(std::make_shared<Solid_color>(0.12, 0.45, 0.15));
|
auto green = std::make_shared<Lambertian>(std::make_shared<Solid_color>(0.12, 0.45, 0.15));
|
||||||
|
|
||||||
auto light = std::make_shared<Diffuse_light>(std::make_shared<Solid_color>(15, 15, 15));
|
//auto light = std::make_shared<Diffuse_light>(std::make_shared<Solid_color>(15, 15, 15));
|
||||||
|
auto light = std::make_shared<Diffuse_light>(std::make_shared<Solid_color>(30, 30, 30));
|
||||||
|
//auto light = std::make_shared<Diffuse_light>(std::make_shared<Solid_color>(100, 100, 100));
|
||||||
|
|
||||||
objects.add(std::make_shared<Flip_face>(std::make_shared<Yz_rect>( 0, 555, 0, 555, 555, green)));
|
objects.add(std::make_shared<Flip_face>(std::make_shared<Yz_rect>( 0, 555, 0, 555, 555, green)));
|
||||||
objects.add(std::make_shared<Yz_rect>( 0, 555, 0, 555, 0, red ));
|
objects.add(std::make_shared<Yz_rect>( 0, 555, 0, 555, 0, red ));
|
||||||
@ -162,6 +165,16 @@ std::pair<Hittable_list, Camera> cornell_box(const double aspect_ratio) {
|
|||||||
objects.add(std::make_shared<Xz_rect>( 0, 555, 0, 555, 555, white));
|
objects.add(std::make_shared<Xz_rect>( 0, 555, 0, 555, 555, white));
|
||||||
objects.add(std::make_shared<Flip_face>(std::make_shared<Xy_rect>( 0, 555, 0, 555, 555, white)));
|
objects.add(std::make_shared<Flip_face>(std::make_shared<Xy_rect>( 0, 555, 0, 555, 555, white)));
|
||||||
|
|
||||||
|
std::shared_ptr<Hittable> box1 = std::make_shared<Box>(Point3(0, 0, 0), Point3(165, 300, 165), white);
|
||||||
|
box1 = std::make_shared<Rotate_y>(box1, 15);
|
||||||
|
box1 = std::make_shared<Translate>(box1, Vec3(265, 0, 295));
|
||||||
|
objects.add(box1);
|
||||||
|
|
||||||
|
std::shared_ptr<Hittable> box2 = std::make_shared<Box>(Point3(0, 0, 0), Point3(165, 165, 165), white);
|
||||||
|
box2 = std::make_shared<Rotate_y>(box2, -18);
|
||||||
|
box2 = std::make_shared<Translate>(box2, Vec3(130, 0, 65));
|
||||||
|
objects.add(box2);
|
||||||
|
|
||||||
Point3 lookfrom(278, 278, -800);
|
Point3 lookfrom(278, 278, -800);
|
||||||
Point3 lookat(278, 278, 0);
|
Point3 lookat(278, 278, 0);
|
||||||
Vec3 vup(0, 1, 0);
|
Vec3 vup(0, 1, 0);
|
||||||
@ -190,15 +203,15 @@ int main() {
|
|||||||
//const int image_width = 384;
|
//const int image_width = 384;
|
||||||
const int image_height = static_cast<int>(image_width / aspect_ratio);
|
const int image_height = static_cast<int>(image_width / aspect_ratio);
|
||||||
//const int samples_per_pixel = 10000;
|
//const int samples_per_pixel = 10000;
|
||||||
//const int samples_per_pixel = 5000;
|
const int samples_per_pixel = 5000;
|
||||||
//const int samples_per_pixel = 2000;
|
//const int samples_per_pixel = 2000;
|
||||||
const int samples_per_pixel = 1000;
|
//const int samples_per_pixel = 1000;
|
||||||
//const int samples_per_pixel = 400;
|
//const int samples_per_pixel = 400;
|
||||||
//const int samples_per_pixel = 50;
|
//const int samples_per_pixel = 50;
|
||||||
const int max_depth = 50;
|
const int max_depth = 50;
|
||||||
|
|
||||||
const int tile_size = 128;
|
//const int tile_size = 128;
|
||||||
//const int tile_size = 64;
|
const int tile_size = 64;
|
||||||
|
|
||||||
//const int sph_i = 3;
|
//const int sph_i = 3;
|
||||||
//const int sph_i = 5;
|
//const int sph_i = 5;
|
||||||
|
6
util.h
6
util.h
@ -14,11 +14,9 @@ inline double degrees_to_radians(double degrees) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
inline double random_double() {
|
inline double random_double() {
|
||||||
static std::uniform_real_distribution<double> distribution(0.0, 1.0);
|
static thread_local std::uniform_real_distribution<double> distribution(0.0, 1.0);
|
||||||
//static std::mt19937 generator(std::time(nullptr));
|
|
||||||
static thread_local std::mt19937 generator(std::random_device{}());
|
static thread_local std::mt19937 generator(std::random_device{}());
|
||||||
static std::function<double()> rand_gen = std::bind(distribution, generator);
|
return distribution(generator);
|
||||||
return rand_gen();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
inline double random_double(double min, double max) {
|
inline double random_double(double min, double max) {
|
||||||
|
Loading…
Reference in New Issue
Block a user