NEMoSys  0.63.0
A modular, extensible resource with robust automated mesh generation, mesh quality analysis, adaptive mesh refinement, and data transfer between arbitrary meshes.
cobalt.C
Go to the documentation of this file.
1 /*******************************************************************************
2 * Promesh *
3 * Copyright (C) 2022, IllinoisRocstar LLC. All rights reserved. *
4 * *
5 * Promesh is the property of IllinoisRocstar LLC. *
6 * *
7 * IllinoisRocstar LLC *
8 * Champaign, IL *
9 * www.illinoisrocstar.com *
10 * promesh@illinoisrocstar.com *
11 *******************************************************************************/
12 /*******************************************************************************
13 * This file is part of Promesh *
14 * *
15 * This version of Promesh is free software: you can redistribute it and/or *
16 * modify it under the terms of the GNU Lesser General Public License as *
17 * published by the Free Software Foundation, either version 3 of the License, *
18 * or (at your option) any later version. *
19 * *
20 * Promesh is distributed in the hope that it will be useful, but WITHOUT ANY *
21 * WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS *
22 * FOR A PARTICULAR PURPOSE. See the GNU Lesser General Public License for more *
23 * details. *
24 * *
25 * You should have received a copy of the GNU Lesser General Public License *
26 * along with this program. If not, see <https://www.gnu.org/licenses/>. *
27 * *
28 *******************************************************************************/
29 #include "Mesh/cobalt.H"
30 
31 // VTK
32 #include <vtkCellTypes.h>
33 #include <vtkCellData.h>
34 #include <vtkThreshold.h>
35 #include <vtkGeometryFilter.h>
36 #include <vtkGenericCell.h>
37 #include <vtkUnstructuredGrid.h>
38 
39 #include "AuxiliaryFunctions.H"
40 
41 using nemAux::operator-; // for vector subtraction.
42 
43 //////////////////////////////////
44 // cobalt class
45 //////////////////////////////////
46 
47 // constructor from meshBase object
48 COBALT::cobalt::cobalt(const std::shared_ptr<meshBase> fullMesh,
49  const std::string &_inFnameVtk,
50  const std::string &_outFnameCgr,
51  const std::string &_outFnameCgi)
52 {
53  inFnameVtk = _inFnameVtk; // vtk input file
54  outFnameCgr = _outFnameCgr; // cobalt output grid file
55  outFnameCgi = _outFnameCgi; // cobalt output patch mapping file
56 
57  // declare array to store element type
58  vtkSmartPointer<vtkDataArray> elementTypeArray
60  elementTypeArray->SetNumberOfComponents(1);
61  elementTypeArray->SetName("elemType");
62 
63  // get element types
64  double tmp[1]; // storing inserted tuples
65  for (int iCell = 0;
66  iCell < fullMesh->getDataSet()->GetNumberOfCells(); iCell++)
67  {
68  if (fullMesh->getDataSet()->GetCellType(iCell) != VTK_TETRA
69  && fullMesh->getDataSet()->GetCellType(iCell) != VTK_TRIANGLE)
70  {
71  std::cout << "Error: only triangle and tetrahedral elements supported."
72  << std::endl;
73  exit(1);
74  }
75  tmp[0] = fullMesh->getDataSet()->GetCellType(iCell);
76  elementTypeArray->InsertNextTuple(tmp);
77  }
78  fullMesh->getDataSet()->GetCellData()->AddArray(elementTypeArray);
79 
80  // threshold to extract only volume cells
81  vtkSmartPointer<vtkThreshold> volThreshold = vtkSmartPointer<vtkThreshold>::New();
82  volThreshold->SetInputData(fullMesh->getDataSet());
83  volThreshold->SetInputArrayToProcess(0, 0, 0,
84  vtkDataObject::FIELD_ASSOCIATION_CELLS,
85  "elemType");
86  volThreshold->ThresholdBetween(VTK_TETRA, VTK_TETRA);
87  volThreshold->Update();
88  // store output in unstructured grid
89  vtkSmartPointer<vtkUnstructuredGrid> volUG = volThreshold->GetOutput();
90  // create meshBase object
91  volMeshBase = meshBase::CreateShared(volUG, "extractedVolume.vtu");
92  //volUG_mb->write(); // write to file
93 
94  // threshold to extract only surface cells
95  vtkSmartPointer<vtkThreshold> surfThreshold = vtkSmartPointer<vtkThreshold>::New();
96  surfThreshold->SetInputData(fullMesh->getDataSet());
97  surfThreshold->SetInputArrayToProcess(0, 0, 0,
98  vtkDataObject::FIELD_ASSOCIATION_CELLS,
99  "elemType");
100  surfThreshold->ThresholdBetween(VTK_TRIANGLE, VTK_TRIANGLE);
101  surfThreshold->Update();
102  // store output in unstructured grid
103  vtkSmartPointer<vtkUnstructuredGrid> surfUG = surfThreshold->GetOutput();
104  // geometry filter to polyData object
105  vtkSmartPointer<vtkGeometryFilter> geomFilter = vtkGeometryFilter::New();
106  geomFilter->SetInputData(surfUG);
107  geomFilter->Update();
108  vtkSmartPointer<vtkPolyData> surfPD = geomFilter->GetOutput();
109  // create meshBase object
110  surfMeshBase = meshBase::CreateShared(surfPD, "extractedSurface.vtp");
111  //surfPD_mb->write(); // write to file
112 }
113 
114 // writes COBALT mesh data into file
116 {
117  std::ofstream outputStream(outFnameCgr);
118  if (!outputStream.good())
119  {
120  std::cout << "Cannot open file " << outFnameCgr << std::endl;
121  exit(1);
122  }
123 
124  if (!surfMeshBase)
125  {
126  std::cout << "surface mesh is empty!" << std::endl;
127  exit(1);
128  }
129  auto surfPatchNoArr =
130  surfMeshBase->getDataSet()->GetCellData()->GetArray("patchNo");
131  if (!surfPatchNoArr) {
132  surfPatchNoArr =
133  surfMeshBase->getDataSet()->GetCellData()->GetArray("PhysGrpId");
134  if (!surfPatchNoArr) {
135  std::cout << "surface mesh must have cell array with name \"patchNo\" or "
136  "\"PhysGrpId\" denoting patch number."
137  << std::endl;
138  exit(1);
139  }
140  }
141 
142  vtkSmartPointer<vtkIdList> facePtIds;
143  vtkSmartPointer<vtkIdList> sharedCellPtIds = vtkSmartPointer<vtkIdList>::New();
144  vtkSmartPointer<vtkGenericCell> genCell1 = vtkSmartPointer<vtkGenericCell>::New();
145  vtkSmartPointer<vtkGenericCell> genCell2 = vtkSmartPointer<vtkGenericCell>::New();
146  // Signed to represent patch numbers as negative integers
147  std::map<std::vector<nemId_t>,
148  std::pair<nemId_t, long long int>,
149  sortNemId_tVec_compare> faceMap;
150  // building cell locator for looking up patch number in remeshed surface mesh
151  vtkSmartPointer<vtkStaticCellLocator> surfCellLocator = surfMeshBase->buildStaticCellLocator();
152  // maximum number of vertices per face (to be found in proceeding loop)
153  vtkIdType nVerticesPerFaceMax = 0;
154  // maximum number of faces per cell (to be found in proceeding loop)
155  int nFacesPerCellMax = 0;
156 
157  for (vtkIdType i = 0; i < volMeshBase->getNumberOfCells(); ++i)
158  {
159  // get cell i
160  volMeshBase->getDataSet()->GetCell(i, genCell1);
161  // get faces, find cells sharing it. if no cell shares it,
162  // use the locator of the surfMeshBase to find the patch number
163  int numFaces = genCell1->GetNumberOfFaces();
164  nFacesPerCellMax = (nFacesPerCellMax < numFaces
165  ? numFaces
166  : nFacesPerCellMax);
167  for (int j = 0; j < numFaces; ++j)
168  {
169  vtkCell *face = genCell1->GetFace(j);
170  // bool shared = false;
171  vtkIdType numVerts = face->GetNumberOfPoints();
172  nVerticesPerFaceMax = (nVerticesPerFaceMax < numVerts
173  ? numVerts
174  : nVerticesPerFaceMax);
175  facePtIds = face->GetPointIds();
176  volMeshBase->getDataSet()->GetCellNeighbors(i, facePtIds,
177  sharedCellPtIds);
178  std::vector<nemId_t> facePntIds(numVerts);
179  for (vtkIdType k = 0; k < numVerts; ++k)
180  {
181  facePntIds[k] = face->GetPointId(k) + 1;
182  }
183  if (sharedCellPtIds->GetNumberOfIds())
184  {
185  faceMap.emplace(facePntIds, std::pair<nemId_t, long long int>(
186  i + 1, sharedCellPtIds->GetId(0) + 1));
187  }
188  else
189  {
190  double bounds[6];
191  face->GetBounds(bounds);
192  auto cellIds = vtkSmartPointer<vtkIdList>::New();
193  surfCellLocator->FindCellsWithinBounds(bounds, cellIds);
194  vtkIdType closestCellId;
195  if (cellIds->GetNumberOfIds() == 0) {
196  std::cerr << "No surface cell found." << std::endl;
197  continue;
198  } else {
199  closestCellId = cellIds->GetId(0);
200  if (cellIds->GetNumberOfIds() > 1) {
201  double p1[3], p2[3], p3[3];
202  face->GetPoints()->GetPoint(0, p1);
203  face->GetPoints()->GetPoint(1, p2);
204  face->GetPoints()->GetPoint(2, p3);
205  std::vector<double> faceCenter(3);
206  for (vtkIdType k = 0; k < numVerts; ++k) {
207  faceCenter[k] = (p1[k] + p2[k] + p3[k]) / 3.0;
208  }
209  double minDist = nemAux::l2_Norm(
210  faceCenter - surfMeshBase->getCellCenter(cellIds->GetId(0)));
211  for (vtkIdType k = 1; k < cellIds->GetNumberOfIds(); ++k) {
212  auto candidateCellCenter =
213  surfMeshBase->getCellCenter(cellIds->GetId(k));
214  auto dist = nemAux::l2_Norm(faceCenter - candidateCellCenter);
215  if (dist < minDist) {
216  minDist = dist;
217  closestCellId = cellIds->GetId(k);
218  }
219  }
220  }
221  }
222  double patchNo = surfPatchNoArr->GetComponent(closestCellId, 0);
223  faceMap.emplace(facePntIds,
224  std::pair<nemId_t, long long int>(i + 1, -1 * patchNo));
225  }
226  }
227  }
228 
229  std::map<nemId_t, nemId_t> patchMap;
230  for (nemId_t i = 0; i < surfMeshBase->getNumberOfCells(); ++i)
231  {
232  double patchNo[1];
233  surfPatchNoArr->GetTuple(i, patchNo);
234  patchMap.insert(std::pair<nemId_t, nemId_t>(patchNo[0], i));
235  }
236 
237  // write patch mapping file
238  writePatchMap(outFnameCgi, patchMap);
239  // write cobalt file
240  outputStream << 3 << " " << 1 << " " << patchMap.size() << std::endl;
241  outputStream << volMeshBase->getNumberOfPoints() << " " << faceMap.size()
242  << " " << volMeshBase->getNumberOfCells() << " "
243  << nVerticesPerFaceMax << " " << nFacesPerCellMax << std::endl;
244  for (nemId_t i = 0; i < volMeshBase->getNumberOfPoints(); ++i)
245  {
246  std::vector<double> pnt(volMeshBase->getPoint(i));
247  outputStream << std::setw(21) << std::fixed << std::setprecision(15)
248  << pnt[0] << " " << pnt[1] << " " << pnt[2] << std::endl;
249  }
250 
251  auto it = faceMap.begin();
252  while (it != faceMap.end())
253  {
254  outputStream << it->first.size() << " ";
255  auto faceIdIter = it->first.begin();
256  while (faceIdIter != it->first.end())
257  {
258  outputStream << *faceIdIter << " ";
259  ++faceIdIter;
260  }
261  outputStream << it->second.first << " " << it->second.second << std::endl;
262  ++it;
263  }
264 }
265 
266 
267 void COBALT::cobalt::writePatchMap(const std::string &mapFile,
268  const std::map<nemId_t, nemId_t> &patchMap) const
269 {
270  std::ofstream outputStream(mapFile);
271  if (!outputStream.good())
272  {
273  std::cout << "Error opening file " << mapFile << std::endl;
274  exit(1);
275  }
276  writePatchMap(outputStream, patchMap);
277 }
278 
279 
280 void COBALT::cobalt::writePatchMap(std::ofstream &outputStream,
281  const std::map<nemId_t, nemId_t> &patchMap) const
282 {
283  outputStream << patchMap.size() << std::endl;
284  outputStream << patchMap.size() << std::endl;
285  auto it = patchMap.begin();
286  size_t normPatchNo = 1;
287  while (it != patchMap.end())
288  {
289  for (int i = 0; i < 2; ++i)
290  {
291  outputStream << std::setw(2) << std::left << it->first << " ";
292  }
293  outputStream << std::setw(2) << std::left << normPatchNo << " ";
294  outputStream << std::endl;
295  ++it;
296  normPatchNo++;
297  }
298 }
std::shared_ptr< meshBase > volMeshBase
Definition: cobalt.H:53
std::string outFnameCgr
Definition: cobalt.H:56
std::string inFnameVtk
Definition: cobalt.H:55
std::size_t nemId_t
Definition: meshBase.H:51
void writePatchMap(const std::string &mapFile, const std::map< nemId_t, nemId_t > &patchMap) const
Definition: cobalt.C:267
geoMeshBase * New(MeshType meshType)
Create a new mesh object.
sum comparison for vectors representing faces inserted into map
Definition: meshBase.H:765
static std::shared_ptr< meshBase > CreateShared(const std::string &fname)
Create shared ptr from fname.
Definition: meshBase.C:171
void write() const
Definition: cobalt.C:115
cobalt(std::shared_ptr< meshBase > fullMesh, const std::string &inFnameVtk, const std::string &outFnameCgr, const std::string &outFnameCgi)
Definition: cobalt.C:48
std::string outFnameCgi
Definition: cobalt.H:57
std::shared_ptr< meshBase > surfMeshBase
Definition: cobalt.H:54
T l2_Norm(const std::vector< T > &x)