BeauChevalet - fast raycast renders

introduction & benchmarks

introduction & benchmarks

BeauChevalet is a project i’ve been working on for the past 1-2 months, and i’m happy to share a functioning build of it.

the primary goal for BeauChevalet is for fast renders with raycasts at a resolution of 1024x1024, and well, i did the impossible :face_with_hand_over_mouth:

the following benchmarks demonstrates how fast can BeauChevalet can render scenes.

these benchmarks has been conducted on a i5-7300u with max clock speeds and with minimal processes running. BeauChevalet settings was set to default.

GI TEST 64 SAMPLES

CITY

GI BOX UNSHADED


get started

get started

you can get BeauChevalet here

insert the module into a server script in ServerScriptService:

image

to use BeauChevalet, you only have to do the following:

local Renderer = require(script.BeauChevalet) -- change where you placed the BeauChevalet module
Renderer:init(32) -- 32 actors, please adjust for your system
print(Renderer:render_frame()) -- returns the number of seconds it took to render

to see your render, make sure you use Run mode instead of Play mode, if you don’t, you’ll get a visual warning:

also, please adjust the viewport resolution to a constant size no greater than 1024x1024 with the device emulation feature located in the “Test” section.

you can adjust a few settings in the init method:

init : (self : BeauChevalet, threads : number, same_color_thres : number?, raycast_dist : number?) -> void;
  • threads
    the number of threads BeauChevalet will utilize. please be reasonable and do not use outrageous amounts. i recommend you use less threads (lower than 32) because more threads may interfere with with the algorithm that speeds up renders.

  • same_color_thres (optional)
    the threshold for determining if a color is similar, the acceptable values are in the range of 0 - 100, decimals are also acceptable.

    higher values means better performance, while lower values means higher picture quality. the default is 1.

  • raycast_dist (optional)
    how far should BeauChevalet raycast. the default is 500.

BeauChevalet allows you to write your own shaders – with a limitation:

i have avoided doing any sort of communication between threads to ensure maximum performance, and as a consequence, each thread cannot access another’s framebuffer, this means some spatial shaders will not work correctly. this is not going to change.

there are two types of shaders, the raycast shader and the fragment shader.

the raycast shader is intended to handle raycasts, you do all your raycast operations here. this can allow you to do reflections on objects, for example.

the fragment shader is intended for post processing, you have access to the color, normal and depth buffer. this can allow you to create an outline shader, for example.

if you want to create a raycast shader, you first need to create a preset module.

  • create the preset module under the “Pipeline” module
  • in the preset, insert the following:
-- this module is called "Preset", parented at "Pipeline"
return require(script.Parent):LoadModules{
}

LoadModules is responsible for loading each shader in the order you specify.

  • create a new shader under the preset and insert the following:
-- this module is called "Shader", parented at "Preset"
--!native
return require(script.Parent):Attach(function(a0): nil 
   -- your shader code
	return
end)


the hierarchy should look like this:
image

a0 is a typed variable, it contains information about the current object that the raycast hit:

--  a0 type
export type ReturnType = {
	r : number; -- modifiable color r
	g : number; -- modifiable color g
	b : number; -- modifiable color b
	x : number; -- x coordinate of the raycast
	y : number; -- y coordinate of the raycast
	raycast : RaycastResult?; -- raycast result, always verify that `a0.raycast` is not `nil`, otherwise your shader will not work.
	direction : Vector3; -- raycast direction
	normal : Vector3; -- raycast direction
	normalized_depth :number; -- normalized depth, 0 is close, 1 is far
}

you modify the pixel values using the a0.r, a0.g, a0.b properties:

--!native
return require(script.Parent):Attach(function(a0): nil 
-- darken pixel by .5
    a0.r *= .5
    a0.g *= .5
    a0.b *= .5
	return
end)

  • add the shader in the preset:
-- this module is called "Preset", parented at "Pipeline"
return require(script.Parent):LoadModules{
	script.Shader
}

if you have multiple shaders chained together,

return require(script.Parent):LoadModules{
	script.Shader;
	script.Shader2;
	script.Shader3;
}

then the a0.r, a0.g, a0.b values will be passed down to the next shader.

if script.Shader modifies the a0.r, a0.g, a0.b properties, then script.Shader2 will also get the modified a0.r, a0.g, a0.b properties.

if you want to create a fragment shader, the same process applies, however, you place your preset module under the PostProcessing module.

image

the a0 return type for the fragment shader:


export type ReturnType = {
	r : number; -- modifiable color r
	g : number; -- modifiable color g
	b : number; -- modifiable color b
	x : number; -- local x coordinate for the thread's framebuffer
	y : number; -- local y coordinate for the thread's framebuffer
	global_y : number; -- global y coordinate, otherwise known as the current y position of the viewport screen size
	local_screen_size : Vector2; -- current thread's framebuffer screen size
	global_screen_size : Vector2;  -- current viewport screen size
	get_pixel : (number,number) -> (number,number,number); -- function to get the r,g,b values in the normal buffer
	get_normal : (number,number) -> (number,number,number); -- function to get the x,y,z values in the normal buffer
	get_depth : (number,number) -> number; -- function to get the depth value in the depth buffer
	
}
how i did it

how i did it

exploiting spatial redundancy

the optimization method i use relies on exploiting the spatial redundancy of pixels. this method assumes that neighboring pixels in a 4x4 block will have simliar pixel values.

  • raycast every 4th pixel in an evenly spaced grid
  • examine each region, which is done by the following:

consider the:

  • top left and top right pixel
  • bottom left and bottom right pixel
  • top left and bottom left pixel
  • top right and bottom right pixel
  • bottom left and top right pixel
  • bottom right and top left pixel

does the following pixels have a similar color determined by a threshold? if so, then interpolate the missing pixels in-between them, otherwise raycast the missing pixels in-between them.

parallel luau

parallel luau has dramatically decreased rendering times since each piece of the render can be worked on by individual threads.

i have avoided doing any sort of communication between threads to ensure maximum performance, and as a consequence, each thread cannot access another’s framebuffer, this means some spatial shaders will not work correctly. this is not going to change.

buffers

buffers has slightly decreased rendering times, probably because each component is represented by a u8, rather than a f64.

native code gen

less overhead which improves performance.

hello! i’ve condensed this post into collapsible sections!
i hope you enjoy BeauChevalet!

32 Likes

do you have a test place or .rblx file we can test. This looks really cool tbh

example.rbxl (392.2 KB)

4 Likes

thank you I could use this for renders now lol

RobloxScreenShot20241020_135108828

version 1b changelogs

added a sharpening filter to boost visual fidelity

performance impact is minimal, though you can disable it by going to the
PostProcessing module and dragging SharpenPreset to the disabled folder

I might’ve done this wrong but I keep getting these errors even after following the instructions and turning on the EditableImages and EditableMesh from Beta features.

 Unable to create an Instance of type "EditableImage"  -  Server - Canvas:95
  13:30:17.891  Stack Begin  -  Studio
  13:30:17.891  Script 'ServerScriptService.Script.BeauChevalet.Modules.Canvas', Line 95 - function New

Followed by this:

ServerScriptService.Script.BeauChevalet.Modules.ThreadsAPI.Actor.Script:410: attempt to index nil with 'pixels'  -  Server - Script:410
  13:30:18.061  Stack Begin  -  Studio
  13:30:18.062  Script 'ServerScriptService.Script.BeauChevalet.Modules.ThreadsAPI.Actor.Script', Line 410 - function render

image

Do tell me what I did wrong or if something was wrong along with the solution. Thanks in advance!

please update, it has been fixed

version 1c changelogs

roblox updated the editableimage api, things broke, now it’s working again

1 Like