All Classes Namespaces Files Functions Variables Typedefs Enumerations Enumerator Friends Macros Pages
probes.C
Go to the documentation of this file.
1 /*---------------------------------------------------------------------------*\
2  ========= |
3  \\ / F ield | OpenFOAM: The Open Source CFD Toolbox
4  \\ / O peration | Website: https://openfoam.org
5  \\ / A nd | Copyright (C) 2011-2018 OpenFOAM Foundation
6  \\/ M anipulation |
7 -------------------------------------------------------------------------------
8 License
9  This file is part of OpenFOAM.
10 
11  OpenFOAM is free software: you can redistribute it and/or modify it
12  under the terms of the GNU General Public License as published by
13  the Free Software Foundation, either version 3 of the License, or
14  (at your option) any later version.
15 
16  OpenFOAM is distributed in the hope that it will be useful, but WITHOUT
17  ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
18  FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
19  for more details.
20 
21  You should have received a copy of the GNU General Public License
22  along with OpenFOAM. If not, see <http://www.gnu.org/licenses/>.
23 
24 \*---------------------------------------------------------------------------*/
25 
26 #include "probes.H"
27 #include "volFields.H"
28 #include "dictionary.H"
29 #include "Time.H"
30 #include "IOmanip.H"
31 #include "mapPolyMesh.H"
33 
34 // * * * * * * * * * * * * * * Static Data Members * * * * * * * * * * * * * //
35 
36 namespace Foam
37 {
38  defineTypeNameAndDebug(probes, 0);
39 
41  (
42  functionObject,
43  probes,
44  dictionary
45  );
46 }
47 
48 
49 // * * * * * * * * * * * * Protected Member Functions * * * * * * * * * * * //
50 
52 {
53  if (debug)
54  {
55  Info<< "probes: resetting sample locations" << endl;
56  }
57 
58  elementList_.clear();
59  elementList_.setSize(size());
60 
61  faceList_.clear();
62  faceList_.setSize(size());
63 
64  forAll(*this, probei)
65  {
66  const vector& location = operator[](probei);
67 
68  const label celli = mesh.findCell(location);
69 
70  elementList_[probei] = celli;
71 
72  if (celli != -1)
73  {
74  const labelList& cellFaces = mesh.cells()[celli];
75  const vector& cellCentre = mesh.cellCentres()[celli];
76  scalar minDistance = great;
77  label minFaceID = -1;
78  forAll(cellFaces, i)
79  {
80  label facei = cellFaces[i];
81  vector dist = mesh.faceCentres()[facei] - cellCentre;
82  if (mag(dist) < minDistance)
83  {
84  minDistance = mag(dist);
85  minFaceID = facei;
86  }
87  }
88  faceList_[probei] = minFaceID;
89  }
90  else
91  {
92  faceList_[probei] = -1;
93  }
94 
95  if (debug && (elementList_[probei] != -1 || faceList_[probei] != -1))
96  {
97  Pout<< "probes : found point " << location
98  << " in cell " << elementList_[probei]
99  << " and face " << faceList_[probei] << endl;
100  }
101  }
102 
103 
104  // Check if all probes have been found.
105  forAll(elementList_, probei)
106  {
107  const vector& location = operator[](probei);
108  label celli = elementList_[probei];
109  label facei = faceList_[probei];
110 
111  // Check at least one processor with cell.
112  reduce(celli, maxOp<label>());
113  reduce(facei, maxOp<label>());
114 
115  if (celli == -1)
116  {
117  if (Pstream::master())
118  {
120  << "Did not find location " << location
121  << " in any cell. Skipping location." << endl;
122  }
123  }
124  else if (facei == -1)
125  {
126  if (Pstream::master())
127  {
129  << "Did not find location " << location
130  << " in any face. Skipping location." << endl;
131  }
132  }
133  else
134  {
135  // Make sure location not on two domains.
136  if (elementList_[probei] != -1 && elementList_[probei] != celli)
137  {
139  << "Location " << location
140  << " seems to be on multiple domains:"
141  << " cell " << elementList_[probei]
142  << " on my domain " << Pstream::myProcNo()
143  << " and cell " << celli << " on some other domain."
144  << endl
145  << "This might happen if the probe location is on"
146  << " a processor patch. Change the location slightly"
147  << " to prevent this." << endl;
148  }
149 
150  if (faceList_[probei] != -1 && faceList_[probei] != facei)
151  {
153  << "Location " << location
154  << " seems to be on multiple domains:"
155  << " cell " << faceList_[probei]
156  << " on my domain " << Pstream::myProcNo()
157  << " and face " << facei << " on some other domain."
158  << endl
159  << "This might happen if the probe location is on"
160  << " a processor patch. Change the location slightly"
161  << " to prevent this." << endl;
162  }
163  }
164  }
165 }
166 
167 
169 {
170  const label nFields = classifyFields();
171 
172  // adjust file streams
173  if (Pstream::master())
174  {
175  wordHashSet currentFields;
176 
177  currentFields.insert(scalarFields_);
178  currentFields.insert(vectorFields_);
179  currentFields.insert(sphericalTensorFields_);
180  currentFields.insert(symmTensorFields_);
181  currentFields.insert(tensorFields_);
182 
183  currentFields.insert(surfaceScalarFields_);
184  currentFields.insert(surfaceVectorFields_);
185  currentFields.insert(surfaceSphericalTensorFields_);
186  currentFields.insert(surfaceSymmTensorFields_);
187  currentFields.insert(surfaceTensorFields_);
188 
189  if (debug)
190  {
191  Info<< "Probing fields: " << currentFields << nl
192  << "Probing locations: " << *this << nl
193  << endl;
194  }
195 
196 
197  fileName probeDir;
198  fileName probeSubDir = name();
199 
200  if (mesh_.name() != polyMesh::defaultRegion)
201  {
202  probeSubDir = probeSubDir/mesh_.name();
203  }
204  probeSubDir = "postProcessing"/probeSubDir/mesh_.time().timeName();
205 
206  if (Pstream::parRun())
207  {
208  // Put in undecomposed case
209  // (Note: gives problems for distributed data running)
210  probeDir = mesh_.time().path()/".."/probeSubDir;
211  }
212  else
213  {
214  probeDir = mesh_.time().path()/probeSubDir;
215  }
216  // Remove ".."
217  probeDir.clean();
218 
219  // ignore known fields, close streams for fields that no longer exist
220  forAllIter(HashPtrTable<OFstream>, probeFilePtrs_, iter)
221  {
222  if (!currentFields.erase(iter.key()))
223  {
224  if (debug)
225  {
226  Info<< "close probe stream: " << iter()->name() << endl;
227  }
228 
229  delete probeFilePtrs_.remove(iter);
230  }
231  }
232 
233  // currentFields now just has the new fields - open streams for them
234  forAllConstIter(wordHashSet, currentFields, iter)
235  {
236  const word& fieldName = iter.key();
237 
238  // Create directory if does not exist.
239  mkDir(probeDir);
240 
241  OFstream* fPtr = new OFstream(probeDir/fieldName);
242 
243  OFstream& fout = *fPtr;
244 
245  if (debug)
246  {
247  Info<< "open probe stream: " << fout.name() << endl;
248  }
249 
250  probeFilePtrs_.insert(fieldName, fPtr);
251 
252  unsigned int w = IOstream::defaultPrecision() + 7;
253 
254  forAll(*this, probei)
255  {
256  fout<< "# Probe " << probei << ' ' << operator[](probei)
257  << endl;
258  }
259 
260  fout<< '#' << setw(IOstream::defaultPrecision() + 6)
261  << "Probe";
262 
263  forAll(*this, probei)
264  {
265  fout<< ' ' << setw(w) << probei;
266  }
267  fout<< endl;
268 
269  fout<< '#' << setw(IOstream::defaultPrecision() + 6)
270  << "Time" << endl;
271  }
272  }
273 
274  return nFields;
275 }
276 
277 
278 // * * * * * * * * * * * * * * * * Constructors * * * * * * * * * * * * * * //
279 
281 (
282  const word& name,
283  const Time& t,
284  const dictionary& dict
285 )
286 :
287  functionObject(name),
288  pointField(0),
289  mesh_
290  (
291  refCast<const fvMesh>
292  (
294  (
296  )
297  )
298  ),
299  loadFromFiles_(false),
300  fieldSelection_(),
301  fixedLocations_(true),
302  interpolationScheme_("cell")
303 {
304  read(dict);
305 }
306 
307 
309 (
310  const word& name,
311  const objectRegistry& obr,
312  const dictionary& dict,
313  const bool loadFromFiles
314 )
315 :
316  functionObject(name),
317  pointField(0),
318  mesh_(refCast<const fvMesh>(obr)),
319  loadFromFiles_(loadFromFiles),
320  fieldSelection_(),
321  fixedLocations_(true),
322  interpolationScheme_("cell")
323 {
324  read(dict);
325 }
326 
327 
328 // * * * * * * * * * * * * * * * * Destructor * * * * * * * * * * * * * * * //
329 
331 {}
332 
333 
334 // * * * * * * * * * * * * * * * Member Functions * * * * * * * * * * * * * //
335 
337 {
338  dict.lookup("probeLocations") >> *this;
339  dict.lookup("fields") >> fieldSelection_;
340 
341  dict.readIfPresent("fixedLocations", fixedLocations_);
342  if (dict.readIfPresent("interpolationScheme", interpolationScheme_))
343  {
344  if (!fixedLocations_ && interpolationScheme_ != "cell")
345  {
347  << "Only cell interpolation can be applied when "
348  << "not using fixedLocations. InterpolationScheme "
349  << "entry will be ignored";
350  }
351  }
352 
353  // Initialise cells to sample from supplied locations
354  findElements(mesh_);
355 
356  prepare();
357 
358  return true;
359 }
360 
361 
363 {
364  return true;
365 }
366 
367 
369 {
370  if (size() && prepare())
371  {
372  sampleAndWrite(scalarFields_);
373  sampleAndWrite(vectorFields_);
374  sampleAndWrite(sphericalTensorFields_);
375  sampleAndWrite(symmTensorFields_);
376  sampleAndWrite(tensorFields_);
377 
378  sampleAndWriteSurfaceFields(surfaceScalarFields_);
379  sampleAndWriteSurfaceFields(surfaceVectorFields_);
380  sampleAndWriteSurfaceFields(surfaceSphericalTensorFields_);
381  sampleAndWriteSurfaceFields(surfaceSymmTensorFields_);
382  sampleAndWriteSurfaceFields(surfaceTensorFields_);
383  }
384 
385  return true;
386 }
387 
388 
390 {
391  DebugInfo<< "probes: updateMesh" << endl;
392 
393  if (&mpm.mesh() != &mesh_)
394  {
395  return;
396  }
397 
398  if (fixedLocations_)
399  {
400  findElements(mesh_);
401  }
402  else
403  {
404  if (debug)
405  {
406  Info<< "probes: remapping sample locations" << endl;
407  }
408 
409  // 1. Update cells
410  {
411  DynamicList<label> elems(elementList_.size());
412 
413  const labelList& reverseMap = mpm.reverseCellMap();
414  forAll(elementList_, i)
415  {
416  label celli = elementList_[i];
417  label newCelli = reverseMap[celli];
418  if (newCelli == -1)
419  {
420  // cell removed
421  }
422  else if (newCelli < -1)
423  {
424  // cell merged
425  elems.append(-newCelli - 2);
426  }
427  else
428  {
429  // valid new cell
430  elems.append(newCelli);
431  }
432  }
433 
434  elementList_.transfer(elems);
435  }
436 
437  // 2. Update faces
438  {
439  DynamicList<label> elems(faceList_.size());
440 
441  const labelList& reverseMap = mpm.reverseFaceMap();
442  forAll(faceList_, i)
443  {
444  label facei = faceList_[i];
445  label newFacei = reverseMap[facei];
446  if (newFacei == -1)
447  {
448  // face removed
449  }
450  else if (newFacei < -1)
451  {
452  // face merged
453  elems.append(-newFacei - 2);
454  }
455  else
456  {
457  // valid new face
458  elems.append(newFacei);
459  }
460  }
461 
462  faceList_.transfer(elems);
463  }
464  }
465 }
466 
467 
469 {
470  DebugInfo<< "probes: movePoints" << endl;
471 
472  if (fixedLocations_ && &mesh == &mesh_)
473  {
474  findElements(mesh_);
475  }
476 }
477 
478 
479 // ************************************************************************* //
A HashTable with keys but without contents.
Definition: HashSet.H:59
#define forAll(list, i)
Loop across all elements in list.
Definition: UList.H:434
intWM_LABEL_SIZE_t label
A label is an int32_t or int64_t as specified by the pre-processor macro WM_LABEL_SIZE.
Definition: label.H:59
A class for handling file names.
Definition: fileName.H:79
virtual bool write()
Sample and write.
Definition: probes.C:368
label prepare()
Classify field type and Open/close file streams,.
Definition: probes.C:168
A list of keyword definitions, which are a keyword followed by any number of values (e...
Definition: dictionary.H:158
virtual void updateMesh(const mapPolyMesh &)
Update for changes of mesh.
Definition: probes.C:389
virtual bool execute()
Execute, currently does nothing.
Definition: probes.C:362
bool clean()
Cleanup file name.
Definition: fileName.C:81
#define forAllIter(Container, container, iter)
Iterate across all elements in the container object of type.
Definition: UList.H:459
Output to file stream.
Definition: OFstream.H:82
static unsigned int defaultPrecision()
Return the default precision.
Definition: IOstream.H:461
static int myProcNo(const label communicator=0)
Number of this process (starting from masterNo() = 0)
Definition: UPstream.H:429
static word defaultRegion
Return the default region name.
Definition: polyMesh.H:309
Ostream & endl(Ostream &os)
Add newline and flush stream.
Definition: Ostream.H:256
static bool master(const label communicator=0)
Am I the master process.
Definition: UPstream.H:423
const cellList & cells() const
A HashTable specialization for hashing pointers.
Definition: HashPtrTable.H:50
Abstract base-class for Time/database function objects.
const fileName & name() const
Return the name of the stream.
Definition: OFstream.H:120
bool insert(const Key &key)
Insert a new entry.
Definition: HashSet.H:108
const Type & lookupObject(const word &name) const
Lookup and return the object of the given Type.
Class to control time during OpenFOAM simulations that is also the top-level objectRegistry.
Definition: Time.H:68
Class containing mesh-to-mesh mapping information after a change in polyMesh topology.
Definition: mapPolyMesh.H:158
Macros for easy insertion into run-time selection tables.
bool erase(const iterator &)
Erase a hashedEntry specified by given iterator.
Definition: HashTable.C:371
bool read(const char *, int32_t &)
Definition: int32IO.C:85
vectorField pointField
pointField is a vectorField.
Definition: pointFieldFwd.H:42
dynamicFvMesh & mesh
A class for handling words, derived from string.
Definition: word.H:59
virtual void findElements(const fvMesh &)
Find cells and faces containing probes.
Definition: probes.C:51
word name() const
Return file name (part beyond last /)
Definition: fileName.C:183
bool readIfPresent(const word &, T &, bool recursive=false, bool patternMatch=true) const
Find an entry if present, and assign to T.
DynamicList< T, SizeInc, SizeMult, SizeDiv > & append(const T &)
Append an element at the end of the list.
Definition: DynamicListI.H:296
const vectorField & cellCentres() const
forAllConstIter(PtrDictionary< phaseModel >, mixture.phases(), phase)
Definition: pEqn.H:29
#define DebugInfo
Report an information message using Foam::Info.
Istream and Ostream manipulators taking arguments.
addToRunTimeSelectionTable(ensightPart, ensightPartCells, istream)
static const char nl
Definition: Ostream.H:265
const labelList & reverseCellMap() const
Reverse cell map.
Definition: mapPolyMesh.H:521
defineTypeNameAndDebug(combustionModel, 0)
bool mkDir(const fileName &, mode_t=0777)
Make a directory and return an error if it could not be created.
Definition: POSIX.C:290
void reduce(const List< UPstream::commsStruct > &comms, T &Value, const BinaryOp &bop, const int tag, const label comm)
word name(const complex &)
Return a string representation of a complex.
Definition: complex.C:47
virtual void movePoints(const polyMesh &)
Update for changes of mesh.
Definition: probes.C:468
const vectorField & faceCentres() const
static bool & parRun()
Is this a parallel run?
Definition: UPstream.H:399
T lookupOrDefault(const word &, const T &, bool recursive=false, bool patternMatch=true) const
Find and return a T,.
#define WarningInFunction
Report a warning using Foam::Warning.
prefixOSstream Pout(cout, "Pout")
Definition: IOstreams.H:53
Mesh data needed to do the Finite Volume discretisation.
Definition: fvMesh.H:78
const polyMesh & mesh() const
Return polyMesh.
Definition: mapPolyMesh.H:352
virtual bool read(const dictionary &)
Read the probes.
Definition: probes.C:336
label findCell(const point &p, const cellDecomposition=CELL_TETS) const
Find cell enclosing this location and return index.
Definition: polyMesh.C:1606
messageStream Info
dimensioned< scalar > mag(const dimensioned< Type > &)
fileName path() const
Return directory path name (part before last /)
Definition: fileName.C:253
probes(const word &name, const Time &time, const dictionary &dict)
Construct from Time and dictionary.
Definition: probes.C:281
Mesh consisting of general polyhedral cells.
Definition: polyMesh.H:74
Omanip< int > setw(const int i)
Definition: IOmanip.H:199
Registry of regIOobjects.
virtual ~probes()
Destructor.
Definition: probes.C:330
const labelList & reverseFaceMap() const
Reverse face map.
Definition: mapPolyMesh.H:490
Namespace for OpenFOAM.
ITstream & lookup(const word &, bool recursive=false, bool patternMatch=true) const
Find and return an entry data stream.
Definition: dictionary.C:583