From 601beb10a0b784abc0662649ca1774e10343a70e Mon Sep 17 00:00:00 2001 From: Robert Garrett Date: Fri, 18 Aug 2023 21:12:39 -0500 Subject: [PATCH] Control those Uniform distributions I wanted to make the Uniform's into `const`s that live at the global scope. Rust can do global const, but the `Uniform::new()` function can't be used in that context. As an intermediate to a *helpful* solution, I've just pushed them into a struct. One less parameter to name even though it's the same stuff. The compiler should be smart enough to hoist the initialization outside the function and leave them around, but maybe not. After all, the function isn't defined to work in such a way with the `const` keyword :v --- src/main.rs | 34 +++++++++++++++++++++++----------- src/thread_utils.rs | 6 +++--- 2 files changed, 26 insertions(+), 14 deletions(-) diff --git a/src/main.rs b/src/main.rs index 436ce82..1c912b2 100644 --- a/src/main.rs +++ b/src/main.rs @@ -24,8 +24,8 @@ fn main() { // image let aspect_ratio = 3.0 / 2.0; let image = ( - 400, - (400.0 / aspect_ratio) as i32 + 1920, + (1920.0 / aspect_ratio) as i32 ); let samples_per_pixel: u32 = 10; let max_depth = 50; @@ -66,7 +66,7 @@ fn main() { }; thread::scope(|s| { - let (mut dispatcher, scanline_receiver) = thread_utils::Dispatcher::new(&small_rng, 4); + let (mut dispatcher, scanline_receiver) = thread_utils::Dispatcher::new(&small_rng, 12); s.spawn(move || { for y in (0..image.1).rev() { @@ -114,7 +114,7 @@ fn main() { // short-circuit noted above, it could. let last_ind = raster_segments.len() - 1; - if raster_segments[last_ind].line_num == sl_output_index{ + if raster_segments[last_ind].line_num == sl_output_index{ let scanline = raster_segments.pop().unwrap(); print_scanline(scanline, samples_per_pixel); sl_output_index -= 1; @@ -147,19 +147,31 @@ pub struct RenderContext{ camera: Camera, } -fn render_line(y: i32, small_rng: &mut SmallRng, context: RenderContext) -> Vec { +pub struct DistributionContianer { + distrib_zero_one: Uniform, + distrib_plusminus_one: Uniform, +} + +impl DistributionContianer { + fn new() -> Self { + DistributionContianer { + distrib_zero_one: Uniform::new(0.0, 1.0), + distrib_plusminus_one: Uniform::new(-1.0, 1.0), + } + } +} + +fn render_line(y: i32, small_rng: &mut SmallRng, context: RenderContext, distr: &DistributionContianer) -> Vec { //TODO: Ensure that the compiler hoists the distribution's out as constants // else, do so manually - let distrib_zero_one = Uniform::new(0.0, 1.0); - let distrib_plusminus_one = Uniform::new(-1.0, 1.0); - (0..context.image.0).map(|x| { + (0..context.image.0).map(|x| { (0..context.samples_per_pixel).into_iter().fold( Vec3::zero(), |color, _sample| { - let u = ((x as f32) + small_rng.sample(distrib_zero_one)) / ((context.image.0 - 1) as f32); - let v = ((y as f32) + small_rng.sample(distrib_zero_one)) / ((context.image.1 - 1) as f32); + let u = ((x as f32) + small_rng.sample(distr.distrib_zero_one)) / ((context.image.0 - 1) as f32); + let v = ((y as f32) + small_rng.sample(distr.distrib_zero_one)) / ((context.image.1 - 1) as f32); let ray = context.camera.get_ray(u, v, small_rng); - color + ray_color(ray, &context.world, context.max_depth, small_rng, distrib_plusminus_one) + color + ray_color(ray, &context.world, context.max_depth, small_rng, distr.distrib_plusminus_one) } ) }).collect() diff --git a/src/thread_utils.rs b/src/thread_utils.rs index 6156828..ed1d093 100644 --- a/src/thread_utils.rs +++ b/src/thread_utils.rs @@ -1,7 +1,7 @@ use crate::RenderContext; use crate::Vec3; -use crate::render_line; +use crate::{render_line, DistributionContianer}; use core::cmp::Ordering; use std::thread; @@ -76,7 +76,7 @@ impl Dispatcher { let mut srng = srng.clone(); let threads_result_tx = render_tx.clone(); - + let distribs = DistributionContianer::new(); let thread_handle = thread::spawn(move || { while let Ok(job) = command_rx.recv() { match job { @@ -84,7 +84,7 @@ impl Dispatcher { break; } RenderCommand::Line { line_num, context } => { - let line = render_line(line_num, &mut srng, context); + let line = render_line(line_num, &mut srng, context, &distribs); let result = RenderResult { line_num, line }; threads_result_tx.send(result).unwrap(); }