Skip to content

acfr/gpu-ray-surface-intersection-in-cuda

Repository files navigation

GPU implementation of a ray-surface intersection algorithm in CUDA

This repository provides an open-source, GPU-based implementation of a ray-surface intersection algorithm in CUDA. The basic problem it solves is that given a set of rays (line segments) and a triangular mesh surface, it identifies the rays that cross the surface. For our use case, we are primarily interested in knowing whether an intersection has occured. However, the program can be configured to return the intersecting point and triangle for each intersecting ray. In our application, ray segments can point in arbitrary directions and do not originate from shared light sources.

This contribution is not geared towards game development or computer graphics per se. The motivation for writing this code is to speed up the solver relative to CPU-based implementations to increase productivity in an R&D project, rather than achieving real-time rendering performance.

Rays in red intersect with the triangular mesh surfaceCode snippet: a device function in bvh_structure.h

Key words

Moller-Trumbore algorithm, ray-triangle intersection, linear bounding volume hierarchy, LBVH construction, binary radix tree, BVH traversal, bounding box collision detection, parallel computing, GPGPU, CUDA.

Audience

  • This post would offer little to experienced programmers who are already skilled in general-purpose GPU computing, parallel algorithms and real-time ray tracing.
  • It might be of interest to people with limited CUDA programming experience, and perhaps researchers/engineers looking for a "ready-made" working implementation of a ray-surface intersection algorithm that runs on Nvidia GPU devices.

Background

There are two technical components to the code. An exact algorithm for finding ray-triangle intersections (such as the Moller-Trumbore algorithm) and accelerating data structures (such as bounding volume hierarchy or BVH [1]) that spares us from having to evaluate bounding box collision for every possible triangle and line-segment combination.

In regard to ray-triangle intersection tests, Jimenez et al. have analysed half a dozen algorithms, and determined the Moller-Trumbore to be the most efficient in a general setting. Those findings are reported in [2]. In regard to BVH, our implementation is inspired by the approaches described by Robbin Marcus [3] and Tero Kerras [4]. Their blog entries provide practical guidance and the necessary background for understanding this code. Interested readers are referred to these online resources for bridging material.

Usage

The three supported use cases are

  • Standard usage (mode=boolean)
    • This returns a boolean indicating whether or not each ray intersects with the surface.
  • Extended usage (mode=barycentric)
    • This returns the intersecting triangle and nearest intersecting point for each surface intersecting ray.
  • Experimental feature (mode=intercept_count)
    • This returns the number of unique intersections with the surface for each ray.

The code is compiled with

/usr/local/cuda/bin/nvcc gpu_ray_surface_intersect.cu -o gpu_ray_surface_intersect

and executed using

gpu_ray_surface_intersect <vertices> <triangles> <rayFrom> <rayTo> <feedback> <mode>

Setting feedback=silent suppresses terminal output. Command line arguments for the three use cases are described in Section 2.3 and 2.4.

For convenience, scripts/py_gpu_ray_surface_intersect.py implements a wrapper class PyGpuRSI that encapsulates the functionality of gpu_ray_surface_intersect.cu. The notebook scripts/demo.ipynb shows how data manipulation, compilation, run and clean-up steps can be managed using a with statement. To produce results in the form of (intersecting_rays, distances, intersecting_triangles, intersecting_points) in lieu of a crossing_detected binary array, please refer to Part B of scripts/demo.ipynb. Example of the last use case is shown in Part C of scripts/experimental_feature.ipynb.

Documentation

The PDF document provides an overall description of the implementation and discusses issues such as collision buffer management. The aim is to shed light on the more obscure parts of the code, and provide some clarity on certain practical considerations that are sometimes omitted in discussion. Compilation instruction, usage command, CPU/GPU specifications and run time measurements are also included for reference.

A self-contained PyCUDA implementation is also available. PyCUDA provides a Python scripting approach to GPU Run-Time Code Generation. Refer to pycuda/README.md for more information.

References

License

📋 This project is licensed under the terms of the BSD 3-Clause license.

If you find this code useful, you are welcome to cite this in your work: