VTK-m  2.3
exec/CellLocatorBoundingIntervalHierarchy.h
Go to the documentation of this file.
1 //============================================================================
2 // Copyright (c) Kitware, Inc.
3 // All rights reserved.
4 // See LICENSE.txt for details.
5 //
6 // This software is distributed WITHOUT ANY WARRANTY; without even
7 // the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR
8 // PURPOSE. See the above copyright notice for more information.
9 //============================================================================
10 #ifndef vtk_m_exec_CellLocatorBoundingIntervalHierarchy_h
11 #define vtk_m_exec_CellLocatorBoundingIntervalHierarchy_h
12 
15 #include <vtkm/cont/ArrayHandle.h>
17 #include <vtkm/exec/CellInside.h>
19 
20 namespace vtkm
21 {
22 namespace exec
23 {
24 
25 
26 
27 
29 {
30 #if defined(VTKM_CLANG)
31 #pragma GCC diagnostic push
32 #pragma GCC diagnostic ignored "-Wnested-anon-types"
33 #endif // gcc || clang
37  union {
38  struct
39  {
42  } Node;
43  struct
44  {
47  } Leaf;
48  };
49 #if defined(VTKM_CLANG)
50 #pragma GCC diagnostic pop
51 #endif // gcc || clang
52 
55  : Dimension()
56  , ParentIndex()
57  , ChildIndex()
58  , Node{ 0, 0 }
59  {
60  }
61 }; // struct CellLocatorBoundingIntervalHierarchyNode
62 
72 template <typename CellSetType>
74 {
75  using NodeArrayHandle =
78 
79 public:
80  VTKM_CONT
82  const NodeArrayHandle& nodes,
83  const CellIdArrayHandle& cellIds,
84  const CellSetType& cellSet,
87  vtkm::cont::Token& token)
88  : Nodes(nodes.PrepareForInput(device, token))
89  , CellIds(cellIds.PrepareForInput(device, token))
90  , CellSet(cellSet.PrepareForInput(device, VisitType(), IncidentType(), token))
91  , Coords(coords.PrepareForInput(device, token))
92  {
93  }
94 
96  struct LastCell
97  {
98  vtkm::Id CellId = -1;
99  vtkm::Id NodeIdx = -1;
100  };
101 
104  vtkm::Id& cellId,
105  vtkm::Vec3f& parametric) const
106  {
107  LastCell lastCell;
108  return this->FindCellImpl(point, cellId, parametric, lastCell);
109  }
110 
113  vtkm::Id& cellId,
114  vtkm::Vec3f& parametric,
115  LastCell& lastCell) const
116  {
117  cellId = -1;
118 
119  //Check the last cell.
120  if ((lastCell.CellId >= 0) && (lastCell.CellId < this->CellSet.GetNumberOfElements()))
121  {
122  if (this->PointInCell(point, lastCell.CellId, parametric) == vtkm::ErrorCode::Success)
123  {
124  cellId = lastCell.CellId;
126  }
127  }
128 
129  //Check the last leaf node.
130  if ((lastCell.NodeIdx >= 0) && (lastCell.NodeIdx < this->Nodes.GetNumberOfValues()))
131  {
132  const auto& node = this->Nodes.Get(lastCell.NodeIdx);
133 
134  if (node.ChildIndex < 0)
135  {
136  VTKM_RETURN_ON_ERROR(this->FindInLeaf(point, parametric, node, cellId));
137  if (cellId != -1)
138  {
139  lastCell.CellId = cellId;
141  }
142  }
143  }
144 
145  //No fastpath. Do a full search.
146  return this->FindCellImpl(point, cellId, parametric, lastCell);
147  }
148 
149  VTKM_EXEC
151  vtkm::Id& cellId,
152  vtkm::Vec3f& parametric,
153  LastCell& lastCell) const
154  {
155  cellId = -1;
156  vtkm::Id nodeIndex = 0;
157  FindCellState state = FindCellState::EnterNode;
158 
159  while ((cellId < 0) && !((nodeIndex == 0) && (state == FindCellState::AscendFromNode)))
160  {
161  switch (state)
162  {
163  case FindCellState::EnterNode:
165  this->EnterNode(state, point, cellId, nodeIndex, parametric, lastCell));
166  break;
167  case FindCellState::AscendFromNode:
168  this->AscendFromNode(state, nodeIndex);
169  break;
170  case FindCellState::DescendLeftChild:
171  this->DescendLeftChild(state, point, nodeIndex);
172  break;
173  case FindCellState::DescendRightChild:
174  this->DescendRightChild(state, point, nodeIndex);
175  break;
176  }
177  }
178 
179  if (cellId >= 0)
180  {
182  }
183  else
184  {
186  }
187  }
188 
189 private:
190  enum struct FindCellState
191  {
192  EnterNode,
193  AscendFromNode,
194  DescendLeftChild,
195  DescendRightChild
196  };
197 
198  VTKM_EXEC
200  const vtkm::Vec3f& point,
201  vtkm::Id& cellId,
202  vtkm::Id nodeIndex,
203  vtkm::Vec3f& parametric,
204  LastCell& lastCell) const
205  {
206  VTKM_ASSERT(state == FindCellState::EnterNode);
207 
208  const vtkm::exec::CellLocatorBoundingIntervalHierarchyNode& node = this->Nodes.Get(nodeIndex);
209 
210  if (node.ChildIndex < 0)
211  {
212  // In a leaf node. Look for a containing cell.
213  VTKM_RETURN_ON_ERROR(this->FindInLeaf(point, parametric, node, cellId));
214  state = FindCellState::AscendFromNode;
215  if (cellId != -1)
216  {
217  lastCell.CellId = cellId;
218  lastCell.NodeIdx = nodeIndex;
219  }
220  }
221  else
222  {
223  state = FindCellState::DescendLeftChild;
224  }
226  }
227 
228  VTKM_EXEC
229  void AscendFromNode(FindCellState& state, vtkm::Id& nodeIndex) const
230  {
231  VTKM_ASSERT(state == FindCellState::AscendFromNode);
232 
233  vtkm::Id childNodeIndex = nodeIndex;
235  this->Nodes.Get(childNodeIndex);
236  nodeIndex = childNode.ParentIndex;
238  this->Nodes.Get(nodeIndex);
239 
240  if (parentNode.ChildIndex == childNodeIndex)
241  {
242  // Ascending from left child. Descend into the right child.
243  state = FindCellState::DescendRightChild;
244  }
245  else
246  {
247  VTKM_ASSERT(parentNode.ChildIndex + 1 == childNodeIndex);
248  // Ascending from right child. Ascend again. (Don't need to change state.)
249  }
250  }
251 
252  VTKM_EXEC
253  void DescendLeftChild(FindCellState& state, const vtkm::Vec3f& point, vtkm::Id& nodeIndex) const
254  {
255  VTKM_ASSERT(state == FindCellState::DescendLeftChild);
256 
257  const vtkm::exec::CellLocatorBoundingIntervalHierarchyNode& node = this->Nodes.Get(nodeIndex);
258  const vtkm::FloatDefault& coordinate = point[node.Dimension];
259  if (coordinate <= node.Node.LMax)
260  {
261  // Left child does contain the point. Do the actual descent.
262  nodeIndex = node.ChildIndex;
263  state = FindCellState::EnterNode;
264  }
265  else
266  {
267  // Left child does not contain the point. Skip to the right child.
268  state = FindCellState::DescendRightChild;
269  }
270  }
271 
272  VTKM_EXEC
273  void DescendRightChild(FindCellState& state, const vtkm::Vec3f& point, vtkm::Id& nodeIndex) const
274  {
275  VTKM_ASSERT(state == FindCellState::DescendRightChild);
276 
277  const vtkm::exec::CellLocatorBoundingIntervalHierarchyNode& node = this->Nodes.Get(nodeIndex);
278  const vtkm::FloatDefault& coordinate = point[node.Dimension];
279  if (coordinate >= node.Node.RMin)
280  {
281  // Right child does contain the point. Do the actual descent.
282  nodeIndex = node.ChildIndex + 1;
283  state = FindCellState::EnterNode;
284  }
285  else
286  {
287  // Right child does not contain the point. Skip to ascent
288  state = FindCellState::AscendFromNode;
289  }
290  }
291 
293  const vtkm::Vec3f& point,
294  vtkm::Vec3f& parametric,
296  vtkm::Id& containingCellId) const
297  {
298  for (vtkm::Id i = node.Leaf.Start; i < node.Leaf.Start + node.Leaf.Size; ++i)
299  {
300  vtkm::Id cellId = this->CellIds.Get(i);
301 
302  if (this->PointInCell(point, cellId, parametric) == vtkm::ErrorCode::Success)
303  {
304  containingCellId = cellId;
306  }
307  }
308 
309  containingCellId = -1;
311  }
312 
313  // template <typename CoordsType, typename CellShapeTag>
315  vtkm::Id& cellId,
316  vtkm::Vec3f& parametric) const
317  {
318  using IndicesType = typename CellSetPortal::IndicesType;
319  IndicesType cellPointIndices = this->CellSet.GetIndices(cellId);
320  vtkm::VecFromPortalPermute<IndicesType, CoordsPortal> cellPoints(&cellPointIndices,
321  this->Coords);
322  auto cellShape = this->CellSet.GetCellShape(cellId);
323  bool isInside;
324  VTKM_RETURN_ON_ERROR(IsPointInCell(point, parametric, cellShape, cellPoints, isInside));
325 
326  if (isInside && vtkm::exec::CellInside(parametric, cellShape))
328 
330  }
331 
332  template <typename CoordsType, typename CellShapeTag>
334  vtkm::Vec3f& parametric,
335  CellShapeTag cellShape,
336  const CoordsType& cellPoints,
337  bool& isInside)
338  {
339  isInside = false;
340  VTKM_RETURN_ON_ERROR(vtkm::exec::WorldCoordinatesToParametricCoordinates(
341  cellPoints, point, cellShape, parametric));
342  isInside = vtkm::exec::CellInside(parametric, cellShape);
344  }
345 
348  using NodePortal = typename NodeArrayHandle::ReadPortalType;
350  using CellSetPortal =
351  typename CellSetType::template ExecConnectivityType<VisitType, IncidentType>;
352  using CoordsPortal = typename vtkm::cont::CoordinateSystem::MultiplexerArrayType::ReadPortalType;
353 
358 }; // class CellLocatorBoundingIntervalHierarchy
359 
360 } // namespace exec
361 
362 } // namespace vtkm
363 
364 #endif //vtk_m_exec_CellLocatorBoundingIntervalHierarchy_h
vtkm::exec::CellLocatorBoundingIntervalHierarchy::PointInCell
vtkm::ErrorCode PointInCell(const vtkm::Vec3f &point, vtkm::Id &cellId, vtkm::Vec3f &parametric) const
Definition: exec/CellLocatorBoundingIntervalHierarchy.h:314
vtkm::TopologyElementTagPoint
A tag used to identify the point elements in a topology.
Definition: TopologyElementTag.h:34
vtkm::exec::CellLocatorBoundingIntervalHierarchyNode
Definition: exec/CellLocatorBoundingIntervalHierarchy.h:28
vtkm::cont::ArrayHandle< vtkm::exec::CellLocatorBoundingIntervalHierarchyNode >
ArrayHandle.h
vtkm::ErrorCode
ErrorCode
Identifies whether an operation was successful or what type of error it had.
Definition: ErrorCode.h:28
vtkm::exec::CellLocatorBoundingIntervalHierarchy::NodePortal
typename NodeArrayHandle::ReadPortalType NodePortal
Definition: exec/CellLocatorBoundingIntervalHierarchy.h:348
VTKM_EXEC
#define VTKM_EXEC
Definition: ExportMacros.h:51
vtkm::exec::CellLocatorBoundingIntervalHierarchy
Structure for locating cells.
Definition: exec/CellLocatorBoundingIntervalHierarchy.h:73
vtkm
Groups connected points that have the same field value.
Definition: Atomic.h:19
vtkm::exec::CellLocatorBoundingIntervalHierarchy::AscendFromNode
void AscendFromNode(FindCellState &state, vtkm::Id &nodeIndex) const
Definition: exec/CellLocatorBoundingIntervalHierarchy.h:229
VTKM_ASSERT
#define VTKM_ASSERT(condition)
Definition: Assert.h:43
VTKM_EXEC_CONT
#define VTKM_EXEC_CONT
Definition: ExportMacros.h:52
vtkm::IdComponent
vtkm::Int32 IdComponent
Base type to use to index small lists.
Definition: Types.h:194
vtkm::exec::CellLocatorBoundingIntervalHierarchy::IsPointInCell
static vtkm::ErrorCode IsPointInCell(const vtkm::Vec3f &point, vtkm::Vec3f &parametric, CellShapeTag cellShape, const CoordsType &cellPoints, bool &isInside)
Definition: exec/CellLocatorBoundingIntervalHierarchy.h:333
vtkm::exec::CellLocatorBoundingIntervalHierarchyNode::ParentIndex
vtkm::Id ParentIndex
Definition: exec/CellLocatorBoundingIntervalHierarchy.h:35
vtkm::ErrorCode::Success
@ Success
A successful operation.
vtkm::exec::CellLocatorBoundingIntervalHierarchyNode::Dimension
vtkm::IdComponent Dimension
Definition: exec/CellLocatorBoundingIntervalHierarchy.h:34
vtkm::exec::CellLocatorBoundingIntervalHierarchy::LastCell::NodeIdx
vtkm::Id NodeIdx
Definition: exec/CellLocatorBoundingIntervalHierarchy.h:99
vtkm::exec::CellLocatorBoundingIntervalHierarchyNode::ChildIndex
vtkm::Id ChildIndex
Definition: exec/CellLocatorBoundingIntervalHierarchy.h:36
CoordinateSystem.h
vtkm::exec::CellLocatorBoundingIntervalHierarchyNode::LMax
vtkm::FloatDefault LMax
Definition: exec/CellLocatorBoundingIntervalHierarchy.h:40
vtkm::exec::CellLocatorBoundingIntervalHierarchy::CellSet
CellSetPortal CellSet
Definition: exec/CellLocatorBoundingIntervalHierarchy.h:356
vtkm::exec::CellLocatorBoundingIntervalHierarchy::FindInLeaf
vtkm::ErrorCode FindInLeaf(const vtkm::Vec3f &point, vtkm::Vec3f &parametric, const vtkm::exec::CellLocatorBoundingIntervalHierarchyNode &node, vtkm::Id &containingCellId) const
Definition: exec/CellLocatorBoundingIntervalHierarchy.h:292
vtkm::cont::ArrayHandle< vtkm::Id >::ReadPortalType
typename StorageType::ReadPortalType ReadPortalType
The type of portal used when accessing data in a read-only mode.
Definition: ArrayHandle.h:312
VecFromPortalPermute.h
vtkm::exec::CellLocatorBoundingIntervalHierarchy::CoordsPortal
typename vtkm::cont::CoordinateSystem::MultiplexerArrayType::ReadPortalType CoordsPortal
Definition: exec/CellLocatorBoundingIntervalHierarchy.h:352
vtkm::exec::CellLocatorBoundingIntervalHierarchy::FindCell
vtkm::ErrorCode FindCell(const vtkm::Vec3f &point, vtkm::Id &cellId, vtkm::Vec3f &parametric, LastCell &lastCell) const
Locate the cell containing the provided point.
Definition: exec/CellLocatorBoundingIntervalHierarchy.h:112
vtkm::exec::CellLocatorBoundingIntervalHierarchy::FindCell
vtkm::ErrorCode FindCell(const vtkm::Vec3f &point, vtkm::Id &cellId, vtkm::Vec3f &parametric) const
Locate the cell containing the provided point.
Definition: exec/CellLocatorBoundingIntervalHierarchy.h:103
vtkm::exec::CellLocatorBoundingIntervalHierarchy::Nodes
NodePortal Nodes
Definition: exec/CellLocatorBoundingIntervalHierarchy.h:354
vtkm::exec::CellLocatorBoundingIntervalHierarchy::CellIds
CellIdPortal CellIds
Definition: exec/CellLocatorBoundingIntervalHierarchy.h:355
vtkm::exec::CellLocatorBoundingIntervalHierarchy::FindCellState
FindCellState
Definition: exec/CellLocatorBoundingIntervalHierarchy.h:190
vtkm::cont::Token
A token to hold the scope of an ArrayHandle or other object.
Definition: Token.h:35
vtkm::exec::CellLocatorBoundingIntervalHierarchy::CellSetPortal
typename CellSetType::template ExecConnectivityType< VisitType, IncidentType > CellSetPortal
Definition: exec/CellLocatorBoundingIntervalHierarchy.h:351
vtkm::cont::CoordinateSystem::MultiplexerArrayType
vtkm::cont::ArrayHandleMultiplexerFromList< vtkm::ListAppend< ArraysFloatDefault, ArraysFloatNonDefault > > MultiplexerArrayType
Definition: CoordinateSystem.h:100
vtkm::exec::CellLocatorBoundingIntervalHierarchyNode::Size
vtkm::Id Size
Definition: exec/CellLocatorBoundingIntervalHierarchy.h:46
CellInside.h
vtkm::exec::CellLocatorBoundingIntervalHierarchyNode::RMin
vtkm::FloatDefault RMin
Definition: exec/CellLocatorBoundingIntervalHierarchy.h:41
vtkm::exec::CellLocatorBoundingIntervalHierarchy::CellLocatorBoundingIntervalHierarchy
CellLocatorBoundingIntervalHierarchy(const NodeArrayHandle &nodes, const CellIdArrayHandle &cellIds, const CellSetType &cellSet, const vtkm::cont::CoordinateSystem::MultiplexerArrayType &coords, vtkm::cont::DeviceAdapterId device, vtkm::cont::Token &token)
Definition: exec/CellLocatorBoundingIntervalHierarchy.h:81
vtkm::exec::CellLocatorBoundingIntervalHierarchy::LastCell
Structure capturing the location of a cell in the search structure.
Definition: exec/CellLocatorBoundingIntervalHierarchy.h:96
vtkm::exec::CellLocatorBoundingIntervalHierarchy::Coords
CoordsPortal Coords
Definition: exec/CellLocatorBoundingIntervalHierarchy.h:357
VTKM_CONT
#define VTKM_CONT
Definition: ExportMacros.h:57
vtkm::Id
vtkm::Int64 Id
Base type to use to index arrays.
Definition: Types.h:227
vtkm::exec::CellLocatorBoundingIntervalHierarchyNode::Leaf
struct vtkm::exec::CellLocatorBoundingIntervalHierarchyNode::@1::@4 Leaf
vtkm::exec::CellLocatorBoundingIntervalHierarchy::DescendLeftChild
void DescendLeftChild(FindCellState &state, const vtkm::Vec3f &point, vtkm::Id &nodeIndex) const
Definition: exec/CellLocatorBoundingIntervalHierarchy.h:253
vtkm::ErrorCode::CellNotFound
@ CellNotFound
A cell matching some given criteria could not be found.
vtkm::cont::DeviceAdapterId
An object used to specify a device.
Definition: DeviceAdapterTag.h:58
vtkm::Vec< vtkm::FloatDefault, 3 >
vtkm::exec::CellLocatorBoundingIntervalHierarchy::FindCellImpl
vtkm::ErrorCode FindCellImpl(const vtkm::Vec3f &point, vtkm::Id &cellId, vtkm::Vec3f &parametric, LastCell &lastCell) const
Definition: exec/CellLocatorBoundingIntervalHierarchy.h:150
vtkm::FloatDefault
vtkm::Float32 FloatDefault
The floating point type to use when no other precision is specified.
Definition: Types.h:236
vtkm::exec::CellLocatorBoundingIntervalHierarchyNode::Start
vtkm::Id Start
Definition: exec/CellLocatorBoundingIntervalHierarchy.h:45
vtkm::exec::CellLocatorBoundingIntervalHierarchy::LastCell::CellId
vtkm::Id CellId
Definition: exec/CellLocatorBoundingIntervalHierarchy.h:98
VTKM_RETURN_ON_ERROR
#define VTKM_RETURN_ON_ERROR(call)
Definition: ErrorCode.h:202
vtkm::exec::CellLocatorBoundingIntervalHierarchy::CellIdPortal
typename CellIdArrayHandle::ReadPortalType CellIdPortal
Definition: exec/CellLocatorBoundingIntervalHierarchy.h:349
vtkm::exec::CellLocatorBoundingIntervalHierarchy::DescendRightChild
void DescendRightChild(FindCellState &state, const vtkm::Vec3f &point, vtkm::Id &nodeIndex) const
Definition: exec/CellLocatorBoundingIntervalHierarchy.h:273
vtkm::TopologyElementTagCell
A tag used to identify the cell elements in a topology.
Definition: TopologyElementTag.h:24
vtkm::exec::CellLocatorBoundingIntervalHierarchyNode::CellLocatorBoundingIntervalHierarchyNode
CellLocatorBoundingIntervalHierarchyNode()
Definition: exec/CellLocatorBoundingIntervalHierarchy.h:54
VTKM_ALWAYS_EXPORT
#define VTKM_ALWAYS_EXPORT
Definition: ExportMacros.h:89
vtkm::exec::CellLocatorBoundingIntervalHierarchyNode::Node
struct vtkm::exec::CellLocatorBoundingIntervalHierarchyNode::@1::@3 Node
ParametricCoordinates.h
vtkm::exec::CellLocatorBoundingIntervalHierarchy::EnterNode
vtkm::ErrorCode EnterNode(FindCellState &state, const vtkm::Vec3f &point, vtkm::Id &cellId, vtkm::Id nodeIndex, vtkm::Vec3f &parametric, LastCell &lastCell) const
Definition: exec/CellLocatorBoundingIntervalHierarchy.h:199
TopologyElementTag.h
vtkm::VecFromPortalPermute
A short vector from an ArrayPortal and a vector of indices.
Definition: VecFromPortalPermute.h:28