Refactor Renderer trait

This commit is contained in:
hal8174 2024-10-04 18:12:00 +02:00
parent 7d122d44b3
commit 7d38e87f6a
9 changed files with 228 additions and 172 deletions

View file

@ -1,7 +1,7 @@
use crate::prelude::*; use crate::{camera::Camera, prelude::*, scene::Scene};
pub trait ClassicalRenderer<R: Rng> { pub trait ClassicalRenderer<R: Rng, S: Scene<R>, C: Camera<R>> {
fn render_pixel(&self, x: u32, y: u32, rng: &mut R) -> Color; fn render_pixel(&self, scene: &S, camera: &C, x: u32, y: u32, rng: &mut R) -> Color;
fn width(&self) -> u32; fn width(&self) -> u32;
fn height(&self) -> u32; fn height(&self) -> u32;
} }

View file

@ -1,13 +1,24 @@
use image::{ImageBuffer, ImageResult, Rgb}; use image::{ImageBuffer, ImageResult, Rgb};
use rand::{rngs::SmallRng, SeedableRng}; use rand::{rngs::SmallRng, SeedableRng};
use ray_tracing_core::{camera::BasicCamera, prelude::*, renderer::ClassicalRenderer}; use ray_tracing_core::{
use ray_tracing_renderer::{path_tracer::PathTracer, path_tracer_importance::PathTracerImportance}; camera::{BasicCamera, Camera},
use ray_tracing_scene::{basic_scene::BasicScene, triangle_bvh::examples::cornel}; prelude::*,
renderer::ClassicalRenderer,
scene::Scene,
};
use ray_tracing_renderer::path_tracer_importance::PathTracerImportance;
use ray_tracing_scene::examples::basic_cornel;
use rayon::prelude::*; use rayon::prelude::*;
use std::path::Path; use std::path::Path;
fn render_image<C: ClassicalRenderer<SmallRng> + Sync>( fn render_image<
renderer: C, R: ClassicalRenderer<SmallRng, S, C> + Sync,
S: Scene<SmallRng> + Sync,
C: Camera<SmallRng> + Sync,
>(
renderer: &R,
scene: &S,
camera: &C,
outputfilename: impl AsRef<Path>, outputfilename: impl AsRef<Path>,
samples_per_pixel: usize, samples_per_pixel: usize,
) -> ImageResult<()> { ) -> ImageResult<()> {
@ -19,7 +30,8 @@ fn render_image<C: ClassicalRenderer<SmallRng> + Sync>(
let mut rng = SmallRng::seed_from_u64((x + y * renderer.width()) as u64); let mut rng = SmallRng::seed_from_u64((x + y * renderer.width()) as u64);
for _ in 0..samples_per_pixel { for _ in 0..samples_per_pixel {
*c += renderer.render_pixel(x, y, &mut rng) / (samples_per_pixel as Float); *c +=
renderer.render_pixel(scene, camera, x, y, &mut rng) / (samples_per_pixel as Float);
} }
}); });
@ -33,18 +45,18 @@ fn render_image<C: ClassicalRenderer<SmallRng> + Sync>(
fn main() -> ImageResult<()> { fn main() -> ImageResult<()> {
// let s = BasicScene::new(); // let s = BasicScene::new();
let s = cornel(); let s = basic_cornel();
let c = BasicCamera::new( let c = BasicCamera::new(
400, 400,
400, 400,
Pos3::new(-6.0, 0.0, 0.0), s.camera_pos,
Dir3::new(1.0, 0.0, 0.0), s.camera_dir,
Dir3::up(), s.camera_up,
Float::to_radians(90.0), s.horizontal_fov,
); );
let r = PathTracerImportance::new(s, c); let r = PathTracerImportance::new(400, 400);
render_image(r, "test.exr", 1048) render_image(&r, &s.scene, &c, "test.exr", 1048)
} }

View file

@ -1,42 +1,38 @@
use ray_tracing_core::{camera::Camera, prelude::*, renderer::ClassicalRenderer, scene::Scene}; use ray_tracing_core::{camera::Camera, prelude::*, renderer::ClassicalRenderer, scene::Scene};
use std::marker::PhantomData; use std::marker::PhantomData;
pub struct DepthRenderer<S, C, R> pub struct DepthRenderer<R>
where where
S: Scene<R>,
C: Camera<R>,
R: Rng, R: Rng,
{ {
scene: S, width: u32,
camera: C, height: u32,
rng: PhantomData<R>, rng: PhantomData<R>,
} }
impl<S, C, R> DepthRenderer<S, C, R> impl<R> DepthRenderer<R>
where where
S: Scene<R>,
C: Camera<R>,
R: Rng, R: Rng,
{ {
pub fn new(scene: S, camera: C) -> Self { pub fn new(width: u32, height: u32) -> Self {
Self { Self {
scene,
camera,
rng: PhantomData {}, rng: PhantomData {},
width,
height,
} }
} }
} }
impl<S, C, R> ClassicalRenderer<R> for DepthRenderer<S, C, R> impl<S, C, R> ClassicalRenderer<R, S, C> for DepthRenderer<R>
where where
S: Scene<R>, S: Scene<R>,
C: Camera<R>, C: Camera<R>,
R: Rng, R: Rng,
{ {
fn render_pixel(&self, x: u32, y: u32, rng: &mut R) -> Color { fn render_pixel(&self, scene: &S, camera: &C, x: u32, y: u32, rng: &mut R) -> Color {
let r = self.camera.forward(x, y, rng); let r = camera.forward(x, y, rng);
if let Some(i) = self.scene.intersect(r, 0.0, Float::INFINITY) { if let Some(i) = scene.intersect(r, 0.0, Float::INFINITY) {
Color::gray(1.0 / i.t()) Color::gray(1.0 / i.t())
// let c = 0.5 * (i.normal() + Dir3::new(1.0, 1.0, 1.0)); // let c = 0.5 * (i.normal() + Dir3::new(1.0, 1.0, 1.0));
// let c = i.normal(); // let c = i.normal();
@ -47,10 +43,10 @@ where
} }
fn width(&self) -> u32 { fn width(&self) -> u32 {
self.camera.width() self.width
} }
fn height(&self) -> u32 { fn height(&self) -> u32 {
self.camera.height() self.height
} }
} }

View file

@ -1,47 +1,43 @@
use ray_tracing_core::{camera::Camera, prelude::*, renderer::ClassicalRenderer, scene::Scene}; use ray_tracing_core::{camera::Camera, prelude::*, renderer::ClassicalRenderer, scene::Scene};
use std::marker::PhantomData; use std::marker::PhantomData;
pub struct PathTracer<S, C, R> pub struct PathTracer<R>
where where
S: Scene<R>,
C: Camera<R>,
R: Rng, R: Rng,
{ {
scene: S, width: u32,
camera: C, height: u32,
rng: PhantomData<R>, rng: PhantomData<R>,
} }
impl<S, C, R> PathTracer<S, C, R> impl<R> PathTracer<R>
where where
S: Scene<R>,
C: Camera<R>,
R: Rng, R: Rng,
{ {
pub fn new(scene: S, camera: C) -> Self { pub fn new(width: u32, height: u32) -> Self {
Self { Self {
scene, width,
camera, height,
rng: PhantomData {}, rng: PhantomData {},
} }
} }
} }
impl<S, C, R> ClassicalRenderer<R> for PathTracer<S, C, R> impl<S, C, R> ClassicalRenderer<R, S, C> for PathTracer<R>
where where
S: Scene<R>, S: Scene<R>,
C: Camera<R>, C: Camera<R>,
R: Rng, R: Rng,
{ {
fn render_pixel(&self, x: u32, y: u32, rng: &mut R) -> Color { fn render_pixel(&self, scene: &S, camera: &C, x: u32, y: u32, rng: &mut R) -> Color {
let mut sum = Color::black(); let mut sum = Color::black();
let mut alpha = Color::white(); let mut alpha = Color::white();
let mut r = self.camera.forward(x, y, rng); let mut r = camera.forward(x, y, rng);
let mut count = 0; let mut count = 0;
while let Some(i) = self.scene.intersect(r, 0.001, Float::INFINITY) { while let Some(i) = scene.intersect(r, 0.001, Float::INFINITY) {
let frame = i.tangent_frame(); let frame = i.tangent_frame();
let w_in = frame.to_frame(-r.dir()); let w_in = frame.to_frame(-r.dir());
let w_out = Dir3::sample_uniform_hemisphere(rng); let w_out = Dir3::sample_uniform_hemisphere(rng);
@ -67,10 +63,10 @@ where
} }
fn width(&self) -> u32 { fn width(&self) -> u32 {
self.camera.width() self.width
} }
fn height(&self) -> u32 { fn height(&self) -> u32 {
self.camera.height() self.height
} }
} }

View file

@ -1,47 +1,43 @@
use ray_tracing_core::{camera::Camera, prelude::*, renderer::ClassicalRenderer, scene::Scene}; use ray_tracing_core::{camera::Camera, prelude::*, renderer::ClassicalRenderer, scene::Scene};
use std::marker::PhantomData; use std::marker::PhantomData;
pub struct PathTracerImportance<S, C, R> pub struct PathTracerImportance<R>
where where
S: Scene<R>,
C: Camera<R>,
R: Rng, R: Rng,
{ {
scene: S, width: u32,
camera: C, height: u32,
rng: PhantomData<R>, rng: PhantomData<R>,
} }
impl<S, C, R> PathTracerImportance<S, C, R> impl<R> PathTracerImportance<R>
where where
S: Scene<R>,
C: Camera<R>,
R: Rng, R: Rng,
{ {
pub fn new(scene: S, camera: C) -> Self { pub fn new(width: u32, height: u32) -> Self {
Self { Self {
scene, width,
camera, height,
rng: PhantomData {}, rng: PhantomData {},
} }
} }
} }
impl<S, C, R> ClassicalRenderer<R> for PathTracerImportance<S, C, R> impl<S, C, R> ClassicalRenderer<R, S, C> for PathTracerImportance<R>
where where
S: Scene<R>, S: Scene<R>,
C: Camera<R>, C: Camera<R>,
R: Rng, R: Rng,
{ {
fn render_pixel(&self, x: u32, y: u32, rng: &mut R) -> Color { fn render_pixel(&self, scene: &S, camera: &C, x: u32, y: u32, rng: &mut R) -> Color {
let mut sum = Color::black(); let mut sum = Color::black();
let mut alpha = Color::white(); let mut alpha = Color::white();
let mut r = self.camera.forward(x, y, rng); let mut r = camera.forward(x, y, rng);
let mut count = 0; let mut count = 0;
while let Some(i) = self.scene.intersect(r, 0.001, Float::INFINITY) { while let Some(i) = scene.intersect(r, 0.001, Float::INFINITY) {
let frame = i.tangent_frame(); let frame = i.tangent_frame();
let w_in = frame.to_frame(-r.dir()); let w_in = frame.to_frame(-r.dir());
@ -68,10 +64,10 @@ where
} }
fn width(&self) -> u32 { fn width(&self) -> u32 {
self.camera.width() self.width
} }
fn height(&self) -> u32 { fn height(&self) -> u32 {
self.camera.height() self.height
} }
} }

View file

@ -0,0 +1,100 @@
use ray_tracing_core::{light::AreaLight, prelude::*};
use ray_tracing_material::{diffuse::DiffuseMaterial, mirror::Mirror};
use std::collections::HashMap;
use crate::triangle_bvh::{BVHMaterial, Triangle, TriangleBVH};
pub struct ExampleScene<R: Rng> {
pub scene: TriangleBVH<R>,
pub camera_pos: Pos3,
pub camera_dir: Dir3,
pub camera_up: Dir3,
pub horizontal_fov: Float,
}
pub fn example_scenes<R: Rng>() -> HashMap<&'static str, fn() -> ExampleScene<R>> {
let mut map: HashMap<&str, fn() -> ExampleScene<R>> = HashMap::new();
map.insert("basic_cornel", basic_cornel::<R>);
// map.insert("cornel2", cornel2::<R>);
map
}
pub fn cornel2<R: Rng>() -> ExampleScene<R> {
todo!()
}
pub fn basic_cornel<R: Rng>() -> ExampleScene<R> {
let side_length = 1.5;
let light_size = 0.5;
let light_offset = 0.01;
let box_size = 0.3;
let scene = TriangleBVH::new(
vec![
Pos3::new(side_length, side_length, side_length),
Pos3::new(side_length, side_length, -side_length),
Pos3::new(side_length, -side_length, side_length),
Pos3::new(side_length, -side_length, -side_length),
Pos3::new(-side_length, side_length, side_length),
Pos3::new(-side_length, side_length, -side_length),
Pos3::new(-side_length, -side_length, side_length),
Pos3::new(-side_length, -side_length, -side_length),
Pos3::new(light_size, side_length - light_offset, light_size),
Pos3::new(light_size, side_length - light_offset, -light_size),
Pos3::new(-light_size, side_length - light_offset, light_size),
Pos3::new(-light_size, side_length - light_offset, -light_size),
Pos3::new(box_size, -side_length, 0.0),
Pos3::new(0.0, -side_length, -box_size),
Pos3::new(-box_size, -side_length, 0.0),
Pos3::new(0.0, -side_length, box_size),
Pos3::new(0.0, box_size - side_length, 0.0),
],
vec![
Triangle::new([0, 1, 2], 0),
Triangle::new([1, 3, 2], 0),
Triangle::new([0, 4, 1], 0),
Triangle::new([1, 4, 5], 0),
Triangle::new([2, 3, 6], 0),
Triangle::new([3, 7, 6], 0),
Triangle::new([0, 2, 4], 1),
Triangle::new([6, 4, 2], 1),
Triangle::new([1, 5, 3], 2),
Triangle::new([7, 3, 5], 2),
Triangle::new([8, 10, 9], 3),
Triangle::new([11, 9, 10], 3),
Triangle::new([12, 13, 16], 4),
Triangle::new([13, 14, 16], 4),
Triangle::new([14, 15, 16], 4),
Triangle::new([15, 12, 16], 4),
],
vec![
BVHMaterial {
material: Some(Box::new(DiffuseMaterial::new(Color::new(0.8, 0.8, 0.8)))),
light: None,
},
BVHMaterial {
material: Some(Box::new(DiffuseMaterial::new(Color::new(0.9, 0.0, 0.0)))),
light: None,
},
BVHMaterial {
material: Some(Box::new(DiffuseMaterial::new(Color::new(0.0, 0.9, 0.0)))),
light: None,
},
BVHMaterial {
material: None,
light: Some(Box::new(AreaLight::new(Color::white() * 30.0))),
},
BVHMaterial {
material: Some(Box::new(Mirror::new(Color::new(1.0, 1.0, 1.0)))),
light: None,
},
],
);
ExampleScene {
scene,
camera_pos: Pos3::new(-6.0, 0.0, 0.0),
camera_dir: Dir3::new(1.0, 0.0, 0.0),
camera_up: Dir3::up(),
horizontal_fov: Float::to_radians(90.0),
}
}

View file

@ -1,3 +1,4 @@
pub mod basic_scene; pub mod basic_scene;
pub mod examples;
pub mod parse_obj; pub mod parse_obj;
pub mod triangle_bvh; pub mod triangle_bvh;

View file

@ -14,9 +14,9 @@ pub struct TriangleBVH<R: Rng> {
} }
#[derive(Debug)] #[derive(Debug)]
struct BVHMaterial<R: Rng> { pub struct BVHMaterial<R: Rng> {
material: Option<Box<dyn Material<R>>>, pub material: Option<Box<dyn Material<R>>>,
light: Option<Box<dyn Light<R>>>, pub light: Option<Box<dyn Light<R>>>,
} }
#[derive(Debug, Clone, Copy)] #[derive(Debug, Clone, Copy)]
@ -40,7 +40,7 @@ pub struct Triangle {
} }
impl Triangle { impl Triangle {
fn new(vertices: [Index; 3], material: Index) -> Self { pub fn new(vertices: [Index; 3], material: Index) -> Self {
Triangle { vertices, material } Triangle { vertices, material }
} }
} }
@ -167,7 +167,7 @@ fn build_bvh(
} }
impl<R: Rng> TriangleBVH<R> { impl<R: Rng> TriangleBVH<R> {
fn new( pub fn new(
vertices: Vec<Pos3>, vertices: Vec<Pos3>,
mut triangles: Vec<Triangle>, mut triangles: Vec<Triangle>,
materials: Vec<BVHMaterial<R>>, materials: Vec<BVHMaterial<R>>,
@ -281,78 +281,3 @@ impl<R: Rng> Scene<R> for TriangleBVH<R> {
)) ))
} }
} }
pub mod examples {
use super::{BVHMaterial, Triangle, TriangleBVH};
use ray_tracing_core::{light::AreaLight, prelude::*};
use ray_tracing_material::{diffuse::DiffuseMaterial, mirror::Mirror};
use std::fmt::Debug;
pub fn cornel<R: Rng + Debug>() -> TriangleBVH<R> {
let side_length = 1.5;
let light_size = 0.5;
let light_offset = 0.01;
let box_size = 0.3;
dbg!(TriangleBVH::new(
vec![
Pos3::new(side_length, side_length, side_length),
Pos3::new(side_length, side_length, -side_length),
Pos3::new(side_length, -side_length, side_length),
Pos3::new(side_length, -side_length, -side_length),
Pos3::new(-side_length, side_length, side_length),
Pos3::new(-side_length, side_length, -side_length),
Pos3::new(-side_length, -side_length, side_length),
Pos3::new(-side_length, -side_length, -side_length),
Pos3::new(light_size, side_length - light_offset, light_size),
Pos3::new(light_size, side_length - light_offset, -light_size),
Pos3::new(-light_size, side_length - light_offset, light_size),
Pos3::new(-light_size, side_length - light_offset, -light_size),
Pos3::new(box_size, -side_length, 0.0),
Pos3::new(0.0, -side_length, -box_size),
Pos3::new(-box_size, -side_length, 0.0),
Pos3::new(0.0, -side_length, box_size),
Pos3::new(0.0, box_size - side_length, 0.0),
],
vec![
Triangle::new([0, 1, 2], 0),
Triangle::new([1, 3, 2], 0),
Triangle::new([0, 4, 1], 0),
Triangle::new([1, 4, 5], 0),
Triangle::new([2, 3, 6], 0),
Triangle::new([3, 7, 6], 0),
Triangle::new([0, 2, 4], 1),
Triangle::new([6, 4, 2], 1),
Triangle::new([1, 5, 3], 2),
Triangle::new([7, 3, 5], 2),
Triangle::new([8, 10, 9], 3),
Triangle::new([11, 9, 10], 3),
Triangle::new([12, 13, 16], 4),
Triangle::new([13, 14, 16], 4),
Triangle::new([14, 15, 16], 4),
Triangle::new([15, 12, 16], 4),
],
vec![
BVHMaterial {
material: Some(Box::new(DiffuseMaterial::new(Color::new(0.8, 0.8, 0.8)))),
light: None,
},
BVHMaterial {
material: Some(Box::new(DiffuseMaterial::new(Color::new(0.9, 0.0, 0.0)))),
light: None,
},
BVHMaterial {
material: Some(Box::new(DiffuseMaterial::new(Color::new(0.0, 0.9, 0.0)))),
light: None,
},
BVHMaterial {
material: None,
light: Some(Box::new(AreaLight::new(Color::white() * 30.0))),
},
BVHMaterial {
material: Some(Box::new(Mirror::new(Color::new(1.0, 1.0, 1.0)))),
light: None,
},
],
))
}
}

View file

@ -1,17 +1,28 @@
use rand::{rngs::SmallRng, SeedableRng}; use rand::{rngs::SmallRng, SeedableRng};
use ray_tracing_core::{camera::BasicCamera, prelude::*, renderer::ClassicalRenderer}; use ray_tracing_core::{
camera::{BasicCamera, Camera},
prelude::*,
renderer::ClassicalRenderer,
scene::Scene,
};
use ray_tracing_renderer::{ use ray_tracing_renderer::{
depth_renderer::DepthRenderer, path_tracer::PathTracer, depth_renderer::DepthRenderer, path_tracer::PathTracer,
path_tracer_importance::PathTracerImportance, path_tracer_importance::PathTracerImportance,
}; };
use ray_tracing_scene::triangle_bvh::{examples::cornel, TriangleBVH}; use ray_tracing_scene::{examples::basic_cornel, triangle_bvh::TriangleBVH};
use rayon::prelude::*; use rayon::prelude::*;
use std::{net::TcpStream, path::Path}; use std::net::TcpStream;
use tev_client::{PacketCreateImage, PacketUpdateImage, TevClient, TevError}; use tev_client::{PacketCreateImage, PacketUpdateImage, TevClient, TevError};
fn render_image<C: ClassicalRenderer<SmallRng> + Sync>( fn render_image<
renderer: C, R: ClassicalRenderer<SmallRng, S, C> + Sync,
S: Scene<SmallRng> + Sync,
C: Camera<SmallRng> + Sync,
>(
name: impl AsRef<str>, name: impl AsRef<str>,
renderer: &R,
scene: &S,
camera: &C,
samples_per_pixel: usize, samples_per_pixel: usize,
tev: &mut TevClient, tev: &mut TevClient,
) -> Result<(), TevError> { ) -> Result<(), TevError> {
@ -23,7 +34,8 @@ fn render_image<C: ClassicalRenderer<SmallRng> + Sync>(
let mut rng = SmallRng::seed_from_u64((x + y * renderer.width()) as u64); let mut rng = SmallRng::seed_from_u64((x + y * renderer.width()) as u64);
for _ in 0..samples_per_pixel { for _ in 0..samples_per_pixel {
let r = renderer.render_pixel(x, y, &mut rng) / (samples_per_pixel as Float); let r =
renderer.render_pixel(scene, camera, x, y, &mut rng) / (samples_per_pixel as Float);
c[0] += r.r(); c[0] += r.r();
c[1] += r.g(); c[1] += r.g();
c[2] += r.b(); c[2] += r.b();
@ -57,44 +69,62 @@ fn render_image<C: ClassicalRenderer<SmallRng> + Sync>(
} }
fn get_scene() -> (TriangleBVH<SmallRng>, BasicCamera) { fn get_scene() -> (TriangleBVH<SmallRng>, BasicCamera) {
let s = cornel(); let s = basic_cornel();
let c = BasicCamera::new( let c = BasicCamera::new(
400, 400,
400, 400,
Pos3::new(-6.0, 0.0, 0.0), s.camera_pos,
Dir3::new(1.0, 0.0, 0.0), s.camera_dir,
Dir3::up(), s.camera_up,
Float::to_radians(90.0), s.horizontal_fov,
); );
(s, c) (s.scene, c)
} }
fn main() { fn main() {
let mut client = TevClient::wrap(TcpStream::connect("127.0.0.1:14158").unwrap()); let mut client = TevClient::wrap(TcpStream::connect("127.0.0.1:14158").unwrap());
let (s, c) = get_scene(); let (s, c) = get_scene();
let r = DepthRenderer::new(s, c); let width = 400;
render_image(r, "depth renderer", 128, &mut client).unwrap(); let height = 400;
let r = DepthRenderer::new(width, height);
render_image("depth renderer", &r, &s, &c, 128, &mut client).unwrap();
let samples_per_pixel = 1024; let samples_per_pixel = 1024;
let (s, c) = get_scene(); let r = PathTracer::new(width, height);
let r = PathTracer::new(s, c);
render_image(r, "example path tracer", samples_per_pixel, &mut client).unwrap();
let (s, c) = get_scene();
let r = PathTracerImportance::new(s, c);
render_image( render_image(
r, "example path tracer",
"example path tracer importance 1024", &r,
&s,
&c,
samples_per_pixel, samples_per_pixel,
&mut client, &mut client,
) )
.unwrap(); .unwrap();
let (s, c) = get_scene(); let r = PathTracerImportance::new(width, height);
let r = PathTracerImportance::new(s, c); render_image(
render_image(r, "example path tracer importance 2048", 2048, &mut client).unwrap(); "example path tracer importance 1024",
&r,
&s,
&c,
samples_per_pixel,
&mut client,
)
.unwrap();
let r = PathTracerImportance::new(width, height);
render_image(
"example path tracer importance 2048",
&r,
&s,
&c,
2048,
&mut client,
)
.unwrap();
} }