parry3d/partitioning/visitor.rs
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170
use crate::math::{Real, SimdBool, SimdReal, SIMD_WIDTH};
#[cfg(all(feature = "std", feature = "parallel"))]
use crate::partitioning::{qbvh::QbvhNode, SimdNodeIndex};
/// The next action to be taken by a BVH traversal algorithm after having visited a node with some data.
pub enum SimdBestFirstVisitStatus<Res> {
/// The traversal can continue.
MaybeContinue {
/// The weight associated to each child of the node being traversed.
weights: SimdReal,
/// Each lane indicates if the corresponding child of the node being traversed
/// should be traversed too.
mask: SimdBool,
/// Optional results associated to each child of the node being traversed.
results: [Option<Res>; SIMD_WIDTH],
},
/// The traversal aborts.
///
/// If a data is provided, then it is returned as the result of the traversal.
/// If no result is provided, then the last best result found becomes the result of the traversal.
ExitEarly(Option<Res>),
}
/// Trait implemented by cost functions used by the best-first search on a `BVT`.
pub trait SimdBestFirstVisitor<LeafData, SimdBV> {
/// The result of a best-first traversal.
type Result;
/// Compute the next action to be taken by the best-first-search after visiting a node containing the given bounding volume.
fn visit(
&mut self,
best_cost_so_far: Real,
bv: &SimdBV,
value: Option<[Option<&LeafData>; SIMD_WIDTH]>,
) -> SimdBestFirstVisitStatus<Self::Result>;
}
/// The status of the spatial partitioning structure traversal.
pub enum SimdVisitStatus {
/// The traversal should continue on the children of the currently visited nodes for which
/// the boolean lane is set to `1`.
MaybeContinue(SimdBool),
/// The traversal should exit immediately.
ExitEarly,
}
/// The status of the simultaneous traversal of two spatial partitioning structures.
pub enum SimdSimultaneousVisitStatus {
/// The traversal should continue on the children of the currently visited nodes for which
/// the boolean lane is set to `1`.
MaybeContinue([SimdBool; SIMD_WIDTH]),
/// The traversal should exit immediately.
ExitEarly,
}
/// Trait implemented by visitor called during the traversal of a spatial partitioning data structure.
pub trait SimdVisitor<LeafData, SimdBV> {
/// Execute an operation on the content of a node of the spatial partitioning structure.
///
/// Returns whether the traversal should continue on the node's children, if it should not continue
/// on those children, or if the whole traversal should be exited early.
fn visit(
&mut self,
bv: &SimdBV,
data: Option<[Option<&LeafData>; SIMD_WIDTH]>,
) -> SimdVisitStatus;
}
impl<F, LeafData, SimdBV> SimdVisitor<LeafData, SimdBV> for F
where
F: FnMut(&SimdBV, Option<[Option<&LeafData>; SIMD_WIDTH]>) -> SimdVisitStatus,
{
fn visit(
&mut self,
bv: &SimdBV,
data: Option<[Option<&LeafData>; SIMD_WIDTH]>,
) -> SimdVisitStatus {
(self)(bv, data)
}
}
/// Trait implemented by visitor called during the traversal of a spatial partitioning data structure.
pub trait SimdVisitorWithContext<LeafData, SimdBV, Context: Clone> {
/// Execute an operation on the content of a node of the spatial partitioning structure.
///
/// Returns whether the traversal should continue on the node's children, if it should not continue
/// on those children, or if the whole traversal should be exited early. Also returns
/// a context, which may or may not be identical to the input context.
fn visit(
&mut self,
bv: &SimdBV,
data: Option<[Option<&LeafData>; SIMD_WIDTH]>,
context: Context,
) -> (SimdVisitStatus, [Context; SIMD_WIDTH]);
}
/// Trait implemented by visitor called during a simultaneous spatial partitioning data structure tarversal.
pub trait SimdSimultaneousVisitor<T1, T2, SimdBV> {
/// Execute an operation on the content of two nodes, one from each structure.
///
/// Returns whether the traversal should continue on the nodes children, if it should not continue
/// on those children, or if the whole traversal should be exited early.
fn visit(
&mut self,
left_bv: &SimdBV,
left_data: Option<[Option<&T1>; SIMD_WIDTH]>,
right_bv: &SimdBV,
right_data: Option<[Option<&T2>; SIMD_WIDTH]>,
) -> SimdSimultaneousVisitStatus;
}
/*
*
* Parallel visitors below.
*
*/
/// Trait implemented by visitor called during the parallel traversal of a spatial partitioning data structure.
#[cfg(all(feature = "std", feature = "parallel"))]
pub trait ParallelSimdVisitor<LeafData>: Sync {
/// Execute an operation on the content of a node of the spatial partitioning structure.
///
/// Returns whether the traversal should continue on the node's children, if it should not continue
/// on those children, or if the whole traversal should be exited early.
fn visit(
&self,
node_id: SimdNodeIndex,
bv: &QbvhNode,
data: Option<[Option<&LeafData>; SIMD_WIDTH]>,
) -> SimdVisitStatus;
}
#[cfg(all(feature = "std", feature = "parallel"))]
impl<F, LeafData> ParallelSimdVisitor<LeafData> for F
where
F: Sync + Fn(&QbvhNode, Option<[Option<&LeafData>; SIMD_WIDTH]>) -> SimdVisitStatus,
{
fn visit(
&self,
_node_id: SimdNodeIndex,
node: &QbvhNode,
data: Option<[Option<&LeafData>; SIMD_WIDTH]>,
) -> SimdVisitStatus {
(self)(node, data)
}
}
/// Trait implemented by visitor called during a parallel simultaneous spatial partitioning
/// data structure traversal.
#[cfg(all(feature = "std", feature = "parallel"))]
pub trait ParallelSimdSimultaneousVisitor<LeafData1, LeafData2>: Sync {
/// Visitor state data that will be passed down the recursion.
type Data: Copy + Sync + Default;
/// Execute an operation on the content of two nodes, one from each structure.
///
/// Returns whether the traversal should continue on the nodes children, if it should not continue
/// on those children, or if the whole traversal should be exited early.
fn visit(
&self,
left_node_id: SimdNodeIndex,
left_node: &QbvhNode,
left_data: Option<[Option<&LeafData1>; SIMD_WIDTH]>,
right_node_id: SimdNodeIndex,
right_node: &QbvhNode,
right_data: Option<[Option<&LeafData2>; SIMD_WIDTH]>,
visitor_data: Self::Data,
) -> (SimdSimultaneousVisitStatus, Self::Data);
}