VTK-m  2.3
exec/CellLocatorTwoLevel.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_CellLocatorTwoLevel_h
11 #define vtk_m_exec_CellLocatorTwoLevel_h
12 
13 #include <vtkm/exec/CellInside.h>
15 
16 #include <vtkm/cont/ArrayHandle.h>
18 
19 #include <vtkm/Math.h>
21 #include <vtkm/Types.h>
23 #include <vtkm/VecTraits.h>
24 
25 namespace vtkm
26 {
27 namespace internal
28 {
29 namespace cl_uniform_bins
30 {
31 
32 using DimensionType = vtkm::Int16;
33 using DimVec3 = vtkm::Vec<DimensionType, 3>;
34 using FloatVec3 = vtkm::Vec3f;
35 
36 struct Grid
37 {
38  DimVec3 Dimensions;
39  // Bug in CUDA 9.2 where having this gap for alignment was for some reason setting garbage
40  // in a union with other cell locators (or perhaps not properly copying data). This appears
41  // to be fixed by CUDA 10.2.
42  DimensionType Padding;
43  FloatVec3 Origin;
44  FloatVec3 BinSize;
45 };
46 
47 struct Bounds
48 {
49  FloatVec3 Min;
50  FloatVec3 Max;
51 };
52 
53 VTKM_EXEC inline vtkm::Id ComputeFlatIndex(const DimVec3& idx, const DimVec3 dim)
54 {
55  return idx[0] + (dim[0] * (idx[1] + (dim[1] * idx[2])));
56 }
57 
58 VTKM_EXEC inline Grid ComputeLeafGrid(const DimVec3& idx, const DimVec3& dim, const Grid& l1Grid)
59 {
60  return { dim,
61  0,
62  l1Grid.Origin + (static_cast<FloatVec3>(idx) * l1Grid.BinSize),
63  l1Grid.BinSize / static_cast<FloatVec3>(dim) };
64 }
65 
66 template <typename PointsVecType>
67 VTKM_EXEC inline Bounds ComputeCellBounds(const PointsVecType& points)
68 {
69  using CoordsType = typename vtkm::VecTraits<PointsVecType>::ComponentType;
71 
72  CoordsType minp = points[0], maxp = points[0];
73  for (vtkm::IdComponent i = 1; i < numPoints; ++i)
74  {
75  minp = vtkm::Min(minp, points[i]);
76  maxp = vtkm::Max(maxp, points[i]);
77  }
78 
79  return { FloatVec3(minp), FloatVec3(maxp) };
80 }
81 }
82 }
83 } // vtkm::internal::cl_uniform_bins
84 
85 namespace vtkm
86 {
87 namespace exec
88 {
89 
90 //--------------------------------------------------------------------
91 
101 template <typename CellStructureType>
103 {
104 private:
107 
108  template <typename T>
110 
111  using CoordsPortalType =
112  typename vtkm::cont::CoordinateSystem::MultiplexerArrayType::ReadPortalType;
113 
114  // TODO: This function may return false positives for non 3D cells as the
115  // tests are done on the projection of the point on the cell. Extra checks
116  // should be added to test if the point actually falls on the cell.
117  template <typename CellShapeTag, typename CoordsType>
119  CellShapeTag cellShape,
120  CoordsType cellPoints,
121  FloatVec3& parametricCoordinates,
122  bool& inside)
123  {
124  auto bounds = vtkm::internal::cl_uniform_bins::ComputeCellBounds(cellPoints);
125  if (point[0] >= bounds.Min[0] && point[0] <= bounds.Max[0] && point[1] >= bounds.Min[1] &&
126  point[1] <= bounds.Max[1] && point[2] >= bounds.Min[2] && point[2] <= bounds.Max[2])
127  {
128  VTKM_RETURN_ON_ERROR(vtkm::exec::WorldCoordinatesToParametricCoordinates(
129  cellPoints, point, cellShape, parametricCoordinates));
130  inside = vtkm::exec::CellInside(parametricCoordinates, cellShape);
131  }
132  else
133  {
134  inside = false;
135  }
136  // Return success error code even point is not inside this cell
138  }
139 
140 public:
141  template <typename CellSetType>
142  VTKM_CONT CellLocatorTwoLevel(const vtkm::internal::cl_uniform_bins::Grid& topLevelGrid,
143  const vtkm::cont::ArrayHandle<DimVec3>& leafDimensions,
144  const vtkm::cont::ArrayHandle<vtkm::Id>& leafStartIndex,
145  const vtkm::cont::ArrayHandle<vtkm::Id>& cellStartIndex,
146  const vtkm::cont::ArrayHandle<vtkm::Id>& cellCount,
147  const vtkm::cont::ArrayHandle<vtkm::Id>& cellIds,
148  const CellSetType& cellSet,
149  const vtkm::cont::CoordinateSystem& coords,
151  vtkm::cont::Token& token)
152  : TopLevel(topLevelGrid)
153  , LeafDimensions(leafDimensions.PrepareForInput(device, token))
154  , LeafStartIndex(leafStartIndex.PrepareForInput(device, token))
155  , CellStartIndex(cellStartIndex.PrepareForInput(device, token))
156  , CellCount(cellCount.PrepareForInput(device, token))
157  , CellIds(cellIds.PrepareForInput(device, token))
158  , CellSet(cellSet.PrepareForInput(device,
161  token))
162  , Coords(coords.GetDataAsMultiplexer().PrepareForInput(device, token))
163  {
164  }
165 
167  struct LastCell
168  {
169  vtkm::Id CellId = -1;
170  vtkm::Id LeafIdx = -1;
171  };
172 
174  VTKM_EXEC
175  vtkm::ErrorCode FindCell(const FloatVec3& point, vtkm::Id& cellId, FloatVec3& parametric) const
176  {
177  LastCell lastCell;
178  return this->FindCellImpl(point, cellId, parametric, lastCell);
179  }
180 
182  VTKM_EXEC
184  vtkm::Id& cellId,
185  FloatVec3& parametric,
186  LastCell& lastCell) const
187  {
188  vtkm::Vec3f pc;
189  //See if point is inside the last cell.
190  if ((lastCell.CellId >= 0) && (lastCell.CellId < this->CellSet.GetNumberOfElements()) &&
191  this->PointInCell(point, lastCell.CellId, pc) == vtkm::ErrorCode::Success)
192  {
193  parametric = pc;
194  cellId = lastCell.CellId;
196  }
197 
198  //See if it's in the last leaf.
199  if ((lastCell.LeafIdx >= 0) && (lastCell.LeafIdx < this->CellCount.GetNumberOfValues()) &&
200  this->PointInLeaf(point, lastCell.LeafIdx, cellId, pc) == vtkm::ErrorCode::Success)
201  {
202  parametric = pc;
203  lastCell.CellId = cellId;
205  }
206 
207  //Call the full point search.
208  return this->FindCellImpl(point, cellId, parametric, lastCell);
209  }
210 
211 private:
212  VTKM_EXEC
214  const vtkm::Id& cid,
215  vtkm::Vec3f& parametric) const
216  {
217  auto indices = this->CellSet.GetIndices(cid);
218  auto pts = vtkm::make_VecFromPortalPermute(&indices, this->Coords);
219  vtkm::Vec3f pc;
220  bool inside;
221  auto status = PointInsideCell(point, this->CellSet.GetCellShape(cid), pts, pc, inside);
222  if (status == vtkm::ErrorCode::Success && inside)
223  {
224  parametric = pc;
226  }
227 
229  }
230 
231  VTKM_EXEC
233  const vtkm::Id& leafIdx,
234  vtkm::Id& cellId,
235  FloatVec3& parametric) const
236  {
237  vtkm::Id start = this->CellStartIndex.Get(leafIdx);
238  vtkm::Id end = start + this->CellCount.Get(leafIdx);
239 
240  for (vtkm::Id i = start; i < end; ++i)
241  {
242  vtkm::Vec3f pc;
243 
244  vtkm::Id cid = this->CellIds.Get(i);
245  if (this->PointInCell(point, cid, pc) == vtkm::ErrorCode::Success)
246  {
247  cellId = cid;
248  parametric = pc;
250  }
251  }
252 
254  }
255 
256 
257  VTKM_EXEC
259  vtkm::Id& cellId,
260  FloatVec3& parametric,
261  LastCell& lastCell) const
262  {
263  using namespace vtkm::internal::cl_uniform_bins;
264 
265  cellId = -1;
266  lastCell.CellId = -1;
267  lastCell.LeafIdx = -1;
268 
269  DimVec3 binId3 = static_cast<DimVec3>((point - this->TopLevel.Origin) / this->TopLevel.BinSize);
270  if (binId3[0] >= 0 && binId3[0] < this->TopLevel.Dimensions[0] && binId3[1] >= 0 &&
271  binId3[1] < this->TopLevel.Dimensions[1] && binId3[2] >= 0 &&
272  binId3[2] < this->TopLevel.Dimensions[2])
273  {
274  vtkm::Id binId = ComputeFlatIndex(binId3, this->TopLevel.Dimensions);
275 
276  auto ldim = this->LeafDimensions.Get(binId);
277  if (!ldim[0] || !ldim[1] || !ldim[2])
278  {
280  }
281 
282  auto leafGrid = ComputeLeafGrid(binId3, ldim, this->TopLevel);
283 
284  DimVec3 leafId3 = static_cast<DimVec3>((point - leafGrid.Origin) / leafGrid.BinSize);
285  // precision issues may cause leafId3 to be out of range so clamp it
286  leafId3 = vtkm::Max(DimVec3(0), vtkm::Min(ldim - DimVec3(1), leafId3));
287 
288  vtkm::Id leafStart = this->LeafStartIndex.Get(binId);
289  vtkm::Id leafIdx = leafStart + ComputeFlatIndex(leafId3, leafGrid.Dimensions);
290 
291  if (this->PointInLeaf(point, leafIdx, cellId, parametric) == vtkm::ErrorCode::Success)
292  {
293  lastCell.CellId = cellId;
294  lastCell.LeafIdx = leafIdx;
296  }
297  }
298 
300  }
301 
302  vtkm::internal::cl_uniform_bins::Grid TopLevel;
303 
306 
310 
311  CellStructureType CellSet;
313 };
314 }
315 } // vtkm::exec
316 
317 #endif //vtk_m_exec_CellLocatorTwoLevel_h
vtkm::TopologyElementTagPoint
A tag used to identify the point elements in a topology.
Definition: TopologyElementTag.h:34
vtkm::exec::CellLocatorTwoLevel
Structure for locating cells.
Definition: exec/CellLocatorTwoLevel.h:102
vtkm::cont::ArrayHandle
Manages an array-worth of data.
Definition: ArrayHandle.h:300
ArrayHandle.h
vtkm::ErrorCode
ErrorCode
Identifies whether an operation was successful or what type of error it had.
Definition: ErrorCode.h:28
vtkm::exec::CellLocatorTwoLevel::Coords
CoordsPortalType Coords
Definition: exec/CellLocatorTwoLevel.h:312
vtkm::exec::CellLocatorTwoLevel::CoordsPortalType
typename vtkm::cont::CoordinateSystem::MultiplexerArrayType::ReadPortalType CoordsPortalType
Definition: exec/CellLocatorTwoLevel.h:112
VTKM_EXEC
#define VTKM_EXEC
Definition: ExportMacros.h:51
vtkm
Groups connected points that have the same field value.
Definition: Atomic.h:19
Types.h
vtkm::exec::CellLocatorTwoLevel::CellSet
CellStructureType CellSet
Definition: exec/CellLocatorTwoLevel.h:311
vtkm::IdComponent
vtkm::Int32 IdComponent
Base type to use to index small lists.
Definition: Types.h:194
vtkm::exec::CellLocatorTwoLevel::CellLocatorTwoLevel
CellLocatorTwoLevel(const vtkm::internal::cl_uniform_bins::Grid &topLevelGrid, const vtkm::cont::ArrayHandle< DimVec3 > &leafDimensions, const vtkm::cont::ArrayHandle< vtkm::Id > &leafStartIndex, const vtkm::cont::ArrayHandle< vtkm::Id > &cellStartIndex, const vtkm::cont::ArrayHandle< vtkm::Id > &cellCount, const vtkm::cont::ArrayHandle< vtkm::Id > &cellIds, const CellSetType &cellSet, const vtkm::cont::CoordinateSystem &coords, vtkm::cont::DeviceAdapterId device, vtkm::cont::Token &token)
Definition: exec/CellLocatorTwoLevel.h:142
vtkm::exec::CellLocatorTwoLevel::LeafDimensions
ReadPortal< DimVec3 > LeafDimensions
Definition: exec/CellLocatorTwoLevel.h:304
vtkm::exec::CellLocatorTwoLevel::CellStartIndex
ReadPortal< vtkm::Id > CellStartIndex
Definition: exec/CellLocatorTwoLevel.h:307
vtkm::ErrorCode::Success
@ Success
A successful operation.
vtkm::VecTraits::ComponentType
T ComponentType
Type of the components in the vector.
Definition: VecTraits.h:71
vtkm::Int16
int16_t Int16
Base type to use for 16-bit signed integer numbers.
Definition: Types.h:173
vtkm::exec::CellLocatorTwoLevel::LastCell::CellId
vtkm::Id CellId
Definition: exec/CellLocatorTwoLevel.h:169
vtkm::exec::CellLocatorTwoLevel::LastCell::LeafIdx
vtkm::Id LeafIdx
Definition: exec/CellLocatorTwoLevel.h:170
CoordinateSystem.h
vtkm::cont::ArrayHandle::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::CellLocatorTwoLevel::ReadPortal
typename vtkm::cont::ArrayHandle< T >::ReadPortalType ReadPortal
Definition: exec/CellLocatorTwoLevel.h:109
vtkm::cont::CoordinateSystem
Manages a coordinate system for a DataSet.
Definition: CoordinateSystem.h:30
vtkm::cont::Token
A token to hold the scope of an ArrayHandle or other object.
Definition: Token.h:35
vtkm::exec::CellLocatorTwoLevel::FindCell
vtkm::ErrorCode FindCell(const FloatVec3 &point, vtkm::Id &cellId, FloatVec3 &parametric) const
Locate the cell containing the provided point.
Definition: exec/CellLocatorTwoLevel.h:175
vtkm::VecTraits::GetNumberOfComponents
static constexpr vtkm::IdComponent GetNumberOfComponents(const T &)
Returns the number of components in the given vector.
Definition: VecTraits.h:94
CellInside.h
Math.h
vtkm::exec::CellLocatorTwoLevel::FindCell
vtkm::ErrorCode FindCell(const FloatVec3 &point, vtkm::Id &cellId, FloatVec3 &parametric, LastCell &lastCell) const
Locate the cell containing the provided point.
Definition: exec/CellLocatorTwoLevel.h:183
vtkm::make_VecFromPortalPermute
VecFromPortalPermute< IndexVecType, PortalType > make_VecFromPortalPermute(const IndexVecType *index, const PortalType &portal)
Definition: VecFromPortalPermute.h:166
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::CellLocatorTwoLevel::CellCount
ReadPortal< vtkm::Id > CellCount
Definition: exec/CellLocatorTwoLevel.h:308
vtkm::ErrorCode::CellNotFound
@ CellNotFound
A cell matching some given criteria could not be found.
vtkm::Vec3f
vtkm::Vec< vtkm::FloatDefault, 3 > Vec3f
Vec3f corresponds to a 3-dimensional vector of floating point values.
Definition: Types.h:1055
vtkm::cont::DeviceAdapterId
An object used to specify a device.
Definition: DeviceAdapterTag.h:58
vtkm::exec::CellLocatorTwoLevel::PointInCell
vtkm::ErrorCode PointInCell(const vtkm::Vec3f &point, const vtkm::Id &cid, vtkm::Vec3f &parametric) const
Definition: exec/CellLocatorTwoLevel.h:213
vtkm::exec::CellLocatorTwoLevel::FindCellImpl
vtkm::ErrorCode FindCellImpl(const FloatVec3 &point, vtkm::Id &cellId, FloatVec3 &parametric, LastCell &lastCell) const
Definition: exec/CellLocatorTwoLevel.h:258
vtkm::Vec< DimensionType, 3 >
vtkm::exec::CellLocatorTwoLevel::PointInsideCell
static vtkm::ErrorCode PointInsideCell(FloatVec3 point, CellShapeTag cellShape, CoordsType cellPoints, FloatVec3 &parametricCoordinates, bool &inside)
Definition: exec/CellLocatorTwoLevel.h:118
vtkm::exec::CellLocatorTwoLevel::PointInLeaf
vtkm::ErrorCode PointInLeaf(const FloatVec3 &point, const vtkm::Id &leafIdx, vtkm::Id &cellId, FloatVec3 &parametric) const
Definition: exec/CellLocatorTwoLevel.h:232
VTKM_RETURN_ON_ERROR
#define VTKM_RETURN_ON_ERROR(call)
Definition: ErrorCode.h:202
vtkm::exec::CellLocatorTwoLevel::TopLevel
vtkm::internal::cl_uniform_bins::Grid TopLevel
Definition: exec/CellLocatorTwoLevel.h:302
vtkm::TopologyElementTagCell
A tag used to identify the cell elements in a topology.
Definition: TopologyElementTag.h:24
vtkm::exec::CellLocatorTwoLevel::LeafStartIndex
ReadPortal< vtkm::Id > LeafStartIndex
Definition: exec/CellLocatorTwoLevel.h:305
VTKM_ALWAYS_EXPORT
#define VTKM_ALWAYS_EXPORT
Definition: ExportMacros.h:89
vtkm::exec::CellLocatorTwoLevel::CellIds
ReadPortal< vtkm::Id > CellIds
Definition: exec/CellLocatorTwoLevel.h:309
vtkm::exec::CellLocatorTwoLevel::LastCell
Structure capturing the location of a cell in the search structure.
Definition: exec/CellLocatorTwoLevel.h:167
ParametricCoordinates.h
VecTraits.h
TopologyElementTag.h