From 96e103ee611aeb88e3c04ac57a8c821faf2486a5 Mon Sep 17 00:00:00 2001 From: Ishan Jain Date: Wed, 31 May 2023 11:36:19 +0530 Subject: [PATCH] Added back codebase from previous projects --- .gitignore | 5 +- Cargo.lock | 345 +++++++++++++++++++++++++- Cargo.toml | 11 +- assets/earthmap.jpg | Bin 0 -> 161345 bytes src/aabb.rs | 36 +++ src/camera.rs | 90 +++++++ src/demos/checkered_motion_blur.rs | 126 ++++++++++ src/demos/cornell_box.rs | 161 ++++++++++++ src/demos/cornell_smoke_and_fog.rs | 123 +++++++++ src/demos/image_texture.rs | 64 +++++ src/demos/instances.rs | 117 +++++++++ src/demos/mod.rs | 234 +++++++++++++++++ src/demos/perlin_noise_ball.rs | 65 +++++ src/demos/simple_light.rs | 80 ++++++ src/demos/two_spheres.rs | 71 ++++++ src/hitable/bvh.rs | 139 +++++++++++ src/hitable/hitable_list.rs | 54 ++++ src/hitable/mod.rs | 117 +++++++++ src/hitable/rotate.rs | 130 ++++++++++ src/hitable/shapes/cuboid.rs | 87 +++++++ src/hitable/shapes/mod.rs | 9 + src/hitable/shapes/moving_sphere.rs | 92 +++++++ src/hitable/shapes/rectangle.rs | 175 +++++++++++++ src/hitable/shapes/sphere.rs | 77 ++++++ src/hitable/translate.rs | 37 +++ src/hitable/volume/constant_medium.rs | 63 +++++ src/hitable/volume/mod.rs | 3 + src/main.rs | 199 ++++++++++++++- src/materials/dielectric.rs | 62 +++++ src/materials/diffuse_light.rs | 18 ++ src/materials/isotropic.rs | 31 +++ src/materials/lambertian.rs | 31 +++ src/materials/metal.rs | 46 ++++ src/materials/mod.rs | 72 ++++++ src/texture/checker.rs | 25 ++ src/texture/image_texture.rs | 56 +++++ src/texture/mod.rs | 17 ++ src/texture/perlin.rs | 120 +++++++++ src/texture/perlin_noise.rs | 34 +++ src/texture/solid.rs | 18 ++ src/types/color.rs | 12 + src/types/dimension.rs | 21 ++ src/types/mod.rs | 17 ++ src/types/ray.rs | 54 ++++ src/types/simd_vec3.rs | 214 ++++++++++++++++ src/types/vec3.rs | 244 ++++++++++++++++++ 46 files changed, 3776 insertions(+), 26 deletions(-) create mode 100644 assets/earthmap.jpg create mode 100644 src/aabb.rs create mode 100644 src/camera.rs create mode 100644 src/demos/checkered_motion_blur.rs create mode 100644 src/demos/cornell_box.rs create mode 100644 src/demos/cornell_smoke_and_fog.rs create mode 100644 src/demos/image_texture.rs create mode 100644 src/demos/instances.rs create mode 100644 src/demos/mod.rs create mode 100644 src/demos/perlin_noise_ball.rs create mode 100644 src/demos/simple_light.rs create mode 100644 src/demos/two_spheres.rs create mode 100644 src/hitable/bvh.rs create mode 100644 src/hitable/hitable_list.rs create mode 100644 src/hitable/mod.rs create mode 100644 src/hitable/rotate.rs create mode 100644 src/hitable/shapes/cuboid.rs create mode 100644 src/hitable/shapes/mod.rs create mode 100644 src/hitable/shapes/moving_sphere.rs create mode 100644 src/hitable/shapes/rectangle.rs create mode 100644 src/hitable/shapes/sphere.rs create mode 100644 src/hitable/translate.rs create mode 100644 src/hitable/volume/constant_medium.rs create mode 100644 src/hitable/volume/mod.rs create mode 100644 src/materials/dielectric.rs create mode 100644 src/materials/diffuse_light.rs create mode 100644 src/materials/isotropic.rs create mode 100644 src/materials/lambertian.rs create mode 100644 src/materials/metal.rs create mode 100644 src/materials/mod.rs create mode 100644 src/texture/checker.rs create mode 100644 src/texture/image_texture.rs create mode 100644 src/texture/mod.rs create mode 100644 src/texture/perlin.rs create mode 100644 src/texture/perlin_noise.rs create mode 100644 src/texture/solid.rs create mode 100644 src/types/color.rs create mode 100644 src/types/dimension.rs create mode 100644 src/types/mod.rs create mode 100644 src/types/ray.rs create mode 100644 src/types/simd_vec3.rs create mode 100644 src/types/vec3.rs diff --git a/.gitignore b/.gitignore index ea8c4bf..fda8fe0 100644 --- a/.gitignore +++ b/.gitignore @@ -1 +1,4 @@ -/target +target/ +*.ppm +*.png +!renders/* diff --git a/Cargo.lock b/Cargo.lock index eed3a4c..595883f 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -2,6 +2,42 @@ # It is not intended for manual editing. version = 3 +[[package]] +name = "adler" +version = "1.0.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f26201604c87b1e01bd3d98f8d5d9a8fcbb815e8cedb41ffccbeb4bf593a35fe" + +[[package]] +name = "autocfg" +version = "1.1.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d468802bab17cbc0cc575e9b053f41e72aa36bfa6b7f55e3529ffa43161b97fa" + +[[package]] +name = "bitflags" +version = "1.3.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "bef38d45163c2f1dde094a7dfd33ccf595c92905c8f8f4fdc18d06fb1037718a" + +[[package]] +name = "bytemuck" +version = "1.13.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "17febce684fd15d89027105661fec94afb475cb995fbc59d2865198446ba2eea" + +[[package]] +name = "byteorder" +version = "1.4.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "14c189c53d098945499cdfa7ecc63567cf3886b3332b312a5b4585d8d3a6a610" + +[[package]] +name = "cc" +version = "1.0.79" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "50d30906286121d95be3d479533b458f87493b30a4b5f79a607db8f5d11aa91f" + [[package]] name = "cfg-if" version = "1.0.0" @@ -9,10 +45,102 @@ source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "baf1de4339761588bc0619e3cbc0120ee582ebb74b53b4efbf79117bd2da40fd" [[package]] -name = "getrandom" -version = "0.2.3" +name = "cmake" +version = "0.1.50" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "7fcd999463524c52659517fe2cea98493cfe485d10565e7b0fb07dbba7ad2753" +checksum = "a31c789563b815f77f4250caee12365734369f942439b7defd71e18a48197130" +dependencies = [ + "cc", +] + +[[package]] +name = "color_quant" +version = "1.1.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "3d7b894f5411737b7867f4827955924d7c254fc9f4d91a6aad6b097804b1018b" + +[[package]] +name = "crc32fast" +version = "1.3.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b540bd8bc810d3885c6ea91e2018302f68baba2129ab3e88f32389ee9370880d" +dependencies = [ + "cfg-if", +] + +[[package]] +name = "crossbeam-channel" +version = "0.5.8" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a33c2bf77f2df06183c3aa30d1e96c0695a313d4f9c453cc3762a6db39f99200" +dependencies = [ + "cfg-if", + "crossbeam-utils", +] + +[[package]] +name = "crossbeam-deque" +version = "0.8.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ce6fd6f855243022dcecf8702fef0c297d4338e226845fe067f6341ad9fa0cef" +dependencies = [ + "cfg-if", + "crossbeam-epoch", + "crossbeam-utils", +] + +[[package]] +name = "crossbeam-epoch" +version = "0.9.14" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "46bd5f3f85273295a9d14aedfb86f6aadbff6d8f5295c4a9edb08e819dcf5695" +dependencies = [ + "autocfg", + "cfg-if", + "crossbeam-utils", + "memoffset", + "scopeguard", +] + +[[package]] +name = "crossbeam-utils" +version = "0.8.15" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "3c063cd8cc95f5c377ed0d4b49a4b21f632396ff690e8470c29b3359b346984b" +dependencies = [ + "cfg-if", +] + +[[package]] +name = "either" +version = "1.8.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "7fcaabb2fef8c910e7f4c7ce9f67a1283a1715879a7c230ca9d6d1ae31f16d91" + +[[package]] +name = "fdeflate" +version = "0.3.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d329bdeac514ee06249dabc27877490f17f5d371ec693360768b838e19f3ae10" +dependencies = [ + "simd-adler32", +] + +[[package]] +name = "flate2" +version = "1.0.26" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "3b9429470923de8e8cbd4d2dc513535400b4b3fef0319fb5c4e1f520a7bef743" +dependencies = [ + "crc32fast", + "miniz_oxide", +] + +[[package]] +name = "getrandom" +version = "0.2.9" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c85e1d9ab2eadba7e5040d4e09cbd6d072b76a557ad64e797c2cb9d4da21d7e4" dependencies = [ "cfg-if", "libc", @@ -20,10 +148,134 @@ dependencies = [ ] [[package]] -name = "libc" -version = "0.2.107" +name = "hermit-abi" +version = "0.2.6" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "fbe5e23404da5b4f555ef85ebed98fb4083e55a00c317800bc2a50ede9f3d219" +checksum = "ee512640fe35acbfb4bb779db6f0d80704c2cacfa2e39b601ef3e3f47d1ae4c7" +dependencies = [ + "libc", +] + +[[package]] +name = "image" +version = "0.24.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "527909aa81e20ac3a44803521443a765550f09b5130c2c2fa1ea59c2f8f50a3a" +dependencies = [ + "bytemuck", + "byteorder", + "color_quant", + "jpeg-decoder", + "num-rational", + "num-traits", + "png", +] + +[[package]] +name = "jpeg-decoder" +version = "0.3.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "bc0000e42512c92e31c2252315bda326620a4e034105e900c98ec492fa077b3e" + +[[package]] +name = "lazy_static" +version = "1.4.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e2abad23fbc42b3700f2f279844dc832adb2b2eb069b2df918f455c4e18cc646" + +[[package]] +name = "libc" +version = "0.2.144" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "2b00cc1c228a6782d0f076e7b232802e0c5689d41bb5df366f2a6b6621cfdfe1" + +[[package]] +name = "libm" +version = "0.1.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "7fc7aa29613bd6a620df431842069224d8bc9011086b1db4c0e0cd47fa03ec9a" + +[[package]] +name = "memoffset" +version = "0.8.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d61c719bcfbcf5d62b3a09efa6088de8c54bc0bfcd3ea7ae39fcc186108b8de1" +dependencies = [ + "autocfg", +] + +[[package]] +name = "miniz_oxide" +version = "0.7.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e7810e0be55b428ada41041c41f32c9f1a42817901b4ccf45fa3d4b6561e74c7" +dependencies = [ + "adler", + "simd-adler32", +] + +[[package]] +name = "num-integer" +version = "0.1.45" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "225d3389fb3509a24c93f5c29eb6bde2586b98d9f016636dff58d7c6f7569cd9" +dependencies = [ + "autocfg", + "num-traits", +] + +[[package]] +name = "num-rational" +version = "0.4.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "0638a1c9d0a3c0914158145bc76cff373a75a627e6ecbfb71cbe6f453a5a19b0" +dependencies = [ + "autocfg", + "num-integer", + "num-traits", +] + +[[package]] +name = "num-traits" +version = "0.2.15" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "578ede34cf02f8924ab9447f50c28075b4d3e5b269972345e7e0372b38c6cdcd" +dependencies = [ + "autocfg", +] + +[[package]] +name = "num_cpus" +version = "1.15.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "0fac9e2da13b5eb447a6ce3d392f23a29d8694bff781bf03a16cd9ac8697593b" +dependencies = [ + "hermit-abi", + "libc", +] + +[[package]] +name = "packed_simd" +version = "0.3.8" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "6cf965495ba3190af4c54375cb7d5d4777954457a215035a783f9aec5380b19f" +dependencies = [ + "cfg-if", + "libm", +] + +[[package]] +name = "png" +version = "0.17.8" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "aaeebc51f9e7d2c150d3f3bfeb667f2aa985db5ef1e3d212847bdedb488beeaa" +dependencies = [ + "bitflags", + "crc32fast", + "fdeflate", + "flate2", + "miniz_oxide", +] [[package]] name = "ppv-lite86" @@ -72,14 +324,83 @@ dependencies = [ ] [[package]] -name = "raytracing-the-rest-of-your-life" -version = "0.1.0" +name = "rayon" +version = "1.7.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1d2df5196e37bcc87abebc0053e20787d73847bb33134a69841207dd0a47f03b" dependencies = [ - "rand", + "either", + "rayon-core", ] [[package]] -name = "wasi" -version = "0.10.2+wasi-snapshot-preview1" +name = "rayon-core" +version = "1.11.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "fd6fbd9a79829dd1ad0cc20627bf1ed606756a7f77edff7b66b7064f9cb327c6" +checksum = "4b8f95bd6966f5c87776639160a66bd8ab9895d9d4ab01ddba9fc60661aebe8d" +dependencies = [ + "crossbeam-channel", + "crossbeam-deque", + "crossbeam-utils", + "num_cpus", +] + +[[package]] +name = "raytracing-the-rest-of-your-life" +version = "0.1.0" +dependencies = [ + "image", + "num-traits", + "packed_simd", + "rand", + "rayon", + "sdl2", +] + +[[package]] +name = "scopeguard" +version = "1.1.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d29ab0c6d3fc0ee92fe66e2d99f700eab17a8d57d1c1d3b748380fb20baa78cd" + +[[package]] +name = "sdl2" +version = "0.35.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f7959277b623f1fb9e04aea73686c3ca52f01b2145f8ea16f4ff30d8b7623b1a" +dependencies = [ + "bitflags", + "lazy_static", + "libc", + "sdl2-sys", +] + +[[package]] +name = "sdl2-sys" +version = "0.35.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e3586be2cf6c0a8099a79a12b4084357aa9b3e0b0d7980e3b67aaf7a9d55f9f0" +dependencies = [ + "cfg-if", + "cmake", + "libc", + "version-compare", +] + +[[package]] +name = "simd-adler32" +version = "0.3.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "238abfbb77c1915110ad968465608b68e869e0772622c9656714e73e5a1a522f" + +[[package]] +name = "version-compare" +version = "0.1.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "579a42fc0b8e0c63b76519a339be31bed574929511fa53c1a3acae26eb258f29" + +[[package]] +name = "wasi" +version = "0.11.0+wasi-snapshot-preview1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "9c8d87e72b64a3b4db28d11ce29237c246188f4f51057d65a7eab63b7987e423" diff --git a/Cargo.toml b/Cargo.toml index 4ef7614..2166e81 100644 --- a/Cargo.toml +++ b/Cargo.toml @@ -4,4 +4,13 @@ version = "0.1.0" edition = "2021" [dependencies] -rand = "0.8.4" +image = { version = "0.24.6", default-features = false, features = ["jpeg", "png"] } +num-traits = "0.2.15" +packed_simd = "0.3.8" +rand = { version = "0.8.4", features = ["small_rng"] } +rayon = "1.7.0" +sdl2 = { version = "0.35.2", features = ["bundled"], optional = true } + +[features] +default = ["gui"] +gui = ["sdl2"] diff --git a/assets/earthmap.jpg b/assets/earthmap.jpg new file mode 100644 index 0000000000000000000000000000000000000000..908c1606a1c794043f95fe09730653f229b5aec1 GIT binary patch literal 161345 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 literal 0 HcmV?d00001 diff --git a/src/aabb.rs b/src/aabb.rs new file mode 100644 index 0000000..52b4b33 --- /dev/null +++ b/src/aabb.rs @@ -0,0 +1,36 @@ +use crate::types::{Ray, Vec3}; + +#[derive(Debug, Copy, Clone)] +pub struct Aabb { + pub min: Vec3, + pub max: Vec3, +} + +impl Aabb { + pub const fn new(min: Vec3, max: Vec3) -> Self { + Self { min, max } + } + + pub fn hit(&self, ray: &Ray, t_min: f64, t_max: f64) -> bool { + let min = (self.min - ray.origin) / ray.direction; + let max = (self.max - ray.origin) / ray.direction; + + let mins = min.min(max); + let maxs = min.max(max); + + let tmin = mins.max_element(t_min); + let tmax = maxs.min_element(t_max); + + tmax > tmin + } + + pub fn surrounding_box(box0: Aabb, box1: Aabb) -> Self { + let smol_box = Vec3::min(box0.min, box1.min); + let big_box = Vec3::max(box0.max, box1.max); + + Self { + min: smol_box, + max: big_box, + } + } +} diff --git a/src/camera.rs b/src/camera.rs new file mode 100644 index 0000000..8a61ce6 --- /dev/null +++ b/src/camera.rs @@ -0,0 +1,90 @@ +use { + crate::types::{Ray, Vec3}, + rand::Rng, +}; + +pub struct Camera { + origin: Vec3, + horizontal: Vec3, + vertical: Vec3, + lower_left_corner: Vec3, + lens_radius: f64, + + // position vectors + u: Vec3, + v: Vec3, + w: Vec3, + + shutter_open: f64, + shutter_close: f64, +} + +impl Camera { + // vertical_fov is the viewable angle from top->bottom + // look_from is basically camera position + // look_at is the point where camera is looking + // v_up is camera's up vector. i.e. it points upwards from the camera + // orthogonal to look_from - look_at vector + pub fn new( + look_from: Vec3, + look_at: Vec3, + v_up: Vec3, + vertical_fov: f64, + aspect: f64, + aperture: f64, + focus_distance: f64, + shutter_open: f64, + shutter_close: f64, + ) -> Self { + // convert degree to radian + let angle = vertical_fov * std::f64::consts::PI / 180.0; + let half_height = (angle / 2.0).tan(); + let half_width = aspect * half_height; + + let origin = look_from; + let w = (look_from - look_at).unit_vector(); + let u = v_up.cross(&w).unit_vector(); + let v = w.cross(&u); + + let lower_left_corner = origin + - u * focus_distance * half_width + - v * focus_distance * half_height + - w * focus_distance; + let horizontal = u * half_width * focus_distance * 2.0; + let vertical = v * half_height * focus_distance * 2.0; + let lens_radius = aperture / 2.0; + + Self { + origin, + horizontal, + vertical, + lower_left_corner, + lens_radius, + u, + v, + w, + shutter_open, + shutter_close, + } + } + + pub fn get_ray(&self, u: f64, v: f64, rng: &mut R) -> Ray { + let rd = random_in_unit_disk(rng) * self.lens_radius; + let offset = self.u * rd.x() + self.v * rd.y(); + let time = rng.gen_range(self.shutter_open..=self.shutter_close); + Ray::new( + self.origin + offset, + self.lower_left_corner + self.horizontal * u + self.vertical * v - self.origin - offset, + time, + ) + } +} + +fn random_in_unit_disk(rng: &mut R) -> Vec3 { + let mut p = Vec3::new(rng.gen::(), rng.gen::(), 0.0) * 2.0 - Vec3::new(1.0, 1.0, 0.0); + + while p.dot(&p) >= 1.0 { + p = Vec3::new(rng.gen::(), rng.gen::(), 0.0) * 2.0 - Vec3::new(1.0, 0.0, 0.0); + } + p +} diff --git a/src/demos/checkered_motion_blur.rs b/src/demos/checkered_motion_blur.rs new file mode 100644 index 0000000..83330ea --- /dev/null +++ b/src/demos/checkered_motion_blur.rs @@ -0,0 +1,126 @@ +use crate::{ + demos::{Demo, ParallelHit}, + hitable::{ + shapes::{MovingSphere, Sphere}, + BvhNode, + }, + materials::{Dielectric, Lambertian, Metal}, + texture::{Checker, Solid}, + types::Vec3, + Camera, +}; +use rand::{rngs::SmallRng, Rng, SeedableRng}; +use std::sync::Arc; + +pub struct CheckeredMotionBlur {} + +impl Demo for CheckeredMotionBlur { + type DemoT = BvhNode>; + + fn name(&self) -> &'static str { + "checkered_motion_blur" + } + + fn get_background(&self) -> Vec3 { + Vec3::new(0.7, 0.8, 1.0) + } + + fn world(&self) -> Self::DemoT { + let mut world: Vec> = Vec::with_capacity(500); + + let mut rng = rand::thread_rng(); + let mut rng = SmallRng::from_rng(&mut rng).unwrap(); + + world.push(Arc::new(Sphere::new( + Vec3::new(0.0, -1000.0, 0.0), + 1000.0, + Lambertian::new(Checker::new( + Solid::new(Vec3::new(0.2, 0.3, 0.1)), + Solid::new(Vec3::new(0.9, 0.9, 0.9)), + )), + ))); + + let radius = 0.2; + let l = Vec3::new(4.0, 0.2, 0.0); + + for a in -10..10 { + let a = a as f64; + for b in -10..10 { + let b = b as f64; + let choose_material_probability = rng.gen::(); + let center = Vec3::new(a + 0.9 * rng.gen::(), 0.2, b + 0.9 * rng.gen::()); + + if (center - l).length() > 0.9 { + if choose_material_probability < 0.8 { + // diffuse material + world.push(Arc::new(MovingSphere::new( + center, + center + Vec3::new(0.0, 0.5 * rng.gen::(), 0.0), + 0.0, + 1.0, + radius, + Lambertian::new(Solid::new(Vec3::new( + rng.gen::() * rng.gen::(), + rng.gen::() * rng.gen::(), + rng.gen::() * rng.gen::(), + ))), + ))); + } else if choose_material_probability < 0.95 { + // metal material + world.push(Arc::new(Sphere::new( + center, + radius, + Metal::with_fuzz( + Vec3::new( + (1.0 + rng.gen::()) * 0.5, + (1.0 + rng.gen::()) * 0.5, + (1.0 + rng.gen::()) * 0.5, + ), + 0.5 * rng.gen::(), + ), + ))); + } else { + // glass material + world.push(Arc::new(Sphere::new(center, radius, Dielectric::new(1.5)))); + } + } + } + } + + world.push(Arc::new(Sphere::new( + Vec3::new(0.0, 1.0, 0.0), + 1.0, + Dielectric::new(1.5), + ))); + world.push(Arc::new(Sphere::new( + Vec3::new(-4.0, 1.0, 0.0), + 1.0, + Lambertian::new(Solid::new(Vec3::new(0.4, 0.2, 0.1))), + ))); + world.push(Arc::new(Sphere::new( + Vec3::new(4.0, 1.0, 0.0), + 1.0, + Metal::with_fuzz(Vec3::new(0.7, 0.6, 0.5), 0.0), + ))); + + BvhNode::new(&mut rng, &mut world, 0.0, 1.0) + } + + fn camera(&self, aspect_ratio: f64) -> Camera { + let lookfrom = Vec3::new(13.0, 2.0, 3.0); + let lookat = Vec3::new(0.0, 0.0, 0.0); + let aperture = 0.1; + let focus_distance = 10.0; + Camera::new( + lookfrom, + lookat, + Vec3::new(0.0, 1.0, 0.0), + 20.0, + aspect_ratio, + aperture, + focus_distance, + 0.0, + 1.0, + ) + } +} diff --git a/src/demos/cornell_box.rs b/src/demos/cornell_box.rs new file mode 100644 index 0000000..dd3c232 --- /dev/null +++ b/src/demos/cornell_box.rs @@ -0,0 +1,161 @@ +use std::sync::Arc; + +use rand::{prelude::SmallRng, Rng, SeedableRng}; + +use crate::{ + demos::Demo, + hitable::{ + hitable_list::HitableList, + shapes::{Cuboid, MovingSphere, RectBuilder, Sphere}, + volume::ConstantMedium, + Hitable, + }, + materials::{Dielectric, DiffuseLight, Isotropic, Lambertian, MaterialBuilder, Metal}, + texture::{ImageTexture, PerlinNoise, Solid}, + types::Vec3, + BvhNode, Camera, +}; + +pub struct CornellBox {} + +impl Demo for CornellBox { + type DemoT = HitableList; + + fn name(&self) -> &'static str { + "cornell_box" + } + + fn world(&self) -> Self::DemoT { + let mut rng = rand::thread_rng(); + let mut rng = SmallRng::from_rng(&mut rng).unwrap(); + + let mut ground_boxes = HitableList { list: Vec::new() }; + let ground = Lambertian::new(Solid::new(Vec3::new(0.48, 0.83, 0.53))); + + for i in 0..20 { + let i = i as f64; + for j in 0..20 { + let j = j as f64; + + let w = 100.0; + let x0 = -1000.0 + i * w; + let z0 = -1000.0 + j * w; + let y0 = 0.0; + + let x1 = x0 + w; + let y1 = rng.gen_range(1.0..=101.0); + let z1 = z0 + w; + + ground_boxes.push(Arc::new(Cuboid::new( + Vec3::new(x0, y0, z0), + Vec3::new(x1, y1, z1), + ground.clone(), + ))); + } + } + + let mut objects = HitableList { list: Vec::new() }; + objects.push(Arc::new(BvhNode::new( + &mut rng, + &mut ground_boxes.list, + 0.0, + 1.0, + ))); + + let light = DiffuseLight::new(Solid::new(Vec3::splat(7.0))); + objects.push(Arc::new( + RectBuilder + .x(123.0..=423.0) + .z(147.0..=412.0) + .y(554.0) + .material(light), + )); + + let center1 = Vec3::new(400.0, 400.0, 200.0); + let center2 = center1 + Vec3::new(30.0, 0.0, 0.0); + objects.push(Arc::new(MovingSphere::new( + center1, + center2, + 0.0, + 1.0, + 50.0, + Lambertian::new(Solid::new(Vec3::new(0.7, 0.3, 0.1))), + ))); + + objects.push(Arc::new(Sphere::new( + Vec3::new(260.0, 150.0, 45.0), + 50.0, + Dielectric::new(1.5), + ))); + + objects.push(Arc::new(Sphere::new( + Vec3::new(0.0, 150.0, 145.0), + 50.0, + Metal::with_fuzz(Vec3::new(0.8, 0.8, 0.9), 1.0), + ))); + + let boundary = Sphere::new(Vec3::new(360.0, 150.0, 145.0), 70.0, Dielectric::new(1.5)); + objects.push(Arc::new(boundary.clone())); + objects.push(Arc::new(ConstantMedium::new( + boundary, + Isotropic::new(Solid::new(Vec3::new(0.2, 0.4, 0.9))), + 0.2, + ))); + + objects.push(Arc::new(ConstantMedium::new( + Sphere::new(Vec3::splat(0.0), 5000.0, Dielectric::new(1.5)), + Isotropic::new(Solid::new(Vec3::splat(1.0))), + 0.0001, + ))); + + let earthmap = ImageTexture::from_filename("assets/earthmap.jpg") + .expect("error in reading assets/earthmap.jpg"); + objects.push(Arc::new(Sphere::new( + Vec3::new(400.0, 200.0, 400.0), + 100.0, + Lambertian::new(earthmap), + ))); + + objects.push(Arc::new(Sphere::new( + Vec3::new(220.0, 280.0, 300.0), + 80.0, + Lambertian::new(PerlinNoise::with_scale(&mut rng, 0.1)), + ))); + + let mut boxes2 = HitableList { list: Vec::new() }; + let white = Lambertian::new(Solid::new(Vec3::splat(0.73))); + for _ in 0..1000 { + boxes2.push(Arc::new(Sphere::new( + Vec3::random_in_range(&mut rng, 0.0..=165.0), + 10.0, + white.clone(), + ))); + } + + objects.push(Arc::new( + BvhNode::new(&mut rng, &mut boxes2.list, 0.0, 1.0) + .rotate_y(15.0) + .translate(Vec3::new(-100.0, 270.0, 395.0)), + )); + + objects + } + + fn camera(&self, aspect_ratio: f64) -> Camera { + let lookfrom = Vec3::new(478.0, 278.0, -600.0); + let lookat = Vec3::new(278.0, 278.0, 0.0); + let aperture = 0.1; + let focus_distance = 40.0; + Camera::new( + lookfrom, + lookat, + Vec3::new(0.0, 1.0, 0.0), + 40.0, + aspect_ratio, + aperture, + focus_distance, + 0.0, + 1.0, + ) + } +} diff --git a/src/demos/cornell_smoke_and_fog.rs b/src/demos/cornell_smoke_and_fog.rs new file mode 100644 index 0000000..d004590 --- /dev/null +++ b/src/demos/cornell_smoke_and_fog.rs @@ -0,0 +1,123 @@ +use std::sync::Arc; + +use rand::{prelude::SmallRng, SeedableRng}; + +use crate::{ + demos::{Demo, ParallelHit}, + hitable::{ + shapes::{Cuboid, RectBuilder}, + volume::ConstantMedium, + Hitable, + }, + materials::{DiffuseLight, Isotropic, Lambertian, MaterialBuilder}, + texture::Solid, + types::Vec3, + BvhNode, Camera, +}; + +pub struct CornellSmokeAndFog {} + +impl Demo for CornellSmokeAndFog { + type DemoT = BvhNode>; + + fn name(&self) -> &'static str { + "cornell_smoke_and_fog" + } + + fn world(&self) -> Self::DemoT { + let mut world: Vec> = Vec::with_capacity(8); + + let mut rng = rand::thread_rng(); + let mut rng = SmallRng::from_rng(&mut rng).unwrap(); + + let red = Lambertian::new(Solid::new(Vec3::new(0.65, 0.05, 0.05))); + let white = Lambertian::new(Solid::new(Vec3::splat(0.73))); + let green = Lambertian::new(Solid::new(Vec3::new(0.12, 0.45, 0.15))); + let light = DiffuseLight::new(Solid::new(Vec3::splat(7.0))); + + world.push(Arc::new( + RectBuilder + .y(0.0..=555.0) + .z(0.0..=555.0) + .x(555.0) + .material(green), + )); + world.push(Arc::new( + RectBuilder + .y(0.0..=555.0) + .z(0.0..=555.0) + .x(0.0) + .material(red), + )); + + world.push(Arc::new( + RectBuilder + .x(113.0..=443.0) + .z(127.0..=432.0) + .y(554.0) + .material(light), + )); + world.push(Arc::new( + RectBuilder + .x(0.0..=555.0) + .z(0.0..=555.0) + .y(0.0) + .material(white.clone()), + )); + + world.push(Arc::new( + RectBuilder + .x(0.0..=555.0) + .z(0.0..=555.0) + .y(555.0) + .material(white.clone()), + )); + world.push(Arc::new( + RectBuilder + .x(0.0..=555.0) + .y(0.0..=555.0) + .z(555.0) + .material(white.clone()), + )); + + // Add the two boxes + world.push(Arc::new(ConstantMedium::new( + Cuboid::new( + Vec3::splat(0.0), + Vec3::new(165.0, 330.0, 165.0), + white.clone(), + ) + .rotate_y(15.0) + .translate(Vec3::new(265.0, 0.0, 295.0)), + Isotropic::new(Solid::new(Vec3::splat(0.0))), + 0.01, + ))); + world.push(Arc::new(ConstantMedium::new( + Cuboid::new(Vec3::splat(0.0), Vec3::splat(165.0), white) + .rotate_y(-18.0) + .translate(Vec3::new(130.0, 0.0, 65.0)), + Isotropic::new(Solid::new(Vec3::splat(1.0))), + 0.01, + ))); + + BvhNode::new(&mut rng, &mut world, 0.0, 1.0) + } + + fn camera(&self, aspect_ratio: f64) -> Camera { + let lookfrom = Vec3::new(278.0, 278.0, -800.0); + let lookat = Vec3::new(278.0, 278.0, 0.0); + let aperture = 0.1; + let focus_distance = 40.0; + Camera::new( + lookfrom, + lookat, + Vec3::new(0.0, 1.0, 0.0), + 40.0, + aspect_ratio, + aperture, + focus_distance, + 0.0, + 1.0, + ) + } +} diff --git a/src/demos/image_texture.rs b/src/demos/image_texture.rs new file mode 100644 index 0000000..612c8bc --- /dev/null +++ b/src/demos/image_texture.rs @@ -0,0 +1,64 @@ +use std::sync::Arc; + +use rand::{prelude::SmallRng, SeedableRng}; + +use crate::{ + demos::{Demo, ParallelHit}, + hitable::shapes::Sphere, + materials::Lambertian, + texture::ImageTexture, + types::Vec3, + BvhNode, Camera, +}; + +pub struct ImageTextureDemo {} + +impl Demo for ImageTextureDemo { + type DemoT = BvhNode>; + + fn name(&self) -> &'static str { + "image_texture" + } + + fn get_background(&self) -> Vec3 { + Vec3::new(0.7, 0.8, 1.0) + } + + fn world(&self) -> Self::DemoT { + let mut world: Vec> = Vec::with_capacity(1); + + let mut rng = rand::thread_rng(); + let mut rng = SmallRng::from_rng(&mut rng).unwrap(); + + let earth_texture = match ImageTexture::from_filename("assets/earthmap.jpg") { + Ok(v) => v, + Err(e) => panic!("error in creating image texture: {}", e), + }; + + world.push(Arc::new(Sphere::new( + Vec3::new(0.0, 0.0, 0.0), + 2.0, + Lambertian::new(earth_texture), + ))); + + BvhNode::new(&mut rng, &mut world, 0.0, 1.0) + } + + fn camera(&self, aspect_ratio: f64) -> Camera { + let lookfrom = Vec3::new(13.0, 2.0, 3.0); + let lookat = Vec3::new(0.0, 0.0, 0.0); + let aperture = 0.1; + let focus_distance = 12.0; + Camera::new( + lookfrom, + lookat, + Vec3::new(0.0, 1.0, 0.0), + 20.0, + aspect_ratio, + aperture, + focus_distance, + 0.0, + 1.0, + ) + } +} diff --git a/src/demos/instances.rs b/src/demos/instances.rs new file mode 100644 index 0000000..7300d94 --- /dev/null +++ b/src/demos/instances.rs @@ -0,0 +1,117 @@ +use std::sync::Arc; + +use rand::{prelude::SmallRng, SeedableRng}; + +use crate::{ + demos::{Demo, ParallelHit}, + hitable::{ + shapes::{Cuboid, RectBuilder}, + Hitable, + }, + materials::{DiffuseLight, Lambertian, MaterialBuilder}, + texture::Solid, + types::Vec3, + BvhNode, Camera, +}; + +pub struct Instances {} + +impl Demo for Instances { + type DemoT = BvhNode>; + + fn name(&self) -> &'static str { + "instances" + } + + fn world(&self) -> Self::DemoT { + let mut world: Vec> = Vec::with_capacity(8); + + let mut rng = rand::thread_rng(); + let mut rng = SmallRng::from_rng(&mut rng).unwrap(); + + let red = Lambertian::new(Solid::new(Vec3::new(0.65, 0.05, 0.05))); + let white = Lambertian::new(Solid::new(Vec3::splat(0.73))); + let green = Lambertian::new(Solid::new(Vec3::new(0.12, 0.45, 0.15))); + let light = DiffuseLight::new(Solid::new(Vec3::splat(15.0))); + + world.push(Arc::new( + RectBuilder + .y(0.0..=555.0) + .z(0.0..=555.0) + .x(555.0) + .material(green), + )); + world.push(Arc::new( + RectBuilder + .y(0.0..=555.0) + .z(0.0..=555.0) + .x(0.0) + .material(red), + )); + world.push(Arc::new( + RectBuilder + .x(213.0..=343.0) + .z(227.0..=332.0) + .y(554.0) + .material(light), + )); + + world.push(Arc::new( + RectBuilder + .x(0.0..=555.0) + .z(0.0..=555.0) + .y(0.0) + .material(white.clone()), + )); + world.push(Arc::new( + RectBuilder + .x(0.0..=555.0) + .z(0.0..=555.0) + .y(555.0) + .material(white.clone()), + )); + world.push(Arc::new( + RectBuilder + .x(0.0..=555.0) + .y(0.0..=555.0) + .z(555.0) + .material(white.clone()), + )); + + // Add the two boxes + world.push(Arc::new( + Cuboid::new( + Vec3::new(0.0, 0.0, 0.0), + Vec3::new(165.0, 330.0, 165.0), + white.clone(), + ) + .rotate_y(15.0) + .translate(Vec3::new(265.0, 0.0, 295.0)), + )); + world.push(Arc::new( + Cuboid::new(Vec3::new(0.0, 0.0, 0.0), Vec3::splat(165.0), white) + .rotate_y(-18.0) + .translate(Vec3::new(130.0, 0.0, 65.0)), + )); + + BvhNode::new(&mut rng, &mut world, 0.0, 1.0) + } + + fn camera(&self, aspect_ratio: f64) -> Camera { + let lookfrom = Vec3::new(278.0, 278.0, -800.0); + let lookat = Vec3::new(278.0, 278.0, 0.0); + let aperture = 0.1; + let focus_distance = 40.0; + Camera::new( + lookfrom, + lookat, + Vec3::new(0.0, 1.0, 0.0), + 40.0, + aspect_ratio, + aperture, + focus_distance, + 0.0, + 1.0, + ) + } +} diff --git a/src/demos/mod.rs b/src/demos/mod.rs new file mode 100644 index 0000000..374ff75 --- /dev/null +++ b/src/demos/mod.rs @@ -0,0 +1,234 @@ +use crate::{ + hitable::{hitable_list::HitableList, BvhNode, Hitable}, + types::{Color, Vec3}, + Camera, HORIZONTAL_PARTITION, VERTICAL_PARTITION, +}; +use rand::{rngs::SmallRng, Rng, SeedableRng}; +use rayon::prelude::*; +use std::{ + fmt::{Display, Formatter, Result as FmtResult}, + fs::File, + io::Write, + sync::{Arc, Mutex}, +}; + +mod checkered_motion_blur; +mod cornell_box; +mod cornell_smoke_and_fog; +mod image_texture; +mod instances; +mod perlin_noise_ball; +mod simple_light; +mod two_spheres; + +pub use checkered_motion_blur::CheckeredMotionBlur; +pub use cornell_box::CornellBox; +pub use cornell_smoke_and_fog::CornellSmokeAndFog; +pub use image_texture::ImageTextureDemo; +pub use instances::Instances; +pub use perlin_noise_ball::PerlinNoiseBall; +pub use simple_light::SimpleLight; +pub use two_spheres::TwoSpheres; + +#[derive(Debug)] +pub struct Chunk { + num: usize, + x: usize, + y: usize, + nx: usize, + ny: usize, + start_x: usize, + start_y: usize, + buffer: Vec, +} + +impl Display for Chunk { + fn fmt(&self, f: &mut Formatter) -> FmtResult { + write!( + f, + "Chunk #{}: Start X = {} Start Y = {} Size X = {} Size = {}", + self.num, self.start_x, self.start_y, self.nx, self.ny + ) + } +} + +pub trait ParallelHit: Hitable + Send + Sync {} +impl ParallelHit for T {} + +pub trait Demo: Send + Sync { + type DemoT: Hitable + Send + Sync; + + fn name(&self) -> &'static str; + + fn world(&self) -> Self::DemoT; + + fn camera(&self, aspect_ratio: f64) -> Camera; + + fn get_background(&self) -> Vec3 { + Vec3::new(0.0, 0.0, 0.0) + } + + fn render_chunk(&self, chunk: &mut Chunk, camera: &Camera, world: &Self::DemoT, samples: u16) { + let &mut Chunk { + num: _, + x, + y, + nx, + ny, + start_x, + start_y, + ref mut buffer, + } = chunk; + let mut offset = 0; + let mut rng = rand::thread_rng(); + let mut rng = SmallRng::from_rng(&mut rng).unwrap(); + let background = self.get_background(); + + assert!(buffer.len() >= nx * ny * 4); + + (start_y..start_y + ny).for_each(|j| { + (start_x..start_x + nx).for_each(|i| { + let mut color = Vec3::new(0.0, 0.0, 0.0); + for _s in 0..samples { + let u = (i as f64 + rng.gen::()) / x as f64; + let v = (j as f64 + rng.gen::()) / y as f64; + + let ray = camera.get_ray(u, v, &mut rng); + color += ray.color(world, &mut rng, &background, 0); + } + + color /= samples as f64; + self.update_rgb(buffer, color, offset); + offset += 4; + }); + }); + } + + fn render(&self, buf: &mut Vec, x: usize, y: usize, samples: u16) { + let world = self.world(); + let delta_x = x / VERTICAL_PARTITION; + let delta_y = y / HORIZONTAL_PARTITION; + let remx = x % VERTICAL_PARTITION; + let remy = y % HORIZONTAL_PARTITION; + + // There can be tiny error here if the canvas height/width is not perfectly divisible + // by vertical/horizontal partitions in the chunks around the edges + // but umm, i'll just ignore those for now. + let camera = self.camera(delta_x as f64 / delta_y as f64); + let buf = Arc::new(Mutex::new(buf)); + + (0..VERTICAL_PARTITION).into_par_iter().for_each(|j| { + let buf = buf.clone(); + (0..HORIZONTAL_PARTITION).into_par_iter().for_each(|i| { + let mut nx = delta_x; + let mut ny = delta_y; + let start_y = j * ny; + let start_x = i * nx; + + match (i + 1, j + 1) { + (HORIZONTAL_PARTITION, VERTICAL_PARTITION) => { + nx += remx; + ny += remy; + } + (HORIZONTAL_PARTITION, _) => nx += remx, + (_, VERTICAL_PARTITION) => ny += remy, + _ => (), + }; + + let mut chunk = Chunk { + num: j * HORIZONTAL_PARTITION + i, + x, + y, + nx, + ny, + start_x, + start_y, + buffer: vec![0; nx * ny * 4], + }; + + println!("{}", chunk); + self.render_chunk(&mut chunk, &camera, &world, samples); + + let mut buf = buf.lock().unwrap(); + let mut temp_offset = 0; + for j in start_y..start_y + ny { + let real_offset = ((y - j - 1) * x + start_x) * 4; + + buf[real_offset..real_offset + nx * 4] + .copy_from_slice(&chunk.buffer[temp_offset..temp_offset + nx * 4]); + + temp_offset += nx * 4; + } + + println!("Rendered {}", chunk); + }); + }); + } + + #[inline] + fn update_rgb(&self, buffer: &mut [u8], color: Vec3, offset: usize) { + let color: Color = color.into(); + + if let Some(pos) = buffer.get_mut(offset) { + *pos = color.0; + } + if let Some(pos) = buffer.get_mut(offset + 1) { + *pos = color.1 + } + if let Some(pos) = buffer.get_mut(offset + 2) { + *pos = color.2; + } + } + + fn save_as_ppm(&self, buf: &[u8], width: usize, height: usize, samples: u16) { + let header = format!("P3\n{} {}\n255\n", width, height); + + let mut file = match File::create(&format!( + "{}-{}x{}_{}.ppm", + self.name(), + width, + height, + samples, + )) { + Ok(file) => file, + Err(e) => panic!("couldn't create {}: {}", self.name(), e), + }; + file.write_all(header.as_bytes()) + .expect("error in writing file header"); + + for i in buf.chunks(4) { + match file.write_all(format!("{} {} {}\n", i[0], i[1], i[2]).as_bytes()) { + Ok(_) => (), + Err(e) => panic!("couldn't write to {}: {}", self.name(), e), + } + } + } +} + +pub enum DemoWrapper { + HitableList(Box>), + BVHNode(Box>>>), +} + +impl DemoWrapper { + pub fn name(&self) -> &'static str { + match self { + DemoWrapper::HitableList(v) => v.name(), + DemoWrapper::BVHNode(v) => v.name(), + } + } + + pub fn save_as_ppm(&self, buf: &[u8], width: usize, height: usize, samples: u16) { + match self { + DemoWrapper::HitableList(v) => v.save_as_ppm(buf, width, height, samples), + DemoWrapper::BVHNode(v) => v.save_as_ppm(buf, width, height, samples), + } + } + + pub fn render(&self, buf: &mut Vec, x: usize, y: usize, samples: u16) { + match self { + DemoWrapper::HitableList(v) => v.render(buf, x, y, samples), + DemoWrapper::BVHNode(v) => v.render(buf, x, y, samples), + } + } +} diff --git a/src/demos/perlin_noise_ball.rs b/src/demos/perlin_noise_ball.rs new file mode 100644 index 0000000..299e88a --- /dev/null +++ b/src/demos/perlin_noise_ball.rs @@ -0,0 +1,65 @@ +use std::sync::Arc; + +use rand::{prelude::SmallRng, SeedableRng}; + +use crate::{ + demos::{Demo, ParallelHit}, + hitable::{shapes::Sphere, BvhNode}, + materials::Lambertian, + texture::PerlinNoise, + types::Vec3, + Camera, +}; + +pub struct PerlinNoiseBall {} + +impl Demo for PerlinNoiseBall { + type DemoT = BvhNode>; + + fn name(&self) -> &'static str { + "perlin_noise" + } + + fn get_background(&self) -> Vec3 { + Vec3::new(0.7, 0.8, 1.0) + } + + fn world(&self) -> Self::DemoT { + let mut world: Vec> = Vec::with_capacity(2); + + let mut rng = rand::thread_rng(); + let mut rng = SmallRng::from_rng(&mut rng).unwrap(); + + world.push(Arc::new(Sphere::new( + Vec3::new(0.0, -1000.0, 0.0), + 1000.0, + Lambertian::new(PerlinNoise::with_scale(&mut rng, 4.0)), + ))); + + world.push(Arc::new(Sphere::new( + Vec3::new(0.0, 2.0, 0.0), + 2.0, + Lambertian::new(PerlinNoise::with_scale(&mut rng, 4.0)), + ))); + + BvhNode::new(&mut rng, &mut world, 0.0, 1.0) + } + + fn camera(&self, aspect_ratio: f64) -> Camera { + let lookfrom = Vec3::new(13.0, 2.0, 3.0); + let lookat = Vec3::new(0.0, 0.0, 0.0); + let aperture = 0.1; + let focus_distance = 10.0; + Camera::new( + lookfrom, + lookat, + Vec3::new(0.0, 1.0, 0.0), + 20.0, + aspect_ratio, + aperture, + focus_distance, + 0.0, + 1.0, + ) + } +} diff --git a/src/demos/simple_light.rs b/src/demos/simple_light.rs new file mode 100644 index 0000000..9c67786 --- /dev/null +++ b/src/demos/simple_light.rs @@ -0,0 +1,80 @@ +use std::sync::Arc; + +use rand::{prelude::SmallRng, SeedableRng}; + +use crate::{ + demos::{Demo, ParallelHit}, + hitable::{ + shapes::{RectBuilder, Sphere}, + BvhNode, + }, + materials::{DiffuseLight, Lambertian, MaterialBuilder}, + texture::{PerlinNoise, Solid}, + types::Vec3, + Camera, +}; + +pub struct SimpleLight {} + +impl Demo for SimpleLight { + type DemoT = BvhNode>; + + fn name(&self) -> &'static str { + "simple_light" + } + + fn world(&self) -> Self::DemoT { + let mut world: Vec> = Vec::with_capacity(5); + let mut rng = rand::thread_rng(); + let mut rng = SmallRng::from_rng(&mut rng).unwrap(); + + world.push(Arc::new(Sphere::new( + Vec3::new(0.0, -1000.0, 0.0), + 1000.0, + Lambertian::new(PerlinNoise::with_scale(&mut rng, 4.0)), + ))); + world.push(Arc::new(Sphere::new( + Vec3::new(0.0, 2.0, 0.0), + 2.0, + Lambertian::new(PerlinNoise::with_scale(&mut rng, 4.0)), + ))); + + world.push(Arc::new( + RectBuilder + .x(3.0..=5.0) + .y(1.0..=3.0) + .z(-2.0) + .material(DiffuseLight::new(Solid::new(Vec3::new(4.0, 4.0, 4.0)))), + )); + world.push(Arc::new(Sphere::new( + Vec3::new(0.0, 7.0, 0.0), + 2.0, + DiffuseLight::new(Solid::new(Vec3::new(4.0, 4.0, 4.0))), + ))); + world.push(Arc::new(Sphere::new( + Vec3::new(-40.0, 2.0, 5.0), + 1.0, + DiffuseLight::new(Solid::new(Vec3::new(4.0, 4.0, 4.0))), + ))); + + BvhNode::new(&mut rng, &mut world, 0.0, 1.0) + } + + fn camera(&self, aspect_ratio: f64) -> crate::Camera { + let lookfrom = Vec3::new(26.0, 3.0, 6.0); + let lookat = Vec3::new(0.0, 2.0, 0.0); + let aperture = 0.0; + let focus_distance = 10.0; + Camera::new( + lookfrom, + lookat, + Vec3::new(0.0, 1.0, 0.0), + 20.0, + aspect_ratio, + aperture, + focus_distance, + 0.0, + 1.0, + ) + } +} diff --git a/src/demos/two_spheres.rs b/src/demos/two_spheres.rs new file mode 100644 index 0000000..76dfe3e --- /dev/null +++ b/src/demos/two_spheres.rs @@ -0,0 +1,71 @@ +use std::sync::Arc; + +use rand::{prelude::SmallRng, SeedableRng}; + +use crate::{ + demos::{Demo, ParallelHit}, + hitable::{shapes::Sphere, BvhNode}, + materials::Lambertian, + texture::{Checker, Solid}, + types::Vec3, + Camera, +}; + +pub struct TwoSpheres {} + +impl Demo for TwoSpheres { + type DemoT = BvhNode>; + + fn name(&self) -> &'static str { + "two_checkered_sphere" + } + + fn get_background(&self) -> Vec3 { + Vec3::new(0.7, 0.8, 1.0) + } + + fn world(&self) -> Self::DemoT { + let mut world: Vec> = Vec::with_capacity(2); + + let mut rng = rand::thread_rng(); + let mut rng = SmallRng::from_rng(&mut rng).unwrap(); + + world.push(Arc::new(Sphere::new( + Vec3::new(0.0, -10.0, 0.0), + 10.0, + Lambertian::new(Checker::new( + Solid::new(Vec3::new(0.2, 0.3, 0.1)), + Solid::new(Vec3::new(0.9, 0.9, 0.9)), + )), + ))); + + world.push(Arc::new(Sphere::new( + Vec3::new(0.0, 10.0, 0.0), + 10.0, + Lambertian::new(Checker::new( + Solid::new(Vec3::new(0.2, 0.3, 0.1)), + Solid::new(Vec3::new(0.9, 0.9, 0.9)), + )), + ))); + + BvhNode::new(&mut rng, &mut world, 0.0, 1.0) + } + + fn camera(&self, aspect_ratio: f64) -> Camera { + let lookfrom = Vec3::new(13.0, 2.0, 3.0); + let lookat = Vec3::new(0.0, 0.0, 0.0); + let aperture = 0.1; + let focus_distance = 10.0; + Camera::new( + lookfrom, + lookat, + Vec3::new(0.0, 1.0, 0.0), + 20.0, + aspect_ratio, + aperture, + focus_distance, + 0.0, + 1.0, + ) + } +} diff --git a/src/hitable/bvh.rs b/src/hitable/bvh.rs new file mode 100644 index 0000000..fa4b6f9 --- /dev/null +++ b/src/hitable/bvh.rs @@ -0,0 +1,139 @@ +use std::cmp::Ordering; + +use rand::{prelude::SliceRandom, Rng}; + +use crate::{ + hitable::{HitRecord, Hitable}, + types::Ray, + Aabb, +}; + +pub struct BvhNode { + bounding_box: Aabb, + left: HitNode, + right: HitNode, +} + +impl BvhNode { + pub fn new(rng: &mut R, objects: &mut [T], t0: f64, t1: f64) -> Self { + let comparator = [ + Self::box_x_compare, + Self::box_y_compare, + Self::box_z_compare, + ] + .choose(rng) + .unwrap(); + + let (left, right) = match objects.len() { + 1 => ( + HitNode::Direct(objects[0].clone()), + HitNode::Direct(objects[0].clone()), + ), + 2 => match comparator(&objects[0], &objects[1]) { + Ordering::Greater => ( + HitNode::Direct(objects[1].clone()), + HitNode::Direct(objects[0].clone()), + ), + _ => ( + HitNode::Direct(objects[0].clone()), + HitNode::Direct(objects[1].clone()), + ), + }, + + n => { + objects.sort_by(comparator); + let (l, r) = objects.split_at_mut(n / 2); + ( + HitNode::Bvh(Box::new(BvhNode::new(rng, l, t0, t1))), + HitNode::Bvh(Box::new(BvhNode::new(rng, r, t0, t1))), + ) + } + }; + + let left_box = left + .bounding_box(t0, t1) + .expect("missing bounding box for left BVH Node"); + let right_box = right + .bounding_box(t0, t1) + .expect("missing bounding box for right BVH Node"); + + Self { + left, + right, + bounding_box: Aabb::surrounding_box(left_box, right_box), + } + } + + fn box_x_compare(obj1: &T, obj2: &T) -> Ordering { + if let (Some(bbox_a), Some(bbox_b)) = + (obj1.bounding_box(0.0, 0.0), obj2.bounding_box(0.0, 0.0)) + { + return bbox_a.min.x().partial_cmp(&bbox_b.min.x()).unwrap(); + } + + panic!("No bounding box for this BVH Node!!") + } + + fn box_y_compare(obj1: &T, obj2: &T) -> Ordering { + if let (Some(bbox_a), Some(bbox_b)) = + (obj1.bounding_box(0.0, 0.0), obj2.bounding_box(0.0, 0.0)) + { + return bbox_a.min.y().partial_cmp(&bbox_b.min.y()).unwrap(); + } + + panic!("No bounding box for this BVH Node!!") + } + + fn box_z_compare(obj1: &T, obj2: &T) -> Ordering { + if let (Some(bbox_a), Some(bbox_b)) = + (obj1.bounding_box(0.0, 0.0), obj2.bounding_box(0.0, 0.0)) + { + return bbox_a.min.z().partial_cmp(&bbox_b.min.z()).unwrap(); + } + + panic!("No bounding box for this BVH Node!!") + } +} + +impl Hitable for BvhNode { + fn hit(&self, ray: &Ray, t_min: f64, t_max: f64) -> Option { + if !self.bounding_box.hit(ray, t_min, t_max) { + return None; + } + + let hbox_left = self.left.hit(ray, t_min, t_max); + + let hbox_right = if let Some(ref hleft) = hbox_left { + self.right.hit(ray, t_min, hleft.t) + } else { + self.right.hit(ray, t_min, t_max) + }; + + hbox_right.or(hbox_left) + } + + fn bounding_box(&self, _t_min: f64, _t_max: f64) -> Option { + Some(self.bounding_box) + } +} + +enum HitNode { + Bvh(Box>), + Direct(T), +} + +impl HitNode { + fn hit(&self, ray: &Ray, t_min: f64, t_max: f64) -> Option { + match self { + HitNode::Bvh(node) => node.hit(ray, t_min, t_max), + HitNode::Direct(node) => node.hit(ray, t_min, t_max), + } + } + + fn bounding_box(&self, t0: f64, t1: f64) -> Option { + match self { + HitNode::Bvh(node) => node.bounding_box(t0, t1), + HitNode::Direct(node) => node.bounding_box(t0, t1), + } + } +} diff --git a/src/hitable/hitable_list.rs b/src/hitable/hitable_list.rs new file mode 100644 index 0000000..4eff7b6 --- /dev/null +++ b/src/hitable/hitable_list.rs @@ -0,0 +1,54 @@ +use std::sync::Arc; + +use crate::{ + demos::ParallelHit, + hitable::{HitRecord, Hitable}, + types::Ray, + Aabb, +}; + +pub struct HitableList { + pub list: Vec>, +} + +impl Hitable for HitableList { + fn hit(&self, ray: &Ray, t_min: f64, t_max: f64) -> Option { + let mut closest_so_far = t_max; + let mut hit_rec: Option = None; + for obj in &self.list { + if let Some(l_hit_rec) = obj.hit(ray, t_min, closest_so_far) { + closest_so_far = l_hit_rec.t; + hit_rec = Some(l_hit_rec); + } + } + hit_rec + } + + fn bounding_box(&self, t0: f64, t1: f64) -> Option { + if self.list.is_empty() { + return None; + } + + let mut output_box = None; + + for obj in self.list.iter() { + if let Some(bbox) = obj.bounding_box(t0, t1) { + if let Some(ref mut opbox) = output_box { + *opbox = Aabb::surrounding_box(*opbox, bbox); + } else { + output_box = Some(bbox); + } + } else { + return output_box; + } + } + + output_box + } +} + +impl HitableList { + pub fn push(&mut self, obj: Arc) { + self.list.push(obj); + } +} diff --git a/src/hitable/mod.rs b/src/hitable/mod.rs new file mode 100644 index 0000000..efd0bb1 --- /dev/null +++ b/src/hitable/mod.rs @@ -0,0 +1,117 @@ +pub mod bvh; +pub mod hitable_list; +mod rotate; +pub mod shapes; +mod translate; +pub mod volume; + +pub use bvh::*; +pub use translate::*; + +use std::sync::Arc; + +use crate::{ + hitable::rotate::Rotate, + types::{Ray, Vec3}, + Aabb, Material, X, Y, Z, +}; + +pub struct HitRecord<'a> { + /// Rays are represented by A + t * B + /// where A is the source point and B destination point + /// by adjusting t we can move forward/back on the ray + /// + /// t is the point at which a ray intersected another object. + /// As in, If we put this value of t in A + t * B equation, We'll get the exact + /// point at which a ray intersects some other object + pub t: f64, + /// Ray object otherwise is represented by the Source/Destination points + /// p is what we get when we perform the operation, A + t * B + /// i.e. A vector from Ray source to the point t + pub p: Vec3, + + /// unit outward facing normal + pub normal: Vec3, + + /// material if any of the surface + pub material: &'a dyn Material, + + /// texture coordinates for an object + pub u: f64, + pub v: f64, + + pub front_face: bool, +} + +impl<'a> HitRecord<'a> { + pub fn new( + t: f64, + p: Vec3, + normal: Vec3, + material: &'a dyn Material, + (u, v): (f64, f64), + ) -> Self { + Self { + t, + p, + normal, + material, + u, + v, + front_face: false, + } + } + + pub fn set_face_normal(&mut self, ray: &Ray) { + self.front_face = ray.direction.dot(&self.normal) < 0.0; + + self.normal = if self.front_face { + self.normal + } else { + -self.normal + } + } +} + +pub trait Hitable { + fn hit(&self, _ray: &Ray, _t_min: f64, _t_max: f64) -> Option; + + fn bounding_box(&self, _t0: f64, _t1: f64) -> Option; + + fn translate(self, offset: impl Into) -> Translate + where + Self: Sized, + { + Translate::new(self, offset.into()) + } + + fn rotate_x(self, angle: f64) -> Rotate + where + Self: Sized, + { + Rotate::new(self, angle) + } + + fn rotate_y(self, angle: f64) -> Rotate + where + Self: Sized, + { + Rotate::new(self, angle) + } + + fn rotate_z(self, angle: f64) -> Rotate + where + Self: Sized, + { + Rotate::new(self, angle) + } +} + +impl Hitable for Arc { + fn hit(&self, ray: &Ray, t_min: f64, t_max: f64) -> Option { + self.as_ref().hit(ray, t_min, t_max) + } + fn bounding_box(&self, t0: f64, t1: f64) -> Option { + self.as_ref().bounding_box(t0, t1) + } +} diff --git a/src/hitable/rotate.rs b/src/hitable/rotate.rs new file mode 100644 index 0000000..60deb4d --- /dev/null +++ b/src/hitable/rotate.rs @@ -0,0 +1,130 @@ +use std::marker::PhantomData; + +use crate::{ + hitable::{HitRecord, Hitable}, + types::{Ray, Vec3}, + Aabb, Dimension, +}; + +pub struct Rotate { + hitable: T, + sin_theta: f64, + cos_theta: f64, + bbox: Option, + + _tag: PhantomData<(D1, D2, D3)>, +} + +impl Rotate +where + D1: Dimension, + D2: Dimension, + D3: Dimension, + T: Hitable, +{ + pub fn new(object: T, angle: f64) -> Rotate { + let radians = angle.to_radians(); + let sin_theta = radians.sin(); + let cos_theta = radians.cos(); + + let mut min = Vec3::splat(f64::MAX); + let mut max = Vec3::splat(f64::MIN); + + let bbox = if let Some(bbox) = object.bounding_box(0.0, 1.0) { + for i in 0..2 { + let i = i as f64; + for j in 0..2 { + let j = j as f64; + for k in 0..2 { + let k = k as f64; + + // D1 will be the axis about which we are rotating + let d1 = i * bbox.max.get::() + (1.0 - i) * bbox.min.get::(); + + let d2 = j * bbox.max.get::() + (1.0 - j) * bbox.min.get::(); + let d3 = k * bbox.max.get::() + (1.0 - k) * bbox.min.get::(); + + let new_d2 = cos_theta * d2 + sin_theta * d3; + let new_d3 = -sin_theta * d2 + cos_theta * d3; + + let tester = Vec3::splat(0.0) + .set::(d1) + .set::(new_d2) + .set::(new_d3); + + min = Vec3::min(tester, min); + max = Vec3::max(tester, max); + } + } + } + + Aabb::new(min, max) + } else { + Aabb::new(min, max) + }; + + Rotate { + hitable: object, + sin_theta, + cos_theta, + bbox: Some(bbox), + _tag: PhantomData, + } + } +} + +impl Hitable for Rotate +where + D1: Dimension, + D2: Dimension, + D3: Dimension, + T: Hitable, +{ + fn hit(&self, ray: &Ray, t_min: f64, t_max: f64) -> Option { + let origin = ray + .origin + .set::( + self.cos_theta * ray.origin.get::() - self.sin_theta * ray.origin.get::(), + ) + .set::( + self.sin_theta * ray.origin.get::() + self.cos_theta * ray.origin.get::(), + ); + + let direction = ray + .direction + .set::( + self.cos_theta * ray.direction.get::() + - self.sin_theta * ray.direction.get::(), + ) + .set::( + self.sin_theta * ray.direction.get::() + + self.cos_theta * ray.direction.get::(), + ); + + let rotated_ray = Ray::new(origin, direction, ray.time()); + + let mut hit = self.hitable.hit(&rotated_ray, t_min, t_max)?; + + hit.p = hit + .p + .set::(self.cos_theta * hit.p.get::() + self.sin_theta * hit.p.get::()) + .set::(-self.sin_theta * hit.p.get::() + self.cos_theta * hit.p.get::()); + + hit.normal = hit + .normal + .set::( + self.cos_theta * hit.normal.get::() + self.sin_theta * hit.normal.get::(), + ) + .set::( + -self.sin_theta * hit.normal.get::() + self.cos_theta * hit.normal.get::(), + ); + + hit.set_face_normal(&rotated_ray); + + Some(hit) + } + + fn bounding_box(&self, _t0: f64, _t1: f64) -> Option { + self.bbox + } +} diff --git a/src/hitable/shapes/cuboid.rs b/src/hitable/shapes/cuboid.rs new file mode 100644 index 0000000..767a690 --- /dev/null +++ b/src/hitable/shapes/cuboid.rs @@ -0,0 +1,87 @@ +use std::sync::Arc; + +use crate::{ + hitable::{hitable_list::HitableList, shapes::RectBuilder, HitRecord, Hitable}, + materials::{Material, MaterialBuilder}, + types::{Ray, Vec3}, + Aabb, +}; + +pub struct Cuboid { + min: Vec3, + max: Vec3, + sides: HitableList, +} + +impl Cuboid { + pub fn new(p0: Vec3, p1: Vec3, mat: impl Material + Clone + 'static) -> Self { + Self { + min: p0, + max: p1, + sides: Self::build_cuboid(p0, p1, mat), + } + } + + fn build_cuboid(p0: Vec3, p1: Vec3, mat: impl Material + Clone + 'static) -> HitableList { + let mut sides = HitableList { + list: Vec::with_capacity(6), + }; + + sides.push(Arc::new( + RectBuilder + .x(p0.x()..=p1.x()) + .y(p0.y()..=p1.y()) + .z(p1.z()) + .material(mat.clone()), + )); + sides.push(Arc::new( + RectBuilder + .x(p0.x()..=p1.x()) + .y(p0.y()..=p1.y()) + .z(p0.z()) + .material(mat.clone()), + )); + + sides.push(Arc::new( + RectBuilder + .x(p0.x()..=p1.x()) + .z(p0.z()..=p1.z()) + .y(p1.y()) + .material(mat.clone()), + )); + sides.push(Arc::new( + RectBuilder + .x(p0.x()..=p1.x()) + .z(p0.z()..=p1.z()) + .y(p0.y()) + .material(mat.clone()), + )); + + sides.push(Arc::new( + RectBuilder + .y(p0.y()..=p1.y()) + .z(p0.z()..=p1.z()) + .x(p1.x()) + .material(mat.clone()), + )); + sides.push(Arc::new( + RectBuilder + .y(p0.y()..=p1.y()) + .z(p0.z()..=p1.z()) + .x(p0.x()) + .material(mat), + )); + + sides + } +} + +impl Hitable for Cuboid { + fn hit(&self, ray: &Ray, t_min: f64, t_max: f64) -> Option { + self.sides.hit(ray, t_min, t_max) + } + + fn bounding_box(&self, _t0: f64, _t1: f64) -> Option { + Some(Aabb::new(self.min, self.max)) + } +} diff --git a/src/hitable/shapes/mod.rs b/src/hitable/shapes/mod.rs new file mode 100644 index 0000000..2e0a28a --- /dev/null +++ b/src/hitable/shapes/mod.rs @@ -0,0 +1,9 @@ +mod cuboid; +mod moving_sphere; +mod rectangle; +mod sphere; + +pub use cuboid::Cuboid; +pub use moving_sphere::MovingSphere; +pub use rectangle::RectBuilder; +pub use sphere::Sphere; diff --git a/src/hitable/shapes/moving_sphere.rs b/src/hitable/shapes/moving_sphere.rs new file mode 100644 index 0000000..45ac5d4 --- /dev/null +++ b/src/hitable/shapes/moving_sphere.rs @@ -0,0 +1,92 @@ +use crate::{ + hitable::{HitRecord, Hitable}, + types::{Ray, Vec3}, + Aabb, Material, +}; + +pub struct MovingSphere { + radius: f64, + center_start: Vec3, + center_end: Vec3, + time_start: f64, + time_end: f64, + material: T, +} + +impl MovingSphere { + pub fn new( + center_start: Vec3, + center_end: Vec3, + time_start: f64, + time_end: f64, + radius: f64, + material: T, + ) -> Self { + Self { + radius, + center_start, + center_end, + time_start, + time_end, + material, + } + } + + fn center(&self, time: f64) -> Vec3 { + self.center_start + + (self.center_end - self.center_start) + * ((time - self.time_start) / (self.time_end - self.time_start)) + } + + /// p is a point on the sphere of radius 1 & center at origin + /// u is between [0,1]. Angle around Y axis from -X axis + /// v is between [0,1]. Angle from -Y to +Y axis + pub fn get_uv(p: Vec3) -> (f64, f64) { + let theta = (-p.y()).acos(); + let phi = f64::atan2(-p.z(), p.x()) + std::f64::consts::PI; + + let u = phi / (2.0 * std::f64::consts::PI); + let v = theta / std::f64::consts::PI; + + (u, v) + } +} + +impl Hitable for MovingSphere { + fn hit(&self, ray: &Ray, t_min: f64, t_max: f64) -> Option { + let oc = ray.origin - self.center(ray.time()); + let a = ray.direction.dot(&ray.direction); + let b = oc.dot(&ray.direction); + let c = oc.dot(&oc) - self.radius * self.radius; + + let discriminant = b * b - a * c; + let discriminant_root = discriminant.sqrt(); + + if discriminant > 0.0 { + let mut root = (-b - discriminant_root) / a; + if root < t_min || root > t_max { + root = (-b + discriminant_root) / a; + } + if root > t_min && root < t_max { + let p = ray.point_at_parameter(root); + let normal = (p - self.center(ray.time())) / self.radius; + + let mut hit_rec = + HitRecord::new(root, p, normal, &self.material, Self::get_uv(normal)); + + hit_rec.set_face_normal(ray); + + return Some(hit_rec); + } + } + None + } + + fn bounding_box(&self, t0: f64, t1: f64) -> Option { + let radius = Vec3::new(self.radius, self.radius, self.radius); + let box_smol = Aabb::new(self.center(t0) - radius, self.center(t0) + radius); + let box_big = Aabb::new(self.center(t1) - radius, self.center(t1) + radius); + + Some(Aabb::surrounding_box(box_smol, box_big)) + } +} diff --git a/src/hitable/shapes/rectangle.rs b/src/hitable/shapes/rectangle.rs new file mode 100644 index 0000000..f26ffd4 --- /dev/null +++ b/src/hitable/shapes/rectangle.rs @@ -0,0 +1,175 @@ +use std::{marker::PhantomData, ops::RangeInclusive}; + +use crate::{ + hitable::{HitRecord, Hitable}, + materials::MaterialBuilder, + types::{Ray, Vec3}, + Aabb, Dimension, Material, X, Y, Z, +}; + +type DimRange = RangeInclusive; + +pub struct Rectangle { + d1_range: DimRange, + d2_range: DimRange, + d3: f64, + material: T, + tag: PhantomData<(D1, D2, D3)>, +} + +impl Hitable for Rectangle +where + T: Material, + D1: Dimension, + D2: Dimension, + D3: Dimension, +{ + fn hit(&self, ray: &Ray, t_min: f64, t_max: f64) -> Option { + let t = (self.d3 - ray.origin.get::()) / ray.direction.get::(); + + if t < t_min || t > t_max { + return None; + } + + let d1 = ray.origin.get::() + t * ray.direction.get::(); + let d2 = ray.origin.get::() + t * ray.direction.get::(); + + if !self.d1_range.contains(&d1) || !self.d2_range.contains(&d2) { + return None; + } + + let u = (d1 - self.d1_range.start()) / (self.d1_range.end() - self.d1_range.start()); + let v = (d2 - self.d2_range.start()) / (self.d2_range.end() - self.d2_range.start()); + + let mut hit_rec = HitRecord::new( + t, + ray.point_at_parameter(t), + Vec3::splat(0.0).set::(1.0), + &self.material, + (u, v), + ); + + hit_rec.set_face_normal(ray); + + Some(hit_rec) + } + + fn bounding_box(&self, _t0: f64, _t1: f64) -> Option { + // Since this is a axis aligned Rectangle and we are using AABB BVH, Gap between the rectangle and + // the bounding box will be infinitely small + + let (&d1_0, &d1_1) = (self.d1_range.start(), self.d1_range.end()); + let (&d2_0, &d2_1) = (self.d2_range.start(), self.d2_range.end()); + + let min = Vec3::splat(self.d3 - 0.0001) + .set::(d1_0) + .set::(d2_0); + let max = Vec3::splat(self.d3 + 0.0001) + .set::(d1_1) + .set::(d2_1); + + Some(Aabb::new(min, max)) + } +} + +// taken from, https://github.com/Globidev/toy-rt/blob/master/trt-core/src/hit/rect.rs#L74 +// because it's amazing! + +pub struct RectBuilder; + +macro_rules! builder { + ($name:ident, $dim:ty) => { + pub fn $name(self, range: RangeInclusive) -> OneBoundedRectBuilder<$dim> { + OneBoundedRectBuilder { + range, + tag: PhantomData, + } + } + }; +} + +pub struct OneBoundedRectBuilder { + range: DimRange, + tag: PhantomData, +} + +impl RectBuilder { + builder!(x, X); + builder!(y, Y); + builder!(z, Z); +} + +macro_rules! one_bounded_rect_builder { + ($name:ident, $dim1: ty, $dim2: ty) => { + pub fn $name(self, d2_range: DimRange) -> TwoBoundedRectBuilder<$dim1, $dim2> { + TwoBoundedRectBuilder { + d1_range: self.range, + d2_range, + tag: PhantomData, + } + } + }; +} + +impl OneBoundedRectBuilder { + one_bounded_rect_builder!(y, X, Y); + one_bounded_rect_builder!(z, X, Z); +} +impl OneBoundedRectBuilder { + one_bounded_rect_builder!(x, Y, X); + one_bounded_rect_builder!(z, Y, Z); +} +impl OneBoundedRectBuilder { + one_bounded_rect_builder!(x, Z, X); + one_bounded_rect_builder!(y, Z, Y); +} + +pub struct TwoBoundedRectBuilder { + d1_range: DimRange, + d2_range: DimRange, + tag: PhantomData<(D1, D2)>, +} + +macro_rules! two_bounded_rect_builder { + ($name:ident, $dim1: ty, $dim2: ty, $dim3: ty) => { + pub fn $name(self, $name: f64) -> ThreeBoundedRectBuilder<$dim1, $dim2, $dim3> { + ThreeBoundedRectBuilder { + d1_range: self.d1_range, + d2_range: self.d2_range, + d3: $name, + tag: PhantomData, + } + } + }; +} + +impl TwoBoundedRectBuilder { + two_bounded_rect_builder!(z, X, Y, Z); +} +impl TwoBoundedRectBuilder { + two_bounded_rect_builder!(y, X, Z, Y); +} +impl TwoBoundedRectBuilder { + two_bounded_rect_builder!(x, Y, Z, X); +} + +pub struct ThreeBoundedRectBuilder { + d1_range: DimRange, + d2_range: DimRange, + d3: f64, + tag: PhantomData<(D1, D2, D3)>, +} + +impl MaterialBuilder for ThreeBoundedRectBuilder { + type Finished = Rectangle; + + fn material(self, material: T) -> Self::Finished { + Rectangle { + d1_range: self.d1_range, + d2_range: self.d2_range, + d3: self.d3, + material, + tag: PhantomData, + } + } +} diff --git a/src/hitable/shapes/sphere.rs b/src/hitable/shapes/sphere.rs new file mode 100644 index 0000000..c4b03e7 --- /dev/null +++ b/src/hitable/shapes/sphere.rs @@ -0,0 +1,77 @@ +use crate::{ + hitable::{HitRecord, Hitable}, + types::{Ray, Vec3}, + Aabb, Material, +}; + +#[derive(Clone)] +pub struct Sphere { + center: Vec3, + radius: f64, + material: T, +} + +impl Sphere { + pub fn new(center: Vec3, radius: f64, material: T) -> Self { + Self { + center, + radius, + material, + } + } + + /// p is a point on the sphere of radius 1 & center at origin + /// u is between [0,1]. Angle around Y axis from -X axis + /// v is between [0,1]. Angle from -Y to +Y axis + pub fn get_uv(p: Vec3) -> (f64, f64) { + let theta = (-p.y()).acos(); + let phi = f64::atan2(-p.z(), p.x()) + std::f64::consts::PI; + + let u = phi / (2.0 * std::f64::consts::PI); + let v = theta / std::f64::consts::PI; + + (u, v) + } +} + +impl Hitable for Sphere { + fn hit(&self, ray: &Ray, t_min: f64, t_max: f64) -> Option { + let oc = ray.origin - self.center; + let a = ray.direction.dot(&ray.direction); + let b = oc.dot(&ray.direction); + let c = oc.dot(&oc) - self.radius * self.radius; + + // The discriminant is calculated using b^2 - 4 * a * c + // but in this specific case, If we put the equation in the + // formula to find quadratic roots, We can get this shorter + // formula to find the discriminant. + // Check this for detailed proof + // https://vchizhov.github.io/resources/ray%20tracing/ray%20tracing%20tutorial%20series%20vchizhov/ray_casting/part1/intersecting_a_sphere.md.html#appendix + let discriminant = b * b - a * c; + let discriminant_root = discriminant.sqrt(); + + if discriminant > 0.0 { + let mut root = (-b - discriminant_root) / a; + if root < t_min || root > t_max { + root = (-b + discriminant_root) / a; + } + if root > t_min && root < t_max { + let p = ray.point_at_parameter(root); + let normal = (p - self.center) / self.radius; + + let mut hit_rec = + HitRecord::new(root, p, normal, &self.material, Self::get_uv(normal)); + + hit_rec.set_face_normal(ray); + + return Some(hit_rec); + } + } + None + } + + fn bounding_box(&self, _t0: f64, _t1: f64) -> Option { + let radius = Vec3::new(self.radius, self.radius, self.radius); + Some(Aabb::new(self.center - radius, self.center + radius)) + } +} diff --git a/src/hitable/translate.rs b/src/hitable/translate.rs new file mode 100644 index 0000000..5ad11d0 --- /dev/null +++ b/src/hitable/translate.rs @@ -0,0 +1,37 @@ +use crate::{ + hitable::{HitRecord, Hitable}, + types::{Ray, Vec3}, + Aabb, +}; + +pub struct Translate { + object: T, + offset: Vec3, +} + +impl Translate { + pub const fn new(object: T, offset: Vec3) -> Self { + Self { object, offset } + } +} + +impl Hitable for Translate { + fn hit(&self, ray: &Ray, t_min: f64, t_max: f64) -> Option { + let moved_ray = Ray::new(ray.origin - self.offset, ray.direction, ray.time()); + + if let Some(mut hit) = self.object.hit(&moved_ray, t_min, t_max) { + hit.p += self.offset; + hit.set_face_normal(&moved_ray); + + Some(hit) + } else { + None + } + } + + fn bounding_box(&self, t0: f64, t1: f64) -> Option { + self.object + .bounding_box(t0, t1) + .map(|bbox| Aabb::new(bbox.min + self.offset, bbox.max + self.offset)) + } +} diff --git a/src/hitable/volume/constant_medium.rs b/src/hitable/volume/constant_medium.rs new file mode 100644 index 0000000..ecf0c98 --- /dev/null +++ b/src/hitable/volume/constant_medium.rs @@ -0,0 +1,63 @@ +use crate::{ + hitable::{HitRecord, Hitable}, + types::{Ray, Vec3}, + Aabb, Material, +}; + +pub struct ConstantMedium { + neg_inv_density: f64, + boundary: A, + phase_function: B, +} + +impl ConstantMedium { + pub fn new(boundary: A, phase_function: B, d: f64) -> Self { + Self { + boundary, + phase_function, + neg_inv_density: -1.0 / d, + } + } +} + +impl Hitable for ConstantMedium { + fn hit(&self, ray: &Ray, t_min: f64, t_max: f64) -> Option { + let mut hit1 = self.boundary.hit(ray, f64::MIN, f64::MAX)?; + let mut hit2 = self.boundary.hit(ray, hit1.t + 0.0001, f64::MAX)?; + + hit1.t = hit1.t.max(t_min); + hit2.t = hit2.t.min(t_max); + + if hit1.t >= hit2.t { + return None; + }; + + hit1.t = hit1.t.max(0.0); + + let ray_length = ray.direction.length(); + let distance_inside_boundary = (hit2.t - hit1.t) * ray_length; + let hit_distance = self.neg_inv_density * rand::random::().ln(); + + if hit_distance > distance_inside_boundary { + return None; + } + + let t = hit1.t + hit_distance / ray_length; + + Some(HitRecord { + t, + p: ray.point_at_parameter(t), + material: &self.phase_function, + u: 0.0, + v: 0.0, + + // Arbitrary + front_face: true, + normal: Vec3::new(1.0, 0.0, 0.0), + }) + } + + fn bounding_box(&self, t0: f64, t1: f64) -> Option { + self.boundary.bounding_box(t0, t1) + } +} diff --git a/src/hitable/volume/mod.rs b/src/hitable/volume/mod.rs new file mode 100644 index 0000000..e97ff87 --- /dev/null +++ b/src/hitable/volume/mod.rs @@ -0,0 +1,3 @@ +mod constant_medium; + +pub use constant_medium::ConstantMedium; diff --git a/src/main.rs b/src/main.rs index 76d6d58..9929a80 100644 --- a/src/main.rs +++ b/src/main.rs @@ -1,20 +1,195 @@ -use rand::Rng; +#![allow(clippy::suspicious_arithmetic_impl)] -#[inline] -fn pdf(x: f64) -> f64 { - 3.0 * x * x / 8.0 +mod aabb; +mod camera; +mod demos; +mod hitable; +mod materials; +mod texture; +mod types; + +pub use aabb::Aabb; +pub use camera::Camera; +pub use materials::Material; +pub use texture::Texture; +pub use types::{Dimension, X, Y, Z}; + +use crate::hitable::BvhNode; +use demos::DemoWrapper; + +use std::time::Instant; + +pub trait Asf64: num_traits::AsPrimitive {} +impl> Asf64 for T {} + +const NUM_SAMPLES: u16 = 500; +const VERTICAL_PARTITION: usize = 30; +const HORIZONTAL_PARTITION: usize = 30; +const WIDTH: usize = 800; +const HEIGHT: usize = 800; + +fn main() -> Result<(), String> { + run(WIDTH, HEIGHT) } -fn main() { - let mut rng = rand::thread_rng(); - const N: u64 = 100000; - let mut sum = 0.0; +#[cfg(feature = "gui")] +fn run(mut width: usize, mut height: usize) -> Result<(), String> { + use sdl2::{ + event::{Event, WindowEvent}, + keyboard::Keycode, + pixels::PixelFormatEnum, + }; - for _ in 0..N { - let x: f64 = rng.gen_range(0.0f64..=8.0).powf(1.0 / 3.0); + let sdl_ctx = sdl2::init()?; + let video_subsys = sdl_ctx.video()?; + let window = video_subsys + .window("Ray tracing the Next Week", width as u32, height as u32) + .position_centered() + .build() + .map_err(|e| e.to_string())?; - sum += x * x / pdf(x); + let mut event_pump = sdl_ctx.event_pump()?; + + let mut canvas = window + .into_canvas() + .target_texture() + .build() + .map_err(|e| e.to_string())?; + + // RGBA framebuffer + let mut buffer = vec![0; height * width * 4]; + + let texture_creator = canvas.texture_creator(); + let mut texture = texture_creator + .create_texture_static(PixelFormatEnum::BGR888, width as u32, height as u32) + .map_err(|e| e.to_string())?; + + let mut active_demo = DemoWrapper::HitableList(Box::new(demos::CornellBox {})); + let mut should_update = true; + + loop { + for event in event_pump.poll_iter() { + match event { + Event::Quit { .. } + | Event::KeyDown { + keycode: Some(Keycode::Escape), + .. + } => return Ok(()), + Event::KeyUp { keycode, .. } => { + match keycode { + Some(Keycode::S) => { + active_demo.save_as_ppm(&buffer, width, height, NUM_SAMPLES); + should_update = false; + } + Some(Keycode::Num1) => { + active_demo = + DemoWrapper::BVHNode(Box::new(demos::CheckeredMotionBlur {})); + should_update = true; + } + Some(Keycode::Num2) => { + active_demo = DemoWrapper::BVHNode(Box::new(demos::TwoSpheres {})); + should_update = true; + } + Some(Keycode::Num3) => { + active_demo = DemoWrapper::BVHNode(Box::new(demos::PerlinNoiseBall {})); + should_update = true; + } + Some(Keycode::Num4) => { + active_demo = + DemoWrapper::BVHNode(Box::new(demos::ImageTextureDemo {})); + should_update = true; + } + Some(Keycode::Num5) => { + active_demo = DemoWrapper::BVHNode(Box::new(demos::SimpleLight {})); + should_update = true; + } + Some(Keycode::Num6) => { + active_demo = DemoWrapper::BVHNode(Box::new(demos::Instances {})); + should_update = true; + } + Some(Keycode::Num7) => { + active_demo = + DemoWrapper::BVHNode(Box::new(demos::CornellSmokeAndFog {})); + should_update = true; + } + Some(Keycode::Num8) => { + active_demo = DemoWrapper::HitableList(Box::new(demos::CornellBox {})); + should_update = true; + } + None => unreachable!(), + _ => (), + }; + } + Event::Window { + win_event: WindowEvent::Resized(w, h), + .. + } => { + width = w as usize; + height = h as usize; + buffer.resize(width * height * 4, 0); + texture = texture_creator + .create_texture_static(PixelFormatEnum::BGR888, width as u32, height as u32) + .expect("error in resizing texture"); + should_update = true; + } + _ => {} + }; + } + if should_update { + let now = Instant::now(); + active_demo.render(&mut buffer, width, height, NUM_SAMPLES); + println!( + "Demo {} Time Taken(s) = {}", + active_demo.name(), + now.elapsed().as_secs_f64() + ); + texture.update(None, &buffer, width * 4).unwrap(); + canvas.copy(&texture, None, None).unwrap(); + canvas.present(); + should_update = false; + } + } +} + +#[cfg(not(feature = "gui"))] +fn run(width: usize, height: usize) -> Result<(), String> { + let demos: [DemoWrapper; 8] = [ + DemoWrapper::BVHNode(Box::new(demos::CheckeredMotionBlur {})), + DemoWrapper::BVHNode(Box::new(demos::TwoSpheres {})), + DemoWrapper::BVHNode(Box::new(demos::PerlinNoiseBall {})), + DemoWrapper::BVHNode(Box::new(demos::ImageTextureDemo {})), + DemoWrapper::BVHNode(Box::new(demos::SimpleLight {})), + DemoWrapper::BVHNode(Box::new(demos::Instances {})), + DemoWrapper::BVHNode(Box::new(demos::CornellSmokeAndFog {})), + DemoWrapper::HitableList(Box::new(demos::CornellBox {})), + ]; + + for demo in demos.iter() { + run_and_save_demo(demo, width, height) } - println!("answer = {}", sum / N as f64); + Ok(()) +} + +#[cfg(not(feature = "gui"))] +fn run_and_save_demo(demo: &DemoWrapper, width: usize, height: usize) { + let mut buffer = vec![0; width * height * 4]; + + println!( + "Starting {} at {}x{} with {} samples", + demo.name(), + width, + height, + NUM_SAMPLES + ); + + let now = Instant::now(); + demo.render(&mut buffer, width, height, NUM_SAMPLES); + println!( + "Rendered Demo {}. Time Taken(s) = {}", + demo.name(), + now.elapsed().as_secs_f64() + ); + + demo.save_as_ppm(&buffer, width, height, NUM_SAMPLES); } diff --git a/src/materials/dielectric.rs b/src/materials/dielectric.rs new file mode 100644 index 0000000..f99bd5e --- /dev/null +++ b/src/materials/dielectric.rs @@ -0,0 +1,62 @@ +use rand::{prelude::SmallRng, Rng}; + +use crate::{ + hitable::HitRecord, + materials::{reflect, refract, schlick}, + types::{Ray, Vec3}, + Material, +}; + +#[derive(Clone)] +pub struct Dielectric { + refraction_index: f64, +} + +impl Dielectric { + pub fn new(refraction_index: f64) -> Self { + Self { refraction_index } + } +} + +impl Material for Dielectric { + fn scatter( + &self, + ray_in: &Ray, + hit_rec: &HitRecord, + rng: &mut SmallRng, + ) -> (Vec3, Option) { + // Glass absorbs nothing! So, Attenuation is always going to be 1.0 for this + let attenuation = Vec3::splat(1.0); + + let refraction_ratio = if hit_rec.front_face { + 1.0 / self.refraction_index + } else { + self.refraction_index + }; + + let unit_direction = ray_in.direction.unit_vector(); + let cosine = (-unit_direction).dot(&hit_rec.normal).min(1.0); + let sin_theta = (1.0 - cosine * cosine).sqrt(); + + let cannot_refract = refraction_ratio * sin_theta > 1.0; + + if cannot_refract || schlick(cosine, refraction_ratio) > rng.gen::() { + let direction = reflect(unit_direction, hit_rec.normal); + ( + attenuation, + Some(Ray::new(hit_rec.p, direction, ray_in.time())), + ) + } else if let Some(direction) = refract(unit_direction, hit_rec.normal, refraction_ratio) { + ( + attenuation, + Some(Ray::new(hit_rec.p, direction, ray_in.time())), + ) + } else { + let direction = reflect(unit_direction, hit_rec.normal); + ( + attenuation, + Some(Ray::new(hit_rec.p, direction, ray_in.time())), + ) + } + } +} diff --git a/src/materials/diffuse_light.rs b/src/materials/diffuse_light.rs new file mode 100644 index 0000000..c84795b --- /dev/null +++ b/src/materials/diffuse_light.rs @@ -0,0 +1,18 @@ +use crate::{types::Vec3, Material, Texture}; + +#[derive(Clone)] +pub struct DiffuseLight { + emit: T, +} + +impl DiffuseLight { + pub fn new(emit: T) -> Self { + Self { emit } + } +} + +impl Material for DiffuseLight { + fn emit(&self, u: f64, v: f64, p: Vec3) -> Vec3 { + self.emit.value(u, v, p) + } +} diff --git a/src/materials/isotropic.rs b/src/materials/isotropic.rs new file mode 100644 index 0000000..8440c75 --- /dev/null +++ b/src/materials/isotropic.rs @@ -0,0 +1,31 @@ +use rand::prelude::SmallRng; + +use crate::{ + hitable::HitRecord, + materials::random_point_in_unit_sphere, + types::{Ray, Vec3}, + Material, Texture, +}; + +pub struct Isotropic { + texture: T, +} + +impl Isotropic { + pub fn new(texture: T) -> Self { + Self { texture } + } +} + +impl Material for Isotropic { + fn scatter(&self, ray: &Ray, hit_rec: &HitRecord, rng: &mut SmallRng) -> (Vec3, Option) { + ( + self.texture.value(hit_rec.u, hit_rec.v, hit_rec.p), + Some(Ray::new( + hit_rec.p, + random_point_in_unit_sphere(rng), + ray.time(), + )), + ) + } +} diff --git a/src/materials/lambertian.rs b/src/materials/lambertian.rs new file mode 100644 index 0000000..ff46239 --- /dev/null +++ b/src/materials/lambertian.rs @@ -0,0 +1,31 @@ +use rand::prelude::SmallRng; + +use crate::{ + hitable::HitRecord, + materials::random_point_in_unit_sphere, + types::{Ray, Vec3}, + Material, Texture, +}; + +#[derive(Clone)] +pub struct Lambertian { + albedo: T, +} + +impl Lambertian { + pub fn new(albedo: T) -> Self { + Self { albedo } + } +} + +impl Material for Lambertian { + fn scatter(&self, ray: &Ray, hit_rec: &HitRecord, rng: &mut SmallRng) -> (Vec3, Option) { + let scatter_direction = hit_rec.normal + random_point_in_unit_sphere(rng); + let scattered_ray = Ray::new(hit_rec.p, scatter_direction, ray.time()); + + ( + self.albedo.value(hit_rec.u, hit_rec.v, hit_rec.p), + Some(scattered_ray), + ) + } +} diff --git a/src/materials/metal.rs b/src/materials/metal.rs new file mode 100644 index 0000000..0023a63 --- /dev/null +++ b/src/materials/metal.rs @@ -0,0 +1,46 @@ +use rand::prelude::SmallRng; + +use crate::{ + hitable::HitRecord, + materials::{random_point_in_unit_sphere, reflect}, + types::{Ray, Vec3}, + Material, +}; + +#[derive(Clone)] +pub struct Metal { + albedo: Vec3, + fuzz: f64, +} + +impl Metal { + #[allow(dead_code)] + pub fn new(albedo: Vec3) -> Self { + Self { albedo, fuzz: 0.0 } + } + pub fn with_fuzz(albedo: Vec3, fuzz: f64) -> Self { + Self { albedo, fuzz } + } +} + +impl Material for Metal { + fn scatter( + &self, + ray_in: &Ray, + hit_rec: &HitRecord, + rng: &mut SmallRng, + ) -> (Vec3, Option) { + let reflected_ray = reflect(ray_in.direction.unit_vector(), hit_rec.normal); + let scattered_ray = Ray::new( + hit_rec.p, + reflected_ray + random_point_in_unit_sphere(rng) * self.fuzz, + ray_in.time(), + ); + + if scattered_ray.direction.dot(&hit_rec.normal) > 0.0 { + (self.albedo, Some(scattered_ray)) + } else { + (self.albedo, None) + } + } +} diff --git a/src/materials/mod.rs b/src/materials/mod.rs new file mode 100644 index 0000000..9904cd0 --- /dev/null +++ b/src/materials/mod.rs @@ -0,0 +1,72 @@ +mod dielectric; +mod diffuse_light; +mod isotropic; +mod lambertian; +mod metal; + +pub use dielectric::Dielectric; +pub use diffuse_light::DiffuseLight; +pub use isotropic::Isotropic; +pub use lambertian::Lambertian; +pub use metal::Metal; +use rand::{prelude::SmallRng, Rng}; + +use crate::{ + hitable::HitRecord, + types::{Ray, Vec3}, +}; + +pub trait Material: Send + Sync { + // scatter returns the attenuation and the scattered ray. + // Attenuation is ignored completely if there is no scattered ray + fn scatter( + &self, + _ray: &Ray, + _hit_rec: &HitRecord, + _rng: &mut SmallRng, + ) -> (Vec3, Option) { + (Vec3::splat(0.0), None) + } + + fn emit(&self, _u: f64, _v: f64, _p: Vec3) -> Vec3 { + Vec3::splat(0.0) + } +} + +// Christophe Schlick's Polynomial approximation to figure out reflectivity as the angle changes +// See Fresnel Equations, https://en.wikipedia.org/wiki/Fresnel_equations +fn schlick(cosine: f64, reflection_index: f64) -> f64 { + let mut r0 = (1.0 - reflection_index) / (1.0 + reflection_index); + r0 = r0 * r0; + r0 + (1.0 - r0) * (1.0 - cosine).powf(5.0) +} + +fn reflect(incident: Vec3, normal: Vec3) -> Vec3 { + incident - normal * incident.dot(&normal) * 2.0 +} + +// Snell's Law +fn refract(incident: Vec3, normal: Vec3, ni_over_nt: f64) -> Option { + let uv = incident.unit_vector(); + let dt = uv.dot(&normal); + let discriminant = 1.0 - ni_over_nt * ni_over_nt * (1.0 - dt * dt); + if discriminant > 0.0 { + Some((uv - normal * dt) * ni_over_nt - normal * discriminant.sqrt()) + } else { + None + } +} + +fn random_point_in_unit_sphere(rng: &mut R) -> Vec3 { + let mut point = Vec3::random(rng) * 2.0 - Vec3::splat(1.0); + while point.sq_len() >= 1.0 { + point = Vec3::random(rng) * 2.0 - Vec3::splat(1.0); + } + point +} + +pub trait MaterialBuilder { + type Finished; + + fn material(self, material: T) -> Self::Finished; +} diff --git a/src/texture/checker.rs b/src/texture/checker.rs new file mode 100644 index 0000000..57e5ab8 --- /dev/null +++ b/src/texture/checker.rs @@ -0,0 +1,25 @@ +use crate::{types::Vec3, Texture}; + +#[derive(Clone)] +pub struct Checker { + odd: T, + even: T, +} + +impl Checker { + pub fn new(even: T, odd: T) -> Self { + Self { odd, even } + } +} + +impl Texture for Checker { + fn value(&self, u: f64, v: f64, p: Vec3) -> Vec3 { + let sine_wave = f64::sin(10.0 * p.x()) * f64::sin(10.0 * p.y()) * f64::sin(10.0 * p.z()); + + if sine_wave < 0.0 { + self.odd.value(u, v, p) + } else { + self.even.value(u, v, p) + } + } +} diff --git a/src/texture/image_texture.rs b/src/texture/image_texture.rs new file mode 100644 index 0000000..15c2ec4 --- /dev/null +++ b/src/texture/image_texture.rs @@ -0,0 +1,56 @@ +use image::{error::ImageError, io::Reader as ImageReader}; + +use crate::{types::Vec3, Texture}; + +#[derive(Clone)] +pub struct ImageTexture { + image: Vec, + // (width, height) + dimensions: (u32, u32), + bytes_per_scanline: u32, + bytes_per_pixel: u32, +} + +impl ImageTexture { + #[allow(dead_code)] + pub fn from_filename(filename: &str) -> Result { + let img = ImageReader::open(filename)?.decode()?; + let img = img.to_rgb8(); + + let (width, _) = img.dimensions(); + + let bytes_per_pixel = 3; + + Ok(Self { + image: img.to_vec(), + dimensions: img.dimensions(), + bytes_per_scanline: bytes_per_pixel * width, + bytes_per_pixel, + }) + } +} + +impl Texture for ImageTexture { + fn value(&self, u: f64, v: f64, _p: Vec3) -> Vec3 { + let (width, height) = self.dimensions; + + let u = u.clamp(0.0, 1.0); + let v = 1.0 - v.clamp(0.0, 1.0); + + let i = (u * width as f64) as u32; + let j = (v * height as f64) as u32; + + let i = i.clamp(0, width - 1); + let j = j.clamp(0, height - 1); + + let color_scale = 1.0 / 255.0; + + let pixel = (j * self.bytes_per_scanline + i * self.bytes_per_pixel) as usize; + + Vec3::new( + color_scale * (self.image[pixel] as f64), + color_scale * (self.image[pixel + 1] as f64), + color_scale * (self.image[pixel + 2] as f64), + ) + } +} diff --git a/src/texture/mod.rs b/src/texture/mod.rs new file mode 100644 index 0000000..7ec8242 --- /dev/null +++ b/src/texture/mod.rs @@ -0,0 +1,17 @@ +mod checker; +mod image_texture; +mod perlin; +mod perlin_noise; +mod solid; + +pub use checker::Checker; +pub use image_texture::ImageTexture; +pub use perlin::Perlin; +pub use perlin_noise::PerlinNoise; +pub use solid::Solid; + +use crate::types::Vec3; + +pub trait Texture { + fn value(&self, u: f64, v: f64, p: Vec3) -> Vec3; +} diff --git a/src/texture/perlin.rs b/src/texture/perlin.rs new file mode 100644 index 0000000..16c1814 --- /dev/null +++ b/src/texture/perlin.rs @@ -0,0 +1,120 @@ +use crate::types::Vec3; +use rand::Rng; + +const POINT_COUNT: usize = 256; + +#[derive(Clone)] +pub struct Perlin { + points: Vec, + + permute_x: Vec, + permute_y: Vec, + permute_z: Vec, +} + +impl Perlin { + pub fn new(rng: &mut R) -> Self { + let points = (0..POINT_COUNT) + .map(|_| Vec3::random(rng).unit_vector()) + .collect::>(); + + let permute_x = Self::perlin_generate_permutation(rng); + let permute_y = Self::perlin_generate_permutation(rng); + let permute_z = Self::perlin_generate_permutation(rng); + + Self { + points, + permute_x, + permute_y, + permute_z, + } + } + + fn perlin_generate_permutation(rng: &mut R) -> Vec { + let mut p = (0..POINT_COUNT).collect::>(); + permute(rng, &mut p); + p + } + + pub fn noise(&self, p: Vec3) -> f64 { + let mut smooth_grid = [[[Vec3::new(0.0, 0.0, 0.0); 2]; 2]; 2]; + + { + let i = p.x().floor() as i32; + let j = p.y().floor() as i32; + let k = p.z().floor() as i32; + for (di, a) in smooth_grid.iter_mut().enumerate() { + let di = di as i32; + for (dj, b) in a.iter_mut().enumerate() { + let dj = dj as i32; + for (dk, c) in b.iter_mut().enumerate() { + let dk = dk as i32; + + *c = self.points[self.permute_x[((i + di) & 255) as usize] + ^ self.permute_y[((j + dj) & 255) as usize] + ^ self.permute_z[((k + dk) & 255) as usize]] + } + } + } + } + + let u = p.x() - p.x().floor(); + let v = p.y() - p.y().floor(); + let w = p.z() - p.z().floor(); + + perlin_interpolate(smooth_grid, u, v, w) + } + + pub fn turbulence(&self, p: Vec3, depth: u32) -> f64 { + let mut acc = 0.0f64; + let mut weight = 1.0; + let mut temp_p = p; + + for _i in 0..depth { + acc += weight * self.noise(temp_p); + weight *= 0.5; + temp_p *= 2.0; + } + + acc.abs() + } +} + +fn perlin_interpolate(smooth_grid: [[[Vec3; 2]; 2]; 2], u: f64, v: f64, w: f64) -> f64 { + // Hermitian smoothing so we don't see obvious grid features in the picture + // Those features show up when we interpolate colors. Those features are + // also called mach bands + let uu = u * u * (3.0 - 2.0 * u); + let vv = v * v * (3.0 - 2.0 * v); + let ww = w * w * (3.0 - 2.0 * w); + + let mut acc = 0.0; + + for (di, a) in smooth_grid.iter().enumerate() { + let di = di as f64; + for (dj, b) in a.iter().enumerate() { + let dj = dj as f64; + for (dk, c) in b.iter().enumerate() { + let dk = dk as f64; + + let wt = Vec3::new(u - di, v - dj, w - dk); + + acc += (di * uu + (1.0 - di) * (1.0 - uu)) + * (dj * vv + (1.0 - dj) * (1.0 - vv)) + * (dk * ww + (1.0 - dk) * (1.0 - ww)) + * c.dot(&wt); + } + } + } + + acc +} + +fn permute(rng: &mut R, p: &mut [usize]) { + let l = p.len(); + + for i in (0..l).rev() { + let r = rng.gen_range(0..=i); + p.swap(i, r); + } +} diff --git a/src/texture/perlin_noise.rs b/src/texture/perlin_noise.rs new file mode 100644 index 0000000..06615ca --- /dev/null +++ b/src/texture/perlin_noise.rs @@ -0,0 +1,34 @@ +use rand::Rng; + +use crate::{texture::Perlin, types::Vec3, Texture}; + +#[derive(Clone)] +pub struct PerlinNoise { + noise: Perlin, + scale: f64, +} + +impl PerlinNoise { + #[allow(dead_code)] + pub fn new(rng: &mut R) -> Self { + Self { + noise: Perlin::new(rng), + scale: 1.0, + } + } + + pub fn with_scale(rng: &mut R, scale: f64) -> Self { + Self { + noise: Perlin::new(rng), + scale, + } + } +} + +impl Texture for PerlinNoise { + fn value(&self, _u: f64, _v: f64, p: Vec3) -> Vec3 { + Vec3::new(1.0, 1.0, 1.0) + * 0.5 + * (1.0 + (self.scale * p.z() + 10.0 * self.noise.turbulence(p, 7)).sin()) + } +} diff --git a/src/texture/solid.rs b/src/texture/solid.rs new file mode 100644 index 0000000..65f55f8 --- /dev/null +++ b/src/texture/solid.rs @@ -0,0 +1,18 @@ +use crate::{types::Vec3, Texture}; + +#[derive(Clone)] +pub struct Solid { + color: Vec3, +} + +impl Solid { + pub const fn new(color: Vec3) -> Self { + Self { color } + } +} + +impl Texture for Solid { + fn value(&self, _u: f64, _v: f64, _p: Vec3) -> Vec3 { + self.color + } +} diff --git a/src/types/color.rs b/src/types/color.rs new file mode 100644 index 0000000..938d08d --- /dev/null +++ b/src/types/color.rs @@ -0,0 +1,12 @@ +use crate::types::Vec3; + +pub struct Color(pub u8, pub u8, pub u8); + +impl From for Color { + fn from(v: Vec3) -> Self { + let v = v.sqrt() * 255.99; + let (r, g, b) = (v.x(), v.y(), v.z()); + + Self(r as u8, g as u8, b as u8) + } +} diff --git a/src/types/dimension.rs b/src/types/dimension.rs new file mode 100644 index 0000000..89b8937 --- /dev/null +++ b/src/types/dimension.rs @@ -0,0 +1,21 @@ +// Taken from, https://github.com/Globidev/toy-rt/tree/master/trt-core/src/dimension.rs + +pub trait Dimension { + const INDEX: usize; +} + +pub struct X; +pub struct Y; +pub struct Z; + +impl Dimension for X { + const INDEX: usize = 0; +} + +impl Dimension for Y { + const INDEX: usize = 1; +} + +impl Dimension for Z { + const INDEX: usize = 2; +} diff --git a/src/types/mod.rs b/src/types/mod.rs new file mode 100644 index 0000000..9f3c95b --- /dev/null +++ b/src/types/mod.rs @@ -0,0 +1,17 @@ +mod color; +mod dimension; +mod ray; + +pub use color::Color; +pub use dimension::{Dimension, X, Y, Z}; +pub use ray::Ray; + +#[cfg(not(target_arch = "x86_64"))] +mod vec3; +#[cfg(not(target_arch = "x86_64"))] +pub use vec3::Vec3; + +#[cfg(target_arch = "x86_64")] +mod simd_vec3; +#[cfg(target_arch = "x86_64")] +pub use simd_vec3::Vec3; diff --git a/src/types/ray.rs b/src/types/ray.rs new file mode 100644 index 0000000..d38d153 --- /dev/null +++ b/src/types/ray.rs @@ -0,0 +1,54 @@ +use rand::prelude::SmallRng; + +use crate::{hitable::Hitable, types::Vec3}; + +pub struct Ray { + pub origin: Vec3, + pub direction: Vec3, + time: f64, +} + +impl Ray { + pub fn new(origin: Vec3, direction: Vec3, time: f64) -> Ray { + Ray { + origin, + direction, + time, + } + } + + #[inline] + pub fn point_at_parameter(&self, t: f64) -> Vec3 { + self.origin + self.direction * t + } + #[inline] + pub const fn time(&self) -> f64 { + self.time + } + + pub fn color( + &self, + world: &T, + rng: &mut SmallRng, + background: &Vec3, + depth: u32, + ) -> Vec3 { + if let Some(hit_rec) = world.hit(self, 0.001, std::f64::MAX) { + if depth >= 50 { + Vec3::splat(0.0f64) + } else { + let material = hit_rec.material; + let emitted_color = hit_rec.material.emit(hit_rec.u, hit_rec.v, hit_rec.p); + + if let (attenuation, Some(scattered_ray)) = material.scatter(self, &hit_rec, rng) { + emitted_color + + attenuation * scattered_ray.color(world, rng, background, depth + 1) + } else { + emitted_color + } + } + } else { + *background + } + } +} diff --git a/src/types/simd_vec3.rs b/src/types/simd_vec3.rs new file mode 100644 index 0000000..51ba16a --- /dev/null +++ b/src/types/simd_vec3.rs @@ -0,0 +1,214 @@ +use std::{ + fmt::{Display, Formatter, Result as FmtResult}, + ops::{Add, AddAssign, Div, DivAssign, Mul, MulAssign, Neg, RangeInclusive, Sub, SubAssign}, +}; + +use rand::Rng; + +use crate::{Asf64, Dimension, X, Y, Z}; + +use packed_simd::{f64x4, shuffle}; + +#[derive(Default, Debug, Copy, Clone)] +pub struct Vec3(f64x4); + +impl Vec3 { + #[inline] + pub fn new(a: impl Asf64, b: impl Asf64, c: impl Asf64) -> Vec3 { + Self(f64x4::new(a.as_(), b.as_(), c.as_(), 0.0)) + } + + pub fn splat(xyz: impl Asf64) -> Self { + Self::new(xyz, xyz, xyz) + } + + pub fn random(rng: &mut R) -> Self { + Self(f64x4::from_slice_unaligned(&rng.gen::<[f64; 4]>())) + } + + pub fn random_in_range(rng: &mut R, range: RangeInclusive) -> Self { + Vec3::new( + rng.gen_range(range.clone()), + rng.gen_range(range.clone()), + rng.gen_range(range), + ) + } + + #[inline] + pub fn x(&self) -> f64 { + self.get::() + } + #[inline] + pub fn y(&self) -> f64 { + self.get::() + } + #[inline] + pub fn z(&self) -> f64 { + self.get::() + } + + pub fn get(&self) -> f64 { + unsafe { self.0.extract_unchecked(D::INDEX) } + } + + pub fn set(self, value: f64) -> Self { + Self(unsafe { self.0.replace_unchecked(D::INDEX, value) }) + } + + #[inline] + pub fn length(&self) -> f64 { + self.sq_len().sqrt() + } + + #[inline] + pub fn sq_len(&self) -> f64 { + (self.0 * self.0).sum() + } + + #[inline] + pub fn dot(&self, v: &Vec3) -> f64 { + (self.0 * v.0).sum() + } + + #[inline] + pub fn cross(&self, v: &Vec3) -> Vec3 { + // https://web.archive.org/web/20210412192227/https://geometrian.com/programming/tutorials/cross-product/index.php + let tmp0: f64x4 = shuffle!(self.0, [1, 2, 0, 3]); + let tmp1: f64x4 = shuffle!(v.0, [2, 0, 1, 3]); + let tmp2: f64x4 = shuffle!(self.0, [2, 0, 1, 3]); + let tmp3: f64x4 = shuffle!(v.0, [1, 2, 0, 3]); + + Vec3(tmp0 * tmp1 - tmp2 * tmp3) + } + + #[inline] + pub fn unit_vector(self) -> Vec3 { + self / self.length() + } + + pub fn min(self, other: Self) -> Vec3 { + Self(self.0.min(other.0)) + } + + pub fn max(self, other: Self) -> Vec3 { + Self(self.0.max(other.0)) + } + + pub fn min_element(self, other: f64) -> f64 { + unsafe { self.0.replace_unchecked(3, other).min_element() } + } + + pub fn max_element(self, other: f64) -> f64 { + unsafe { self.0.replace_unchecked(3, other).max_element() } + } + + #[inline] + pub fn sqrt(self) -> Self { + Self(self.0.sqrt()) + } +} + +impl Add for Vec3 { + type Output = Vec3; + + fn add(self, o: Vec3) -> Vec3 { + Vec3(self.0 + o.0) + } +} + +impl AddAssign for Vec3 { + fn add_assign(&mut self, o: Vec3) { + self.0 += o.0 + } +} + +impl Sub for Vec3 { + type Output = Vec3; + + fn sub(self, o: Vec3) -> Vec3 { + Vec3(self.0 - o.0) + } +} + +impl SubAssign for Vec3 { + fn sub_assign(&mut self, o: Vec3) { + self.0 -= o.0; + } +} + +impl Neg for Vec3 { + type Output = Vec3; + + fn neg(self) -> Vec3 { + Vec3(-self.0) + } +} + +impl MulAssign for Vec3 { + fn mul_assign(&mut self, o: Vec3) { + self.0 *= o.0 + } +} + +impl MulAssign for Vec3 { + fn mul_assign(&mut self, o: f64) { + self.0 *= o + } +} + +impl Mul for Vec3 { + type Output = Vec3; + fn mul(self, o: f64) -> Vec3 { + Vec3(self.0 * o) + } +} + +impl Mul for Vec3 { + type Output = Vec3; + fn mul(self, o: Vec3) -> Vec3 { + Vec3(self.0 * o.0) + } +} + +impl Div for Vec3 { + type Output = Vec3; + + fn div(self, o: Vec3) -> Vec3 { + Vec3(self.0 / o.0) + } +} + +impl Div for Vec3 { + type Output = Vec3; + + fn div(self, o: f64) -> Vec3 { + let o = 1.0 / o; + + Vec3(self.0 * o) + } +} + +impl DivAssign for Vec3 { + fn div_assign(&mut self, o: f64) { + let o = 1.0 / o; + + self.0 *= o + } +} + +impl Display for Vec3 { + fn fmt(&self, f: &mut Formatter) -> FmtResult { + f.write_fmt(format_args!( + "{} {} {}", + self.get::(), + self.get::(), + self.get::() + )) + } +} + +impl From<(A, B, C)> for Vec3 { + fn from((x, y, z): (A, B, C)) -> Self { + Self::new(x, y, z) + } +} diff --git a/src/types/vec3.rs b/src/types/vec3.rs new file mode 100644 index 0000000..9ba86fc --- /dev/null +++ b/src/types/vec3.rs @@ -0,0 +1,244 @@ +use std::{ + fmt::{Display, Formatter, Result as FmtResult}, + ops::{ + Add, AddAssign, Div, DivAssign, Index, IndexMut, Mul, MulAssign, Neg, RangeInclusive, Sub, + SubAssign, + }, +}; + +use rand::Rng; + +use crate::{Asf64, Dimension, X, Y, Z}; + +#[derive(Default, Debug, Copy, Clone)] +pub struct Vec3([f64; 3]); + +impl Vec3 { + #[inline] + pub fn new(a: impl Asf64, b: impl Asf64, c: impl Asf64) -> Vec3 { + Self([a.as_(), b.as_(), c.as_()]) + } + + pub fn splat(xyz: impl Asf64) -> Self { + Self::new(xyz, xyz, xyz) + } + + pub fn random(rng: &mut R) -> Self { + Self(rng.gen()) + } + + pub fn random_in_range(rng: &mut R, range: RangeInclusive) -> Self { + Vec3::new( + rng.gen_range(range.clone()), + rng.gen_range(range.clone()), + rng.gen_range(range), + ) + } + + #[inline] + pub fn x(&self) -> f64 { + self.get::() + } + #[inline] + pub fn y(&self) -> f64 { + self.get::() + } + #[inline] + pub fn z(&self) -> f64 { + self.get::() + } + + pub fn get(&self) -> f64 { + self.0[D::INDEX] + } + + pub fn set(mut self, value: f64) -> Self { + self.0[D::INDEX] = value; + self + } + + #[inline] + pub fn length(&self) -> f64 { + self.sq_len().sqrt() + } + + #[inline] + pub fn sq_len(&self) -> f64 { + self.x() * self.x() + self.y() * self.y() + self.z() * self.z() + } + + #[inline] + pub fn dot(&self, v: &Vec3) -> f64 { + self.x() * v.x() + self.y() * v.y() + self.z() * v.z() + } + + #[inline] + pub fn cross(&self, v: &Vec3) -> Vec3 { + Vec3([ + self.y() * v.z() - self.z() * v.y(), + self.z() * v.x() - self.x() * v.z(), + self.x() * v.y() - self.y() * v.x(), + ]) + } + + #[inline] + pub fn unit_vector(self) -> Vec3 { + self / self.length() + } + + pub fn min(self, other: Self) -> Vec3 { + Self([ + self.x().min(other.x()), + self.y().min(other.y()), + self.z().min(other.z()), + ]) + } + + pub fn max(self, other: Self) -> Vec3 { + Self([ + self.x().max(other.x()), + self.y().max(other.y()), + self.z().max(other.z()), + ]) + } + + pub fn min_element(self, other: f64) -> f64 { + self.x().min(self.y()).min(self.z()).min(other) + } + + pub fn max_element(self, other: f64) -> f64 { + self.x().max(self.y()).max(self.z()).max(other) + } + + #[inline] + pub fn sqrt(self) -> Self { + Vec3::new(self.x().sqrt(), self.y().sqrt(), self.z().sqrt()) + } +} + +impl Add for Vec3 { + type Output = Vec3; + + fn add(self, o: Vec3) -> Vec3 { + Vec3([self.x() + o.x(), self.y() + o.y(), self.z() + o.z()]) + } +} + +impl AddAssign for Vec3 { + fn add_assign(&mut self, o: Vec3) { + self.0[0] += o.0[0]; + self.0[1] += o.0[1]; + self.0[2] += o.0[2]; + } +} + +impl Sub for Vec3 { + type Output = Vec3; + + fn sub(self, o: Vec3) -> Vec3 { + Vec3([self.x() - o.x(), self.y() - o.y(), self.z() - o.z()]) + } +} + +impl SubAssign for Vec3 { + fn sub_assign(&mut self, o: Vec3) { + self.0[0] -= o.0[0]; + self.0[1] -= o.0[1]; + self.0[2] -= o.0[2]; + } +} + +impl Neg for Vec3 { + type Output = Vec3; + + fn neg(self) -> Vec3 { + Vec3([-self.x(), -self.y(), -self.z()]) + } +} + +impl MulAssign for Vec3 { + fn mul_assign(&mut self, o: Vec3) { + self.0[0] *= o.0[0]; + self.0[1] *= o.0[1]; + self.0[2] *= o.0[2]; + } +} + +impl MulAssign for Vec3 { + fn mul_assign(&mut self, o: f64) { + self.0[0] *= o; + self.0[1] *= o; + self.0[2] *= o; + } +} + +impl Mul for Vec3 { + type Output = Vec3; + fn mul(self, o: f64) -> Vec3 { + Vec3([self.x() * o, self.y() * o, self.z() * o]) + } +} + +impl Mul for Vec3 { + type Output = Vec3; + fn mul(self, o: Vec3) -> Vec3 { + Vec3([self.x() * o.x(), self.y() * o.y(), self.z() * o.z()]) + } +} + +impl Div for Vec3 { + type Output = Vec3; + + fn div(self, o: Vec3) -> Vec3 { + Vec3([self.x() / o.x(), self.y() / o.y(), self.z() / o.z()]) + } +} + +impl Div for Vec3 { + type Output = Vec3; + + fn div(self, o: f64) -> Vec3 { + let o = 1.0 / o; + Vec3([self.x() * o, self.y() * o, self.z() * o]) + } +} + +impl DivAssign for Vec3 { + fn div_assign(&mut self, o: f64) { + let o = 1.0 / o; + self.0[0] *= o; + self.0[1] *= o; + self.0[2] *= o; + } +} + +impl Index for Vec3 { + type Output = f64; + + fn index(&self, q: usize) -> &f64 { + &self.0[q] + } +} + +impl IndexMut for Vec3 { + fn index_mut(&mut self, q: usize) -> &mut f64 { + &mut self.0[q] + } +} + +impl Display for Vec3 { + fn fmt(&self, f: &mut Formatter) -> FmtResult { + f.write_fmt(format_args!( + "{} {} {}", + self.get::(), + self.get::(), + self.get::() + )) + } +} + +impl From<(A, B, C)> for Vec3 { + fn from((x, y, z): (A, B, C)) -> Self { + Self::new(x, y, z) + } +}