Carlos Ureña / Varios

Efficient ray-triangle intersection test

1. Abstract

I introduce here a time-optimized ray-triangle intersection test, where time optimization is obtained at the cost of using a dual representation of the triangle. The algorithm is a variant of known algorithms, which precomputes as much values as possible. I show how this dual representation can be obtained from the usual representation, and vice-versa, the usual representation can be recovered from the dual.

2. Triangle and ray

The triangle is defined by three arbitrary 3D vertex points, we call them v0,v1,v2 (in no particular order), while the ray is defined by origin o and direction vector d (not necessarily normalized). In what follows, vertex v0 is called the triangle origin, while edges from the origin are called e1=v1v0 and e2=v2v0. The (normalized) normal of the triangle is n, which can be obtained as: n  =  e1×e2e1×e2 all these vectors are represented by 3-tuples, holding their coordinates w.r.t world reference system , whose origin is 0.

3. Intersection point properties

Let us call p to the point of intersection between the line containing the ray and the plane Π containing the triangle (assuming they are not parallel), then obviously there exist three real values t,c1,c2 such that: (1)p  =  o+td  =  v0+c1e1+c2e2 note that t is the distance to the point of intersection from ray origin, and (c1,c2) are the coordinates of p with respect to the coordinate frame defined by e1,e2 in plane Π (the pair of vectors {e1,e2} form a basis for free vectors in plane Π).

The problem of intersection computation is in fact the problem of computing t,c1,c2, because there is an intersection if and only if each of c1,c2 and c1+c2 are in the interval [0,1], and t>0. In that case p is easily computed from (1).

Computation of t can be done, as usual, by observing that vector pv0 is in Π thus perpendicular to n, that is (pv0)n=0. From this equality we can solve for t, we get: (2)t  =  (ov0)ndn  =  onk3dn            where:   k3=v0n

Once t is computed, we obtain p from (1). Then we can directly compute both c1 and c2 as follows: (3)c1  =  (pv0)e¯1  =  pe¯1k1c2  =  (pv0)e¯2  =  pe¯2k2           where:      k1  =  v0e¯1k2  =  v0e¯2 vectors e¯1 and e¯2 are called dual edges and are introduced in the next section.

4. Reciprocal basis

As stated, vectors {e1,e2} form a basis for the space of all free vectors in plane Π. Thus there exists another basis for that space which is called the reciprocal or dual basis, and which we call {e¯1,e¯2}. The following equalities hold: (4)e1e¯1 = 1      e1e¯2 = 0e2e¯1 = 0      e2e¯2 = 1

These equalities can be used to obtain explicit expressions for {e¯1,e¯2} in terms of {e1,e2}. We know there exist four unknown real values a,b,c and d such that e¯1=ae1+be2 and e¯2=ce1+de2. By substituting these expression in (4) we arrive to a system of four equation and four unknowns, then by solving it we can write the dual basis in terms of the original one, as follows: (5)e¯1  =  s[(e2e2)e1(e1e2)e2]e¯2  =  s[(e1e1)e2(e1e2)e1]      where:   s  =  1(e1e1)(e2e2)(e1e2)2 value s must be finite when e1 and e2 are not parallel neither null. By substituting (5) into (4) it can be verified that (4) holds.

5. Dual representation of triangles

The intersection algorithm must precompute and store, for each triangle, these values and vectors (which we call the dual representation of the triangle): e¯1 = s[(e2e2)e1(e1e2)e2]                          k1 = v0e¯1e¯2 = s[(e1e1)e2(e1e2)e1]                          k2 = v0e¯2n  =  (e1×e2)/e1×e2                         k3 = v0n note that basis {e¯1,e¯2,n} is the dual basis of {e1,e2,n} (because, as n is normalized and perpendicular to Π, it holds nn=1, and ne¯i=nei=0). Tuple (k1,k2,k3) holds the coordinates of v0 (as a free vector) w.r.t. basis {e1,e2,n}.

For a nearly degenerated triangle (e1 nearly parallel to e2, or either e1 or e2 nearly null), the absolute value of 1/s is near 0. Thus, computing 1/s can be used for discarding nearly degenerated triangles.

6. The algorithm

The intersection algorithm can be written by using the dual representation, in such a way that it exits as soon as possible when there is no intersection:

  1. Let a=dn
  2. If a<ϵ then return (no intersection: line nearly parallel to plane)
  3. Let t= (onk3)/a
  4. If t0 then return (no intersection: line intersects plane in the negative half-line)
  5. Let p= o+td
  6. Let c1= pe¯1k1
  7. If c1<0 or 1<c1 then return (no intersection: point not in triangle)
  8. Let c2= pe¯2k2
  9. If c2<0 or 1<c2 then return (no intersection: point not in triangle)
  10. If 1<c1+c2 then return (no intersection: point not in triangle)
  11. Return (p,t,c1,c2) (there is an intersection)
This algorithm requires at least two and at most four dot products. On the other hand, it requires storage for 12 real values per triangle.

7. Using homogeneous coordinates

In some case it can be efficient to use 4-tuple dot product routines implemented in hardware. In the algorithm above, this can be done for all the dots products involved, by extending the dual basis vectors with k1,k2 and k3 as their homogeneous coordinates, then t,c1 and c2 can be computed as: a=dnt=(on)/ap=o+tdc1=pe¯1c2=pe¯2         where:    {o=[o:1]d=[d:0]n=[n:k3]e¯1=[e¯1:k1]e¯2=[e¯2:k2]

8. Recovering original representation.

Vectors v0,e1 and e2 can be recovered from the dual representation. Edges {e1,e2} can be easily obtained from {e¯1,e¯2} by using the inverse version of (5), because equalities (4) are symmetric, that is: (6)e1  =  s[(e¯2e¯2)e¯1(e¯1e¯2)e¯2]e2  =  s[(e¯1e¯1)e¯2(e¯1e¯2)e¯1]          where:  s = 1(e¯1e¯1)(e¯2e¯2)(e¯1e¯2)2 Vertex v0 can be computed from its coordinates (k1,k2,k3) (w.r.t basis {e1,e2,n}), namely: v0 = k1e1+k2e2+k3n

9. See also:

(this page uses MathJax)
URI: https://www.ugr.es/~curena/varios/rtint/
Actualizado: 09/11/2023 10:37:01.