Loading [MathJax]/extensions/tex2jax.js
Mesh Oriented datABase  (version 5.5.1)
An array-based unstructured mesh library
All Classes Namespaces Files Functions Variables Typedefs Enumerations Enumerator Friends Macros Pages
LloydRelaxation.cpp File Reference
#include "moab/Core.hpp"
#include "moab/Skinner.hpp"
#include "moab/CN.hpp"
#include "moab/CartVect.hpp"
#include <iostream>
#include <sstream>
+ Include dependency graph for LloydRelaxation.cpp:

Go to the source code of this file.

Functions

ErrorCode perform_lloyd_relaxation (Interface *mb, Range &verts, Range &cells, Tag fixed, int num_its, int report_its)
 
int main (int argc, char **argv)
 

Variables

string test_file_name = string( MESH_DIR ) + string( "/surfrandomtris-4part.h5m" )
 

Function Documentation

◆ main()

int main ( int  argc,
char **  argv 
)
Examples
LloydRelaxation.cpp.

Definition at line 38 of file LloydRelaxation.cpp.

39 { 40  int num_its = 10; 41  int report_its = 1; 42  43 #ifdef MOAB_HAVE_MPI 44  MPI_Init( &argc, &argv ); 45 #endif 46  47  // Need option handling here for input filename 48  if( argc > 1 ) 49  { 50  // User has input a mesh file 51  test_file_name = argv[1]; 52  } 53  54  // Get MOAB instance 55  Interface* mb = new( std::nothrow ) Core; 56  if( NULL == mb ) return 1; 57  58 #ifdef MOAB_HAVE_MPI 59  // Get the ParallelComm instance 60  ParallelComm* pcomm = new ParallelComm( mb, MPI_COMM_WORLD ); 61  int nprocs = pcomm->size(); 62 #else 63  int nprocs = 1; 64 #endif 65  string options; 66  if( nprocs > 1 ) // If reading in parallel, need to tell it how 67  options = "PARALLEL=READ_PART;PARTITION=PARALLEL_PARTITION;PARALLEL_RESOLVE_SHARED_ENTS;" 68  "PARALLEL_GHOSTS=2.0.1;DEBUG_IO=0;DEBUG_PIO=0"; 69  70  // Load the test file with specified options 71  ErrorCode rval = mb->load_file( test_file_name.c_str(), 0, options.c_str() );MB_CHK_ERR( rval ); 72  73  // Make tag to specify fixed vertices, since it's input to the algorithm; use a default value of 74  // non-fixed so we only need to set the fixed tag for skin vertices 75  Tag fixed; 76  int def_val = 0; 77  rval = mb->tag_get_handle( "fixed", 1, MB_TYPE_INTEGER, fixed, MB_TAG_CREAT | MB_TAG_DENSE, &def_val );MB_CHK_ERR( rval ); 78  79  // Get all vertices and faces 80  Range verts, faces, skin_verts; 81  rval = mb->get_entities_by_type( 0, MBVERTEX, verts );MB_CHK_ERR( rval ); 82  rval = mb->get_entities_by_dimension( 0, 2, faces );MB_CHK_ERR( rval ); 83  84  // Get the skin vertices of those faces and mark them as fixed; we don't want to fix the 85  // vertices on a part boundary, but since we exchanged a layer of ghost faces, those vertices 86  // aren't on the skin locally ok to mark non-owned skin vertices too, I won't move those anyway 87  // use MOAB's skinner class to find the skin 88  Skinner skinner( mb ); 89  rval = skinner.find_skin( 0, faces, true, skin_verts );MB_CHK_ERR( rval ); // 'true' param indicates we want vertices back, not faces 90  91  vector< int > fix_tag( skin_verts.size(), 1 ); // Initialized to 1 to indicate fixed 92  rval = mb->tag_set_data( fixed, skin_verts, &fix_tag[0] );MB_CHK_ERR( rval ); 93  94  // Now perform the Lloyd relaxation 95  rval = perform_lloyd_relaxation( mb, verts, faces, fixed, num_its, report_its );MB_CHK_ERR( rval ); 96  97  // Delete fixed tag, since we created it here 98  rval = mb->tag_delete( fixed );MB_CHK_ERR( rval ); 99  100  // Output file, using parallel write 101  102 #ifdef MOAB_HAVE_MPI 103  options = "PARALLEL=WRITE_PART"; 104 #endif 105  106  rval = mb->write_file( "lloydfinal.h5m", NULL, options.c_str() );MB_CHK_ERR( rval ); 107  108  // Delete MOAB instance 109  delete mb; 110  111 #ifdef MOAB_HAVE_MPI 112  MPI_Finalize(); 113 #endif 114  115  return 0; 116 }

References ErrorCode, moab::Skinner::find_skin(), moab::Core::get_entities_by_dimension(), moab::Core::get_entities_by_type(), moab::Core::load_file(), mb, MB_CHK_ERR, MB_TAG_CREAT, MB_TAG_DENSE, MB_TYPE_INTEGER, MBVERTEX, perform_lloyd_relaxation(), moab::Range::size(), moab::ParallelComm::size(), moab::Core::tag_delete(), moab::Core::tag_get_handle(), moab::Core::tag_set_data(), test_file_name, and moab::Core::write_file().

◆ perform_lloyd_relaxation()

ErrorCode perform_lloyd_relaxation ( Interface mb,
Range verts,
Range cells,
Tag  fixed,
int  num_its,
int  report_its 
)
Examples
LloydRelaxation.cpp.

Definition at line 118 of file LloydRelaxation.cpp.

119 { 120  ErrorCode rval; 121  122 #ifdef MOAB_HAVE_MPI 123  ParallelComm* pcomm = ParallelComm::get_pcomm( mb, 0 ); 124  int nprocs = pcomm->size(); 125 #else 126  int nprocs = 1; 127 #endif 128  129  // Perform Lloyd relaxation: 130  // 1. Setup: set vertex centroids from vertex coords; filter to owned verts; get fixed tags 131  132  // Get all verts coords into tag; don't need to worry about filtering out fixed verts, 133  // we'll just be setting to their fixed coords 134  vector< double > vcentroids( 3 * verts.size() ); 135  rval = mb->get_coords( verts, &vcentroids[0] );MB_CHK_ERR( rval ); 136  137  Tag centroid; 138  rval = mb->tag_get_handle( "centroid", 3, MB_TYPE_DOUBLE, centroid, MB_TAG_CREAT | MB_TAG_DENSE );MB_CHK_ERR( rval ); 139  rval = mb->tag_set_data( centroid, verts, &vcentroids[0] );MB_CHK_ERR( rval ); 140  141  // Filter verts down to owned ones and get fixed tag for them 142  Range owned_verts, shared_owned_verts; 143  if( nprocs > 1 ) 144  { 145 #ifdef MOAB_HAVE_MPI 146  rval = pcomm->filter_pstatus( verts, PSTATUS_NOT_OWNED, PSTATUS_NOT, -1, &owned_verts );MB_CHK_ERR( rval ); 147 #endif 148  } 149  else 150  owned_verts = verts; 151  vector< int > fix_tag( owned_verts.size() ); 152  rval = mb->tag_get_data( fixed, owned_verts, &fix_tag[0] );MB_CHK_ERR( rval ); 153  154  // Now fill vcentroids array with positions of just owned vertices, since those are the ones 155  // we're actually computing 156  vcentroids.resize( 3 * owned_verts.size() ); 157  rval = mb->tag_get_data( centroid, owned_verts, &vcentroids[0] );MB_CHK_ERR( rval ); 158  159 #ifdef MOAB_HAVE_MPI 160  // Get shared owned verts, for exchanging tags 161  rval = pcomm->get_shared_entities( -1, shared_owned_verts, 0, false, true );MB_CHK_ERR( rval ); 162  // Workaround: if no shared owned verts, put a non-shared one in the list, to prevent exchanging 163  // tags for all shared entities 164  if( shared_owned_verts.empty() ) shared_owned_verts.insert( *verts.begin() ); 165 #endif 166  167  // Some declarations for later iterations 168  vector< double > fcentroids( 3 * faces.size() ); // fcentroids for face centroids 169  vector< double > ctag( 3 * CN::MAX_NODES_PER_ELEMENT ); // Temporary coordinate storage for verts bounding a face 170  const EntityHandle* conn; // const ptr & size to face connectivity 171  int nconn; 172  Range::iterator fit, vit; // For iterating over faces, verts 173  int f, v; // For indexing into centroid vectors 174  vector< EntityHandle > adj_faces; // Used in vertex iteration 175  176  // 2. For num_its iterations: 177  for( int nit = 0; nit < num_its; nit++ ) 178  { 179  double mxdelta = 0.0; 180  181  // 2a. For each face: centroid = sum(vertex centroids)/num_verts_in_cell 182  for( fit = faces.begin(), f = 0; fit != faces.end(); ++fit, f++ ) 183  { 184  // Get verts for this face 185  rval = mb->get_connectivity( *fit, conn, nconn );MB_CHK_ERR( rval ); 186  // Get centroid tags for those verts 187  rval = mb->tag_get_data( centroid, conn, nconn, &ctag[0] );MB_CHK_ERR( rval ); 188  fcentroids[3 * f + 0] = fcentroids[3 * f + 1] = fcentroids[3 * f + 2] = 0.0; 189  for( v = 0; v < nconn; v++ ) 190  { 191  fcentroids[3 * f + 0] += ctag[3 * v + 0]; 192  fcentroids[3 * f + 1] += ctag[3 * v + 1]; 193  fcentroids[3 * f + 2] += ctag[3 * v + 2]; 194  } 195  for( v = 0; v < 3; v++ ) 196  fcentroids[3 * f + v] /= nconn; 197  } 198  rval = mb->tag_set_data( centroid, faces, &fcentroids[0] );MB_CHK_ERR( rval ); 199  200  // 2b. For each owned vertex: 201  for( vit = owned_verts.begin(), v = 0; vit != owned_verts.end(); ++vit, v++ ) 202  { 203  // if !fixed 204  if( fix_tag[v] ) continue; 205  // vertex centroid = sum(cell centroids)/ncells 206  adj_faces.clear(); 207  rval = mb->get_adjacencies( &( *vit ), 1, 2, false, adj_faces );MB_CHK_ERR( rval ); 208  rval = mb->tag_get_data( centroid, &adj_faces[0], adj_faces.size(), &fcentroids[0] );MB_CHK_ERR( rval ); 209  double vnew[] = { 0.0, 0.0, 0.0 }; 210  for( f = 0; f < (int)adj_faces.size(); f++ ) 211  { 212  vnew[0] += fcentroids[3 * f + 0]; 213  vnew[1] += fcentroids[3 * f + 1]; 214  vnew[2] += fcentroids[3 * f + 2]; 215  } 216  for( f = 0; f < 3; f++ ) 217  vnew[f] /= adj_faces.size(); 218  double delta = ( CartVect( vnew ) - CartVect( &vcentroids[3 * v] ) ).length(); 219  mxdelta = std::max( delta, mxdelta ); 220  for( f = 0; f < 3; f++ ) 221  vcentroids[3 * v + f] = vnew[f]; 222  } 223  224  // Set the centroid tag; having them only in vcentroids array isn't enough, as vertex 225  // centroids are accessed randomly in loop over faces 226  rval = mb->tag_set_data( centroid, owned_verts, &vcentroids[0] );MB_CHK_ERR( rval ); 227  228  // 2c. Exchange tags on owned verts 229  if( nprocs > 1 ) 230  { 231 #ifdef MOAB_HAVE_MPI 232  rval = pcomm->exchange_tags( centroid, shared_owned_verts );MB_CHK_ERR( rval ); 233 #endif 234  } 235  236  if( !( nit % report_its ) ) 237  { 238  // Global reduce for maximum delta, then report it 239  double global_max = mxdelta; 240  int myrank = 0; 241 #ifdef MOAB_HAVE_MPI 242  if( nprocs > 1 ) MPI_Reduce( &mxdelta, &global_max, 1, MPI_DOUBLE, MPI_MAX, 0, pcomm->comm() ); 243  myrank = pcomm->rank(); 244 #endif 245  if( 1 == nprocs || 0 == myrank ) cout << "Max delta = " << global_max << endl; 246  } 247  } 248  249  // Write the tag back onto vertex coordinates 250  rval = mb->set_coords( owned_verts, &vcentroids[0] );MB_CHK_ERR( rval ); 251  252  // Delete the centroid tag, since we don't need it anymore 253  rval = mb->tag_delete( centroid );MB_CHK_ERR( rval ); 254  255  return MB_SUCCESS; 256 }

References moab::Range::begin(), moab::ParallelComm::comm(), moab::Range::empty(), moab::Range::end(), ErrorCode, moab::ParallelComm::exchange_tags(), moab::ParallelComm::filter_pstatus(), moab::Core::get_adjacencies(), moab::Core::get_connectivity(), moab::Core::get_coords(), moab::ParallelComm::get_pcomm(), moab::ParallelComm::get_shared_entities(), moab::Range::insert(), length(), moab::CN::MAX_NODES_PER_ELEMENT, mb, MB_CHK_ERR, MB_SUCCESS, MB_TAG_CREAT, MB_TAG_DENSE, MB_TYPE_DOUBLE, PSTATUS_NOT, PSTATUS_NOT_OWNED, moab::ParallelComm::rank(), moab::Core::set_coords(), moab::Range::size(), moab::ParallelComm::size(), moab::Core::tag_delete(), moab::Core::tag_get_data(), moab::Core::tag_get_handle(), and moab::Core::tag_set_data().

Referenced by main().

Variable Documentation

◆ test_file_name

string test_file_name = string( MESH_DIR ) + string( "/surfrandomtris-4part.h5m" )
Examples
LloydRelaxation.cpp.

Definition at line 34 of file LloydRelaxation.cpp.

Referenced by main().