This project uses CMake to build a WebIDL bound version of Embree4. https://github.com/embree/embree https://emscripten.org/docs/porting/connecting_cpp_and_javascript/WebIDL-Binder.html
I am not affiliated with Embree and have no position in this project. All code here is made as a personal project and has no guarantees.
Due to the C-function nature of Embree, most functions in the rtcore_* are wrapped under a singular RTC
object and accessed statically.
The build has been setup to use multiple IDL file fragments to compose a final IDL file. Custom builds may be made by modifying or conditionally including these files to include as much or little functionality as you want.
The IDL fragments containing rtcFunctions have the originating Header.H file note at the top. During the build process, these originating Header files will be accessed out of the Embree project and REGEX analyzed (warning, no guarantees on stability of this) to attempt to derive the method names and parameters. These files will write temporary headers, which will then be dynamically included into the singleton RTC object to form the JS-to-Singleton-to-Embree interface.
Additional methods are added to the Typing and Javascript using the *.extras files. Mainly these deal with memory alignment/management or wrapping/unwrapping of arrays.
There is a docker file that will build the project using the standard Emscripten docker image. This may be changed to alpine and a custom configuration in the future if it is smaller and trustworthy.
This project requires Emscripten and NodeJS to build. I personally use:
EMSDK will require Python3 as well for the webidl_binder and core emscripten functionality. NodeJS will be used to transform the IDL into header files, and for the NPX command used to produce the Typescript typing files (*.d.ts)
To build, run sh build.sh
The build will produce a dist
folder containing:
- embree.d.ts - typescript typing
- embree.js - Emscripten loader module and WebIDL binding
- embree.wasm - Core WASM module - ~7.7MB in size, compressible to 1.5-2MB using zip compression if needed.
If compressing the library, most browsers should support using decompression-streams to stream a raw compressed stream directly into the streaming-compile of wasm, though most server hosting will allow transparent compression.
https://developer.mozilla.org/en-US/docs/Web/API/Compression_Streams_API https://developer.mozilla.org/en-US/docs/WebAssembly/JavaScript_interface/instantiateStreaming_static
For in-library multi-threading, this is only used for BVH building (scene/commits) and not for the ray-tracing. This means that many of the tutorials will work without multi-threading being enabled during the compile, but there are some exceptions.
Anything using tesselation-cache (Grid Geometry Tutorial) during RTC.interpolate1
will freeze if it requires tesselation cache and multi-threading is not enabled.
By default, the tessellation_cach_size
for the project will cause Embree4 to by default attempt to allocate over 100MB of RAM. You can keep this as-is and enable a higher TOTAL_MEMORY, or when requesting a device send the config tessellation_cache_size=0
. You can also compile with ALLOW_MEMORY_GROWTH to allow growth to whatever is needed.
Some tutorials may require Tesselation, in which case they will assert and fail if the cache size is below required. The Grid Geometry
tutorial requires at least a tesselation cache size of 1.
When loading a larger model (Standford Bunny causes this issue), you will need a high STACK_SIZE
(already compiled as is) during the accelerated BVH construction to avoid running the stack out and crashing in the middle of a parallel-for with no error other than 'memory access'. The emscripten stack default was 64k, but the CMake existing build uses a 1MB stack, which alleviated my personal issues, but may be excess for what you require.
With my initial tests of one or two complex models, I did not need more than 6-8MB of RAM, and the default 16MB was more than enough. Most RAM I have needed was purely for storing RayCast results. This can be reduced in the WASM side by implementing tiling and copying off to a shared-array or other location, should you wish to not keep it controlled in the WASM layer. Tutorials may require a higher amount in general. 75-96MB can handle most tutorials.
Some of the higher instance counts, like Forest, may require higher memory limits or compiling with Memory Resize enabled. Forest can handle ~500,000 instances at 96MB.
Custom methods have been added for:
- TotalMemory/FreeMemory - Derived from emscripten's heap tests. Utility usage for checking HEAP space
- Allocating RayHit(1|4|8) - embree appears to not list these structs as being MemAlign in their headers, so custom methods were made to memalign allocate these to prevent assert failures during intersect. If marked as requiring alignment in the Embree header, the WebIDL binding would automatically align these.
- SizeOf RayHit - to assist in direct memory access via Linear memory locations, or assist JS performing a memset to clear the memory of a RayHit between queries.
- Allocate RayMask - this is simply a
int*
allocator, required for the mask-param in any multi-intersect (4/8/16) calls or associated point-query/occlude for thevalid
parameter. Named asRayMask
in the WebIDL side for typing purposes. - tileIntersect(1|4) - demo routines added for personal testing purposes. They will render a tile of data in a single call, and save offset/stride segments of the RayHit item to a supplied buffer, such as just the Z channel of a Ray4 or just the across all 4-units worth of a Ng 12-byte channel for storing hit normals.
- alloc/wrap/copy Typed Array - These methods deal with creation of typed arrays on the Heap. You can pass them to functions using the
typed_array_foo.byteOffset
value. This is understood by web IDL correctly as the pointer. These methods use the Typed Array's knowledge of their own element size during creation, so use element count, not byte size when referencing length- Wrap - (pointer, N length, type) - will take a pointer (such as from C) and wrap it in a typed array of N length. The library has no knowledge of the actual when given an arbitrary poitner length, so it must be provided.
- Alloc ( N length, type) - will allocate an array of N items. (2, F32) will allocate 8 bytes. This will then call wrap using the N length.
- Copy ( array-like, type)- will take an existing array and type, then perform alloc, wrap, and set the contents of the array.
- alloc/copy Aligned Type Array This is the same as above, but uses memalign due to how many internal methods require specific alignment. Wrap does not apply, as you cannot take a non-aligned pointer and wrap it as aligned. The pointer can not change.
Images and commentary on tutorials that have been completed here LINK
import Embree from './em/embree';
// Hacks to make 'static' methods act as
// WebIDL Typescript generator expects
const RTC = (embree.RTC.prototype as typeof Embree.RTC);
function addGroundPlane(device: Embree.Device, scene: Embree.Scene) {
const mesh = RTC.newGeometry(device, embree.RTC_GEOMETRY_TYPE_TRIANGLE);
const vertices = RTC.setNewGeometryBuffer(
mesh, embree.RTC_BUFFER_TYPE_VERTEX,
0, embree.RTC_FORMAT_FLOAT3,
4 * 4, 4);
const f32_vert = new Float32Array(embree.HEAP8.buffer, vertices, 16);
f32_vert.set([
-10, -2, -10, 1,
-10, -2, 10, 1,
10, -2, -10, 1,
10, -2, 10, 1]);
const triangles = RTC.setNewGeometryBuffer(
mesh, embree.RTC_BUFFER_TYPE_INDEX,
0,embree.RTC_FORMAT_UINT3,
3 * 4, 2);
const u32_index = new Uint32Array(embree.HEAP8.buffer, triangles, 6);
u32_index.set([ 0, 1, 2, 1, 3, 2]);
RTC.commitGeometry(mesh);
const geomID = RTC.attachGeometry(scene, mesh);
RTC.releaseGeometry(mesh);
return geomID;
}
const device = RTC.newDevice('verbose=3,threads=1,tessellation_cache_size=0');
addGroundPlane(device, scene);
RTC.commitScene(scene);
//raytrace here