Xyst test code coverage report
Current view: top level - Inciter - Refiner.cpp (source / functions) Hit Total Coverage
Commit: b2278901c7a653f0d92b235cc98ed02988a87738 Lines: 701 902 77.7 %
Date: 2024-12-18 15:54:33 Functions: 40 44 90.9 %
Legend: Lines: hit not hit | Branches: + taken - not taken # not executed Branches: 571 1614 35.4 %

           Branch data     Line data    Source code
       1                 :            : // *****************************************************************************
       2                 :            : /*!
       3                 :            :   \file      src/Inciter/Refiner.cpp
       4                 :            :   \copyright 2012-2015 J. Bakosi,
       5                 :            :              2016-2018 Los Alamos National Security, LLC.,
       6                 :            :              2019-2021 Triad National Security, LLC.,
       7                 :            :              2022-2024 J. Bakosi
       8                 :            :              All rights reserved. See the LICENSE file for details.
       9                 :            :   \brief     Mesh refiner for interfacing the mesh refinement library
      10                 :            :   \see       Refiner.h for more info.
      11                 :            : */
      12                 :            : // *****************************************************************************
      13                 :            : 
      14                 :            : #include <vector>
      15                 :            : #include <algorithm>
      16                 :            : 
      17                 :            : #include "Refiner.hpp"
      18                 :            : #include "Reorder.hpp"
      19                 :            : #include "AMR/mesh_adapter.hpp"
      20                 :            : #include "AMR/Error.hpp"
      21                 :            : #include "InciterConfig.hpp"
      22                 :            : #include "DerivedData.hpp"
      23                 :            : #include "UnsMesh.hpp"
      24                 :            : #include "Centering.hpp"
      25                 :            : #include "Around.hpp"
      26                 :            : #include "Sorter.hpp"
      27                 :            : #include "Discretization.hpp"
      28                 :            : #include "Problems.hpp"
      29                 :            : #include "Vector.hpp"
      30                 :            : 
      31                 :            : namespace inciter {
      32                 :            : 
      33                 :            : extern ctr::Config g_cfg;
      34                 :            : 
      35                 :            : } // inciter::
      36                 :            : 
      37                 :            : using inciter::Refiner;
      38                 :            : 
      39                 :       2475 : Refiner::Refiner( std::size_t meshid,
      40                 :            :                   const CProxy_Transporter& transporter,
      41                 :            :                   const CProxy_Sorter& sorter,
      42                 :            :                   const tk::CProxy_MeshWriter& meshwriter,
      43                 :            :                   const CProxy_Discretization& discretization,
      44                 :            :                   const CProxy_RieCG& riecg,
      45                 :            :                   const CProxy_LaxCG& laxcg,
      46                 :            :                   const CProxy_ZalCG& zalcg,
      47                 :            :                   const CProxy_KozCG& kozcg,
      48                 :            :                   const CProxy_ChoCG& chocg,
      49                 :            :                   const CProxy_LohCG& lohcg,
      50                 :            :                   const tk::CProxy_ConjugateGradients& cgpre,
      51                 :            :                   const tk::CProxy_ConjugateGradients& cgmom,
      52                 :            :                   const tk::RefinerCallback& cbr,
      53                 :            :                   const tk::SorterCallback& cbs,
      54                 :            :                   const std::vector< std::size_t >& ginpoel,
      55                 :            :                   const tk::UnsMesh::CoordMap& coordmap,
      56                 :            :                   const std::map< int, std::vector< std::size_t > >& bface,
      57                 :            :                   const std::vector< std::size_t >& triinpoel,
      58                 :            :                   const std::map< int, std::vector< std::size_t > >& bnode,
      59                 :       2475 :                   int nchare ) :
      60                 :       2475 :   m_meshid( meshid ),
      61                 :       2475 :   m_ncit(0),
      62                 :       2475 :   m_host( transporter ),
      63         [ +  - ]:       2475 :   m_sorter( sorter ),
      64         [ +  - ]:       2475 :   m_meshwriter( meshwriter ),
      65         [ +  - ]:       2475 :   m_disc( discretization ),
      66         [ +  - ]:       2475 :   m_riecg( riecg ),
      67         [ +  - ]:       2475 :   m_laxcg( laxcg ),
      68         [ +  - ]:       2475 :   m_zalcg( zalcg ),
      69         [ +  - ]:       2475 :   m_kozcg( kozcg ),
      70         [ +  - ]:       2475 :   m_chocg( chocg ),
      71         [ +  - ]:       2475 :   m_lohcg( lohcg ),
      72         [ +  - ]:       2475 :   m_cgpre( cgpre ),
      73         [ +  - ]:       2475 :   m_cgmom( cgmom ),
      74                 :       2475 :   m_cbr( cbr ),
      75                 :       2475 :   m_cbs( cbs ),
      76         [ +  - ]:       2475 :   m_ginpoel( ginpoel ),
      77         [ +  - ]:       2475 :   m_el( tk::global2local( ginpoel ) ),     // fills m_inpoel, m_gid, m_lid
      78                 :       2475 :   m_coordmap( coordmap ),
      79         [ +  - ]:       2475 :   m_coord( flatcoord(coordmap) ),
      80         [ +  - ]:       2475 :   m_bface( bface ),
      81         [ +  - ]:       2475 :   m_bnode( bnode ),
      82         [ +  - ]:       2475 :   m_triinpoel( triinpoel ),
      83                 :       2475 :   m_nchare( nchare ),
      84                 :       2475 :   m_mode( RefMode::T0REF ),
      85         [ +  - ]:       2475 :   m_initref( g_cfg.get< tag::href_init >() ),
      86                 :       2475 :   m_ninitref( m_initref.size() ),
      87         [ +  - ]:       2475 :   m_refiner( g_cfg.get< tag::href_maxlevels >(), m_inpoel ),
      88                 :       2475 :   m_nref( 0 ),
      89                 :       2475 :   m_nbnd( 0 ),
      90                 :       2475 :   m_extra( 0 ),
      91                 :       2475 :   m_oldntets( 0 ),
      92         [ +  - ]:       2475 :   m_rid( m_coord[0].size() ),
      93                 :            : //  m_oldrid(),
      94         [ +  - ]:       2475 :   m_lref( m_rid.size() ),
      95                 :            : //  m_oldparent(),
      96 [ +  - ][ +  - ]:       9900 :   m_writeCallback()
      97                 :            : // *****************************************************************************
      98                 :            : //  Constructor
      99                 :            : //! \param[in] meshid Mesh ID
     100                 :            : //! \param[in] transporter Transporter (host) proxy
     101                 :            : //! \param[in] sorter Mesh reordering (sorter) proxy
     102                 :            : //! \param[in] meshwriter Mesh writer proxy
     103                 :            : //! \param[in] discretization Discretization base proxy
     104                 :            : //! \param[in] riecg Discretization scheme proxy
     105                 :            : //! \param[in] laxcg Discretization scheme proxy
     106                 :            : //! \param[in] zalcg Discretization scheme proxy
     107                 :            : //! \param[in] kozcg Discretization scheme proxy
     108                 :            : //! \param[in] chocg Discretization scheme proxy
     109                 :            : //! \param[in] lohcg Discretization scheme proxy
     110                 :            : //! \param[in] cgpre ConjugateGradients Charm++ proxy for pressure solve
     111                 :            : //! \param[in] cgmom ConjugateGradients Charm++ proxy for momentum solve
     112                 :            : //! \param[in] cbr Charm++ callbacks for Refiner
     113                 :            : //! \param[in] cbs Charm++ callbacks for Sorter
     114                 :            : //! \param[in] ginpoel Mesh connectivity (this chare) using global node IDs
     115                 :            : //! \param[in] coordmap Mesh node coordinates (this chare) for global node IDs
     116                 :            : //! \param[in] bface File-internal elem ids of side sets
     117                 :            : //! \param[in] triinpoel Triangle face connectivity with global IDs
     118                 :            : //! \param[in] bnode Node lists of side sets
     119                 :            : //! \param[in] nchare Total number of refiner chares (chare array elements)
     120                 :            : // *****************************************************************************
     121                 :            : {
     122 [ -  + ][ -  - ]:       2475 :   Assert( !m_ginpoel.empty(), "No elements assigned to refiner chare" );
         [ -  - ][ -  - ]
     123 [ +  - ][ -  + ]:       2475 :   Assert( tk::positiveJacobians( m_inpoel, m_coord ),
         [ -  - ][ -  - ]
                 [ -  - ]
     124                 :            :           "Input mesh to Refiner Jacobian non-positive" );
     125 [ +  - ][ +  - ]:       2475 :   Assert( !tk::leakyPartition(
         [ +  - ][ -  + ]
         [ -  - ][ -  - ]
                 [ -  - ]
     126                 :            :             tk::genEsuelTet( m_inpoel, tk::genEsup(m_inpoel,4) ),
     127                 :            :             m_inpoel, m_coord ),
     128                 :            :           "Input mesh to Refiner leaky" );
     129                 :            : 
     130                 :            :   #if not defined(__INTEL_COMPILER) || defined(NDEBUG)
     131                 :            :   // The above ifdef skips running the conformity test with the intel compiler
     132                 :            :   // in debug mode only. This is necessary because in tk::conforming(), filling
     133                 :            :   // up the map can fail with some meshes (only in parallel), e.g., tube.exo,
     134                 :            :   // used by some regression tests, due to the intel compiler generating some
     135                 :            :   // garbage incorrect code - only in debug, only in parallel, only with that
     136                 :            :   // mesh.
     137 [ +  - ][ -  + ]:       2475 :   Assert( tk::conforming( m_inpoel, m_coord, true, m_rid ),
         [ -  - ][ -  - ]
                 [ -  - ]
     138                 :            :           "Input mesh to Refiner not conforming" );
     139                 :            :   #endif
     140                 :            : 
     141                 :            :   // Generate local -> refiner lib node id map and its inverse
     142         [ +  - ]:       2475 :   libmap();
     143                 :            : 
     144                 :            :   // Reverse initial mesh refinement type list (will pop from back)
     145         [ +  - ]:       2475 :   std::reverse( begin(m_initref), end(m_initref) );
     146                 :            : 
     147                 :            :   // Generate boundary data structures for coarse mesh
     148         [ +  - ]:       2475 :   coarseMesh();
     149                 :            : 
     150                 :            :   // If initial mesh refinement is configured, start initial mesh refinement.
     151 [ +  + ][ +  - ]:       2475 :   if (g_cfg.get< tag::href_t0 >() && m_ninitref > 0) {
                 [ +  + ]
     152         [ +  - ]:         27 :     t0ref();
     153                 :            :   } else {
     154         [ +  - ]:       2448 :     endt0ref();
     155                 :            :   }
     156                 :       2475 : }
     157                 :            : 
     158                 :            : void
     159                 :       2475 : Refiner::libmap()
     160                 :            : // *****************************************************************************
     161                 :            : // (Re-)generate local -> refiner lib node id map and its inverse
     162                 :            : // *****************************************************************************
     163                 :            : {
     164                 :            :   // Fill initial (matching) mapping between local and refiner node ids
     165                 :       2475 :   std::iota( begin(m_rid), end(m_rid), 0 );
     166                 :            : 
     167                 :            :   // Fill in inverse, mapping refiner to local node ids
     168                 :       2475 :   std::size_t i = 0;
     169 [ +  - ][ +  + ]:     292066 :   for (auto r : m_rid) m_lref[r] = i++;
     170                 :       2475 : }
     171                 :            : 
     172                 :            : void
     173                 :       2484 : Refiner::coarseMesh()
     174                 :            : // *****************************************************************************
     175                 :            : // (Re-)generate side set and block data structures for coarse mesh
     176                 :            : // *****************************************************************************
     177                 :            : {
     178                 :            :   // Generate unique set of faces for each side set of the input (coarsest) mesh
     179                 :       2484 :   m_coarseBndFaces.clear();
     180         [ +  + ]:       8253 :   for (const auto& [ setid, faceids ] : m_bface) {
     181         [ +  - ]:       5769 :     auto& faces = m_coarseBndFaces[ setid ];
     182         [ +  + ]:     246927 :     for (auto f : faceids) {
     183         [ +  - ]:     241158 :       faces.insert(
     184                 :     241158 :         {{{ m_triinpoel[f*3+0], m_triinpoel[f*3+1], m_triinpoel[f*3+2] }}} );
     185                 :            :     }
     186                 :            :   }
     187                 :            : 
     188                 :            :   // Generate unique set of nodes for each side set of the input (coarsest) mesh
     189                 :       2484 :   m_coarseBndNodes.clear();
     190         [ +  + ]:       5117 :   for (const auto& [ setid, nodes ] : m_bnode) {
     191 [ +  - ][ +  - ]:       2633 :     m_coarseBndNodes[ setid ].insert( begin(nodes), end(nodes) );
     192                 :            :   }
     193                 :       2484 : }
     194                 :            : 
     195                 :            : void
     196                 :       2475 : Refiner::sendProxy()
     197                 :            : // *****************************************************************************
     198                 :            : // Send Refiner proxy to Discretization objects
     199                 :            : //! \details This should be called when bound Discretization chare array
     200                 :            : //!   elements have already been created.
     201                 :            : // *****************************************************************************
     202                 :            : {
     203                 :            :   // Make sure (bound) Discretization chare is already created and accessible
     204 [ +  - ][ +  - ]:       2475 :   Assert( m_disc[ thisIndex ].ckLocal() != nullptr,
         [ -  + ][ -  - ]
         [ -  - ][ -  - ]
     205                 :            :           "About to dereference nullptr" );
     206                 :            : 
     207                 :            :   // Pass Refiner Charm++ chare proxy to fellow (bound) Discretization object
     208 [ +  - ][ +  - ]:       2475 :   m_disc[ thisIndex ].ckLocal()->setRefiner( thisProxy );
                 [ +  - ]
     209                 :       2475 : }
     210                 :            : 
     211                 :            : void
     212                 :          9 : Refiner::reorder()
     213                 :            : // *****************************************************************************
     214                 :            : // Query Sorter and update local mesh with the reordered one
     215                 :            : // *****************************************************************************
     216                 :            : {
     217                 :          9 :   m_sorter[thisIndex].ckLocal()->
     218 [ +  - ][ +  - ]:          9 :     mesh( m_ginpoel, m_coordmap, m_triinpoel, m_bnode );
                 [ +  - ]
     219                 :            : 
     220                 :            :   // Update local mesh data based on data just received from Sorter
     221         [ +  - ]:          9 :   m_el = tk::global2local( m_ginpoel );     // fills m_inpoel, m_gid, m_lid
     222         [ +  - ]:          9 :   m_coord = flatcoord( m_coordmap );
     223                 :            : 
     224                 :            :   // Re-generate boundary data structures for coarse mesh
     225                 :          9 :   coarseMesh();
     226                 :            : 
     227                 :            :   // WARNING: This re-creates the AMR lib which is probably not what we
     228                 :            :   // ultimately want, beacuse this deletes its history recorded during initial
     229                 :            :   // (t<0) refinement. However, this appears to correctly update the local mesh
     230                 :            :   // based on the reordered one (from Sorter) at least when t0ref is off.
     231         [ +  - ]:         18 :   m_refiner = AMR::mesh_adapter_t(
     232                 :         18 :     g_cfg.get< tag::href_maxlevels >(), m_inpoel );
     233                 :          9 : }
     234                 :            : 
     235                 :            : tk::UnsMesh::Coords
     236                 :       2484 : Refiner::flatcoord( const tk::UnsMesh::CoordMap& coordmap )
     237                 :            : // *****************************************************************************
     238                 :            : // Generate flat coordinate data from coordinate map
     239                 :            : //! \param[in] coordmap Coordinates associated to global node IDs of mesh chunk
     240                 :            : //! \return Flat coordinate data
     241                 :            : // *****************************************************************************
     242                 :            : {
     243                 :       2484 :   tk::UnsMesh::Coords coord;
     244                 :            : 
     245                 :            :   // Convert node coordinates associated to global node IDs to a flat vector
     246                 :       2484 :   auto npoin = coordmap.size();
     247 [ -  + ][ -  - ]:       2484 :   Assert( m_gid.size() == npoin, "Size mismatch" );
         [ -  - ][ -  - ]
     248         [ +  - ]:       2484 :   coord[0].resize( npoin );
     249         [ +  - ]:       2484 :   coord[1].resize( npoin );
     250         [ +  - ]:       2484 :   coord[2].resize( npoin );
     251         [ +  + ]:     294385 :   for (const auto& [ gid, coords ] : coordmap) {
     252         [ +  - ]:     291901 :     auto i = tk::cref_find( m_lid, gid );
     253 [ -  + ][ -  - ]:     291901 :     Assert( i < npoin, "Indexing out of coordinate map" );
         [ -  - ][ -  - ]
     254                 :     291901 :     coord[0][i] = coords[0];
     255                 :     291901 :     coord[1][i] = coords[1];
     256                 :     291901 :     coord[2][i] = coords[2];
     257                 :            :   }
     258                 :            : 
     259                 :       2484 :   return coord;
     260                 :          0 : }
     261                 :            : 
     262                 :            : void
     263                 :          0 : Refiner::dtref( const std::map< int, std::vector< std::size_t > >& bface,
     264                 :            :                 const std::map< int, std::vector< std::size_t > >& bnode,
     265                 :            :                 const std::vector< std::size_t >& triinpoel )
     266                 :            : // *****************************************************************************
     267                 :            : // Start mesh refinement (during time stepping, t>0)
     268                 :            : //! \param[in] bface Boundary-faces mapped to side set ids
     269                 :            : //! \param[in] bnode Boundary-node lists mapped to side set ids
     270                 :            : //! \param[in] triinpoel Boundary-face connectivity
     271                 :            : // *****************************************************************************
     272                 :            : {
     273                 :          0 :   m_mode = RefMode::DTREF;
     274                 :            : 
     275                 :            :   // Update boundary node lists
     276                 :          0 :   m_bface = bface;
     277                 :          0 :   m_bnode = bnode;
     278         [ -  - ]:          0 :   m_triinpoel = tk::remap(triinpoel, m_gid);
     279                 :            : 
     280                 :          0 :   start();
     281                 :          0 : }
     282                 :            : 
     283                 :            : void
     284                 :         71 : Refiner::t0ref()
     285                 :            : // *****************************************************************************
     286                 :            : // Output mesh to file before a new step mesh refinement
     287                 :            : // *****************************************************************************
     288                 :            : {
     289 [ -  + ][ -  - ]:         71 :   Assert( m_ninitref > 0, "No initial mesh refinement steps configured" );
         [ -  - ][ -  - ]
     290                 :            :   // Output initial mesh to file
     291                 :         71 :   auto l = m_ninitref - m_initref.size();  // num initref steps completed
     292                 :         71 :   auto t0 = g_cfg.get< tag::t0 >();
     293         [ +  + ]:         71 :   if (l == 0) {
     294 [ +  - ][ +  - ]:         27 :     writeMesh( "t0ref", l, t0-1.0,
     295 [ +  - ][ +  - ]:         54 :       CkCallback( CkIndex_Refiner::start(), thisProxy[thisIndex] ) );
                 [ +  - ]
     296                 :            :   } else {
     297                 :         44 :     start();
     298                 :            :   }
     299                 :         71 : }
     300                 :            : 
     301                 :            : void
     302                 :         71 : Refiner::start()
     303                 :            : // *****************************************************************************
     304                 :            : //  Start new step of mesh refinement
     305                 :            : // *****************************************************************************
     306                 :            : {
     307                 :         71 :   m_extra = 0;
     308                 :         71 :   m_ch.clear();
     309                 :         71 :   m_remoteEdgeData.clear();
     310                 :         71 :   m_remoteEdges.clear();
     311                 :            : 
     312                 :         71 :   updateEdgeData();
     313                 :            : 
     314                 :            :   // Generate and communicate boundary edges
     315                 :         71 :   bndEdges();
     316                 :         71 : }
     317                 :            : 
     318                 :            : void
     319                 :         71 : Refiner::bndEdges()
     320                 :            : // *****************************************************************************
     321                 :            : // Generate boundary edges and send them to all chares
     322                 :            : //! \details Extract edges on the boundary only. The boundary edges (shared by
     323                 :            : //!   multiple chares) will be agreed on a refinement that yields a conforming
     324                 :            : //!   mesh across chares boundaries.
     325                 :            : // *****************************************************************************
     326                 :            : {
     327                 :            :   // Compute the number of edges (chunksize) a chare will respond to when
     328                 :            :   // computing shared edges
     329                 :         71 :   auto N = static_cast< std::size_t >( m_nchare );
     330                 :            :   // cppcheck-suppress unreadVariable
     331                 :         71 :   std::size_t chunksize = std::numeric_limits< std::size_t >::max() / N;
     332                 :            : 
     333                 :            :   // Generate boundary edges of our mesh chunk
     334                 :         71 :   std::unordered_map< int, EdgeSet > chbedges;
     335         [ +  - ]:         71 :   auto esup = tk::genEsup( m_inpoel, 4 );         // elements surrounding points
     336         [ +  - ]:         71 :   auto esuel = tk::genEsuelTet( m_inpoel, esup ); // elems surrounding elements
     337         [ +  + ]:      72875 :   for (std::size_t e=0; e<esuel.size()/4; ++e) {
     338                 :      72804 :     auto mark = e*4;
     339         [ +  + ]:     364020 :     for (std::size_t f=0; f<4; ++f) {
     340         [ +  + ]:     291216 :       if (esuel[mark+f] == -1) {
     341                 :      22868 :         auto A = m_ginpoel[ mark+tk::lpofa[f][0] ];
     342                 :            :         // cppcheck-suppress unreadVariable
     343                 :      22868 :         auto B = m_ginpoel[ mark+tk::lpofa[f][1] ];
     344                 :            :         // cppcheck-suppress unreadVariable
     345                 :      22868 :         auto C = m_ginpoel[ mark+tk::lpofa[f][2] ];
     346 [ +  - ][ -  + ]:      22868 :         Assert( m_lid.find( A ) != end(m_lid), "Local node ID not found" );
         [ -  - ][ -  - ]
                 [ -  - ]
     347 [ +  - ][ -  + ]:      22868 :         Assert( m_lid.find( B ) != end(m_lid), "Local node ID not found" );
         [ -  - ][ -  - ]
                 [ -  - ]
     348 [ +  - ][ -  + ]:      22868 :         Assert( m_lid.find( C ) != end(m_lid), "Local node ID not found" );
         [ -  - ][ -  - ]
                 [ -  - ]
     349                 :            :         // assign edges to bins a single chare will respond to when computing
     350                 :            :         // shared edges
     351                 :      22868 :         auto bin = A / chunksize;
     352 [ -  + ][ -  - ]:      22868 :         Assert( bin < N, "Will index out of number of chares" );
         [ -  - ][ -  - ]
     353 [ +  - ][ +  - ]:      22868 :         chbedges[ static_cast<int>(bin) ].insert( {A,B} );
     354                 :      22868 :         bin = B / chunksize;
     355 [ -  + ][ -  - ]:      22868 :         Assert( bin < N, "Will index out of number of chares" );
         [ -  - ][ -  - ]
     356 [ +  - ][ +  - ]:      22868 :         chbedges[ static_cast<int>(bin) ].insert( {B,C} );
     357                 :      22868 :         bin = C / chunksize;
     358 [ -  + ][ -  - ]:      22868 :         Assert( bin < N, "Will index out of number of chares" );
         [ -  - ][ -  - ]
     359 [ +  - ][ +  - ]:      22868 :         chbedges[ static_cast<int>(bin) ].insert( {C,A} );
     360                 :            :       }
     361                 :            :     }
     362                 :            :   }
     363                 :            : 
     364                 :            :   // Send edges in bins to chares that will compute shared edges
     365                 :         71 :   m_nbnd = chbedges.size();
     366         [ -  + ]:         71 :   if (m_nbnd == 0)
     367         [ -  - ]:          0 :     contribute( sizeof(std::size_t), &m_meshid, CkReduction::nop,
     368                 :          0 :                 m_cbr.get< tag::queried >() );
     369                 :            :   else
     370         [ +  + ]:        168 :     for (const auto& [ targetchare, bndedges ] : chbedges)
     371 [ +  - ][ +  - ]:         97 :       thisProxy[ targetchare ].query( thisIndex, bndedges );
     372                 :         71 : }
     373                 :            : 
     374                 :            : void
     375                 :         97 : Refiner::query( int fromch, const EdgeSet& edges )
     376                 :            : // *****************************************************************************
     377                 :            : // Incoming query for a list boundary edges for which this chare compiles
     378                 :            : // shared edges
     379                 :            : //! \param[in] fromch Sender chare ID
     380                 :            : //! \param[in] edges Chare-boundary edge list from another chare
     381                 :            : // *****************************************************************************
     382                 :            : {
     383                 :            :   // Store incoming edges in edge->chare and its inverse, chare->edge, maps
     384 [ +  - ][ +  - ]:      44086 :   for (const auto& e : edges) m_edgech[ e ].push_back( fromch );
                 [ +  + ]
     385                 :         97 :   m_chedge[ fromch ].insert( begin(edges), end(edges) );
     386                 :            :   // Report back to chare message received from
     387 [ +  - ][ +  - ]:         97 :   thisProxy[ fromch ].recvquery();
     388                 :         97 : }
     389                 :            : 
     390                 :            : void
     391                 :         97 : Refiner::recvquery()
     392                 :            : // *****************************************************************************
     393                 :            : // Receive receipt of boundary edge lists to query
     394                 :            : // *****************************************************************************
     395                 :            : {
     396         [ +  + ]:         97 :   if (--m_nbnd == 0)
     397                 :         71 :     contribute( sizeof(std::size_t), &m_meshid, CkReduction::nop,
     398                 :         71 :                 m_cbr.get< tag::queried >() );
     399                 :         97 : }
     400                 :            : 
     401                 :            : void
     402                 :         71 : Refiner::response()
     403                 :            : // *****************************************************************************
     404                 :            : //  Respond to boundary edge list queries
     405                 :            : // *****************************************************************************
     406                 :            : {
     407                 :         71 :   std::unordered_map< int, std::vector< int > > exp;
     408                 :            : 
     409                 :            :   // Compute shared edges whose chare ids will be sent back to querying chares
     410         [ +  + ]:        168 :   for (const auto& [ neighborchare, bndedges ] : m_chedge) {
     411         [ +  - ]:         97 :     auto& e = exp[ neighborchare ];
     412         [ +  + ]:      44086 :     for (const auto& ed : bndedges)
     413 [ +  - ][ +  + ]:      94126 :       for (auto d : tk::cref_find(m_edgech,ed))
     414         [ +  + ]:      50137 :         if (d != neighborchare)
     415                 :            :           // cppcheck-suppress useStlAlgorithm
     416         [ +  - ]:       6148 :           e.push_back( d );
     417                 :            :   }
     418                 :            : 
     419                 :            :   // Send chare ids of shared edges to chares that issued a query to us. Shared
     420                 :            :   // boundary edges assigned to chare ids sharing the boundary edge were
     421                 :            :   // computed above for those chares that queried this map from us. These
     422                 :            :   // boundary edges form a distributed table and we only work on a chunk of it.
     423                 :            :   // Note that we only send data back to those chares that have queried us. The
     424                 :            :   // receiving sides do not know in advance if they receive messages or not.
     425                 :            :   // Completion is detected by having the receiver respond back and counting
     426                 :            :   // the responses on the sender side, i.e., this chare.
     427                 :         71 :   m_nbnd = exp.size();
     428         [ +  + ]:         71 :   if (m_nbnd == 0)
     429         [ +  - ]:         20 :     contribute( sizeof(std::size_t), &m_meshid, CkReduction::nop,
     430                 :         20 :                 m_cbr.get< tag::responded >() );
     431                 :            :   else
     432         [ +  + ]:        148 :     for (const auto& [ targetchare, bndedges ] : exp)
     433 [ +  - ][ +  - ]:         97 :       thisProxy[ targetchare ].bnd( thisIndex, bndedges );
     434                 :         71 : }
     435                 :            : 
     436                 :            : void
     437                 :         97 : Refiner::bnd( int fromch, const std::vector< int >& chares )
     438                 :            : // *****************************************************************************
     439                 :            : // Receive shared boundary edges for our mesh chunk
     440                 :            : //! \param[in] fromch Sender chare ID
     441                 :            : //! \param[in] chares Chare ids we share edges with
     442                 :            : // *****************************************************************************
     443                 :            : {
     444                 :            :   // Store chare ids we share edges with
     445                 :         97 :   m_ch.insert( begin(chares), end(chares) );
     446                 :            : 
     447                 :            :   // Report back to chare message received from
     448 [ +  - ][ +  - ]:         97 :   thisProxy[ fromch ].recvbnd();
     449                 :         97 : }
     450                 :            : 
     451                 :            : void
     452                 :         97 : Refiner::recvbnd()
     453                 :            : // *****************************************************************************
     454                 :            : // Receive receipt of shared boundary edges
     455                 :            : // *****************************************************************************
     456                 :            : {
     457         [ +  + ]:         97 :   if (--m_nbnd == 0)
     458                 :         51 :     contribute( sizeof(std::size_t), &m_meshid, CkReduction::nop,
     459                 :         51 :                 m_cbr.get< tag::responded >() );
     460                 :         97 : }
     461                 :            : 
     462                 :            : void
     463                 :         71 : Refiner::refine()
     464                 :            : // *****************************************************************************
     465                 :            : //  Do a single step of mesh refinement (really, only tag edges)
     466                 :            : //! \details During initial (t<0, t0ref) mesh refinement, this is a single step
     467                 :            : //!   in a potentially multiple-entry list of initial adaptive mesh refinement
     468                 :            : //!   steps. Distribution of the chare-boundary edges must have preceded this
     469                 :            : //!   step, so that boundary edges (shared by multiple chares) can agree on a
     470                 :            : //!   refinement that yields a conforming mesh across chare boundaries.
     471                 :            : //!
     472                 :            : //!   During-timestepping (t>0, dtref) mesh refinement this is called once, as
     473                 :            : //!   we only do a single step during time stepping.
     474                 :            : // *****************************************************************************
     475                 :            : {
     476                 :            :   // Free memory used for computing shared boundary edges
     477                 :         71 :   tk::destroy( m_edgech );
     478                 :         71 :   tk::destroy( m_chedge );
     479                 :            : 
     480                 :            :   // Perform leak test on old mesh
     481 [ +  - ][ +  - ]:         71 :   Assert( !tk::leakyPartition(
         [ +  - ][ -  + ]
         [ -  - ][ -  - ]
                 [ -  - ]
     482                 :            :             tk::genEsuelTet( m_inpoel, tk::genEsup(m_inpoel,4) ),
     483                 :            :             m_inpoel, m_coord ),
     484                 :            :           "Mesh partition before refinement leaky" );
     485                 :            : 
     486         [ +  - ]:         71 :   if (m_mode == RefMode::T0REF) {
     487                 :            : 
     488                 :            :     // Refine mesh based on next initial refinement type
     489         [ +  - ]:         71 :     if (!m_initref.empty()) {
     490         [ +  - ]:         71 :       auto r = m_initref.back();    // consume (reversed) list from its back
     491         [ +  + ]:         71 :       if (r == "uniform")
     492         [ +  - ]:         31 :         uniformRefine();
     493         [ +  + ]:         40 :       else if (r == "uniform_deref")
     494         [ +  - ]:         24 :         uniformDeRefine();
     495         [ +  - ]:         16 :       else if (r == "ic")
     496         [ +  - ]:         16 :         errorRefine();
     497         [ -  - ]:          0 :       else if (r == "coord")
     498         [ -  - ]:          0 :         coordRefine();
     499         [ -  - ]:          0 :       else if (r == "edges")
     500         [ -  - ]:          0 :         edgelistRefine();
     501 [ -  - ][ -  - ]:          0 :       else Throw( "Initial AMR type not implemented" );
                 [ -  - ]
     502                 :         71 :     }
     503                 :            : 
     504         [ -  - ]:          0 :   } else if (m_mode == RefMode::DTREF) {
     505                 :            : 
     506                 :            :     //if (true)//g_cfg.get< tag::amr, tag::dtref_uniform >())
     507                 :          0 :       uniformRefine();
     508                 :            :     //else
     509                 :            :       //errorRefine();
     510                 :            : 
     511 [ -  - ][ -  - ]:          0 :   } else Throw( "RefMode not implemented" );
                 [ -  - ]
     512                 :            : 
     513                 :            :   // Communicate extra edges
     514                 :         71 :   comExtra();
     515                 :         71 : }
     516                 :            : 
     517                 :            : void
     518                 :         71 : Refiner::comExtra()
     519                 :            : // *****************************************************************************
     520                 :            : // Communicate extra edges along chare boundaries
     521                 :            : // *****************************************************************************
     522                 :            : {
     523                 :            :   // Export extra added nodes on our mesh chunk boundary to other chares
     524         [ +  + ]:         71 :   if (m_ch.empty()) {
     525                 :         13 :     correctref();
     526                 :            :   } else {
     527         [ +  + ]:        146 :     for (auto c : m_ch) {  // for all chares we share at least an edge with
     528 [ +  - ][ +  - ]:         88 :       thisProxy[c].addRefBndEdges(thisIndex, m_localEdgeData, m_intermediates);
                 [ +  - ]
     529                 :            :     }
     530                 :            :   }
     531                 :         71 : }
     532                 :            : 
     533                 :            : void
     534                 :         88 : Refiner::addRefBndEdges(
     535                 :            :   int fromch,
     536                 :            :   const AMR::EdgeData& ed,
     537                 :            :   const std::unordered_set< std::size_t >& intermediates )
     538                 :            : // *****************************************************************************
     539                 :            : //! Receive edges on our chare boundary from other chares
     540                 :            : //! \param[in] fromch Chare call coming from
     541                 :            : //! \param[in] ed Edges on chare boundary
     542                 :            : //! \param[in] intermediates Intermediate nodes
     543                 :            : //! \details Other than update remoteEdge data, this function also updates
     544                 :            : //!   locking information for such edges whos nodes are marked as intermediate
     545                 :            : //!   by neighboring chares.
     546                 :            : // *****************************************************************************
     547                 :            : {
     548                 :            :   // Save/augment buffers of edge data for each sender chare
     549                 :         88 :   auto& red = m_remoteEdgeData[ fromch ];
     550                 :         88 :   auto& re = m_remoteEdges[ fromch ];
     551                 :            :   using edge_data_t = std::tuple< Edge, int, int, AMR::Edge_Lock_Case >;
     552         [ +  + ]:      77072 :   for (const auto& [ edge, data ] : ed) {
     553         [ +  - ]:     153968 :     red.push_back( edge_data_t{ edge, std::get<0>(data), std::get<1>(data),
     554                 :      76984 :       std::get<2>(data) } );
     555         [ +  - ]:      76984 :     re.push_back( edge );
     556                 :            :   }
     557                 :            : 
     558                 :            :   // Add intermediates to mesh refiner lib
     559                 :            :   // needs to be done only when mesh has been actually updated, i.e. first iter
     560         [ +  - ]:         88 :   if (m_ncit == 0) {
     561         [ +  - ]:         88 :     auto esup = tk::genEsup( m_inpoel, 4 );
     562         [ +  - ]:         88 :     auto psup = tk::genPsup( m_inpoel, 4, esup );
     563         [ +  + ]:        239 :     for (const auto g : intermediates) {
     564         [ +  - ]:        151 :       auto l = m_lid.find( g ); // convert to local node ids
     565         [ -  + ]:        151 :       if (l != end(m_lid)) {
     566                 :            :         // lock all edges connected to intermediate node
     567                 :          0 :         auto p = l->second;
     568         [ -  - ]:          0 :         for (auto q : tk::Around(psup,p)) {
     569         [ -  - ]:          0 :           AMR::edge_t e(m_rid[p], m_rid[q]);
     570         [ -  - ]:          0 :           auto& refedge = m_refiner.tet_store.edge_store.get(e);
     571         [ -  - ]:          0 :           if (refedge.lock_case == AMR::Edge_Lock_Case::unlocked) {
     572                 :          0 :             refedge.lock_case = AMR::Edge_Lock_Case::temporary; //intermediate;
     573                 :          0 :             refedge.needs_refining = 0;
     574                 :            :           }
     575                 :            :         }
     576                 :            :       }
     577                 :            :     }
     578                 :         88 :   }
     579                 :            : 
     580                 :            :   // Heard from every worker we share at least a single edge with
     581         [ +  + ]:         88 :   if (++m_nref == m_ch.size()) {
     582                 :         58 :     m_nref = 0;
     583                 :            : 
     584         [ +  - ]:         58 :     updateBndEdgeData();
     585                 :            : 
     586         [ +  - ]:         58 :     std::vector< std::size_t > meshdata{ m_meshid };
     587         [ +  - ]:         58 :     contribute( meshdata, CkReduction::max_ulong,
     588                 :         58 :                 m_cbr.get< tag::compatibility >() );
     589                 :         58 :   }
     590                 :         88 : }
     591                 :            : 
     592                 :            : void
     593                 :         71 : Refiner::correctref()
     594                 :            : // *****************************************************************************
     595                 :            : //  Correct extra edges to arrive at conforming mesh across chare boundaries
     596                 :            : //! \details This function is called repeatedly until there is not a a single
     597                 :            : //!    edge that needs correction for the whole distributed problem to arrive at
     598                 :            : //!    a conforming mesh across chare boundaries during a mesh refinement step.
     599                 :            : // *****************************************************************************
     600                 :            : {
     601                 :         71 :   auto unlocked = AMR::Edge_Lock_Case::unlocked;
     602                 :            : 
     603                 :            :   // Storage for edge data that need correction to yield a conforming mesh
     604                 :         71 :   AMR::EdgeData extra;
     605                 :         71 :   std::size_t neigh_extra(0);
     606                 :            : 
     607                 :            :   // Vars for debugging purposes
     608                 :            :   // cppcheck-suppress unreadVariable
     609                 :         71 :   std::size_t nlocked(0);
     610                 :         71 :   std::array< std::size_t, 4 > ncorrcase{{0,0,0,0}};
     611                 :            : 
     612                 :            :   // loop through all edges shared with other chares
     613         [ +  + ]:        159 :   for (const auto& [ neighborchare, edgedata ] : m_remoteEdgeData) {
     614                 :      77072 :     for (const auto& [edge,remote_needs_refining,remote_needs_derefining,
     615         [ +  + ]:     154144 :       remote_lock_case] : edgedata) {
     616                 :            :       // find local data of remote edge
     617         [ +  - ]:      76984 :       auto it = m_localEdgeData.find( edge );
     618         [ +  + ]:      76984 :       if (it != end(m_localEdgeData)) {
     619                 :       4484 :         auto& local = it->second;
     620                 :       4484 :         auto& local_needs_refining = std::get<0>(local);
     621                 :       4484 :         auto& local_needs_derefining = std::get<1>(local);
     622                 :       4484 :         auto& local_lock_case = std::get<2>(local);
     623                 :            : 
     624                 :            :         // cppcheck-suppress unreadVariable
     625                 :       4484 :         auto local_needs_refining_orig = local_needs_refining;
     626                 :            :         // cppcheck-suppress unreadVariable
     627                 :       4484 :         auto local_needs_derefining_orig = local_needs_derefining;
     628                 :            :         // cppcheck-suppress unreadVariable
     629                 :       4484 :         auto local_lock_case_orig = local_lock_case;
     630                 :            : 
     631 [ -  + ][ -  - ]:       4484 :         Assert( !(local_lock_case > unlocked && local_needs_refining),
         [ -  - ][ -  - ]
                 [ -  - ]
     632                 :            :                 "Invalid local edge: locked & needs refining" );
     633 [ -  + ][ -  - ]:       4484 :         Assert( !(remote_lock_case > unlocked && remote_needs_refining),
         [ -  - ][ -  - ]
                 [ -  - ]
     634                 :            :                 "Invalid remote edge: locked & needs refining" );
     635 [ +  + ][ -  + ]:       4484 :         Assert( !(local_needs_derefining == 1 && local_needs_refining > 0),
         [ -  - ][ -  - ]
                 [ -  - ]
     636                 :            :                 "Invalid local edge: needs refining and derefining" );
     637                 :            : 
     638                 :            :         // The parallel compatibility (par-compat) algorithm
     639                 :            : 
     640                 :            :         // compute lock from local and remote locks as most restrictive
     641                 :       4484 :         local_lock_case = std::max( local_lock_case, remote_lock_case );
     642                 :            : 
     643         [ -  + ]:       4484 :         if (local_lock_case > unlocked) {
     644                 :          0 :           local_needs_refining = 0;
     645         [ -  - ]:          0 :           if (local_needs_refining != local_needs_refining_orig ||
     646         [ -  - ]:          0 :             local_lock_case != local_lock_case_orig)
     647                 :          0 :             ++ncorrcase[0];
     648                 :            :         }
     649                 :            : 
     650                 :            :         // Possible combinations of remote-local ref-deref decisions
     651                 :            :         // rows 1, 5, 9: no action needed.
     652                 :            :         // rows 4, 7, 8: no action on local-side; comm needed.
     653                 :            :         //
     654                 :            :         //    LOCAL          |        REMOTE    |  Result
     655                 :            :         // 1  d              |        d         |  d
     656                 :            :         // 2  d              |        -         |  -
     657                 :            :         // 3  d              |        r         |  r
     658                 :            :         // 4  -              |        d         |  -
     659                 :            :         // 5  -              |        -         |  -
     660                 :            :         // 6  -              |        r         |  r
     661                 :            :         // 7  r              |        d         |  r
     662                 :            :         // 8  r              |        -         |  r
     663                 :            :         // 9  r              |        r         |  r
     664                 :            : 
     665                 :            :         // Rows 3, 6
     666                 :            :         // If remote wants to refine
     667         [ +  + ]:       4484 :         if (remote_needs_refining == 1) {
     668         [ +  - ]:       2198 :           if (local_lock_case == unlocked) {
     669                 :       2198 :             local_needs_refining = 1;
     670                 :       2198 :             local_needs_derefining = false;
     671         [ +  - ]:       2198 :             if (local_needs_refining != local_needs_refining_orig ||
     672         [ -  + ]:       2198 :               local_needs_derefining != local_needs_derefining_orig)
     673                 :          0 :               ++ncorrcase[1];
     674                 :            :           }
     675                 :            :           else {
     676                 :          0 :            ++nlocked;
     677                 :            :           }
     678                 :            :         }
     679                 :            : 
     680                 :            :         // Row 2
     681                 :            :         // If remote neither wants to refine nor derefine
     682 [ +  + ][ +  + ]:       4484 :         if (remote_needs_refining == 0 && remote_needs_derefining == false) {
     683                 :        126 :           local_needs_derefining = false;
     684         [ -  + ]:        126 :           if (local_needs_derefining != local_needs_derefining_orig)
     685                 :          0 :             ++ncorrcase[2];
     686                 :            :         }
     687                 :            : 
     688                 :            :         // Row 1: special case
     689                 :            :         // If remote wants to deref-ref (either of 8:4, 8:2, 4:2)
     690                 :            :         // and local does not want to refine (neither pure ref nor deref-ref)
     691 [ -  + ][ -  - ]:       4484 :         if (remote_needs_refining == 2 && local_needs_refining == 0) {
     692         [ -  - ]:          0 :           if (local_lock_case == unlocked) {
     693                 :          0 :             local_needs_refining = 1;
     694                 :          0 :             local_needs_derefining = false;
     695         [ -  - ]:          0 :             if (local_needs_refining != local_needs_refining_orig ||
     696         [ -  - ]:          0 :               local_needs_derefining != local_needs_derefining_orig)
     697                 :          0 :               ++ncorrcase[3];
     698                 :            :           }
     699                 :            :           else {
     700                 :            :             // cppcheck-suppress unreadVariable
     701                 :          0 :             ++nlocked;
     702                 :            :           }
     703                 :            :         }
     704                 :            : 
     705                 :            :         // Rows 4, 7, 8
     706                 :            : 
     707                 :            :         // if the remote sent us data that makes us change our local state,
     708                 :            :         // e.g., local{-,0} + remote{r,0} -> local{r,0}, i.e., I changed my
     709                 :            :         // state I need to tell the world about it
     710         [ +  - ]:       4484 :         if (local_lock_case != local_lock_case_orig ||
     711         [ +  - ]:       4484 :             local_needs_refining != local_needs_refining_orig ||
     712         [ -  + ]:       4484 :             local_needs_derefining != local_needs_derefining_orig)
     713                 :            :         {
     714         [ -  - ]:          0 :           auto l1 = tk::cref_find( m_lid, edge[0] );
     715         [ -  - ]:          0 :           auto l2 = tk::cref_find( m_lid, edge[1] );
     716 [ -  - ][ -  - ]:          0 :           Assert( l1 != l2, "Edge end-points local ids are the same" );
         [ -  - ][ -  - ]
     717                 :          0 :           auto r1 = m_rid[ l1 ];
     718                 :          0 :           auto r2 = m_rid[ l2 ];
     719 [ -  - ][ -  - ]:          0 :           Assert( r1 != r2, "Edge end-points refiner ids are the same" );
         [ -  - ][ -  - ]
     720                 :            :           //std::cout << thisIndex << ": " << r1 << ", " << r2 << std::endl;
     721                 :            :           //if (m_refiner.tet_store.edge_store.get(AMR::edge_t(r1,r2)).lock_case > local_lock_case) {
     722                 :            :           //  std::cout << thisIndex << ": edge " << r1 << "-" << r2 <<
     723                 :            :           //    "; prev=" << local_lock_case_orig <<
     724                 :            :           //    "; new=" << local_lock_case <<
     725                 :            :           //    "; amr-lib=" << m_refiner.tet_store.edge_store.get(AMR::edge_t(r1,r2)).lock_case <<
     726                 :            :           //    " | parcompatiter " << m_ncit << std::endl;
     727                 :            :           //}
     728         [ -  - ]:          0 :            extra[ {{ std::min(r1,r2), std::max(r1,r2) }} ] =
     729                 :          0 :              { local_needs_refining, local_needs_derefining, local_lock_case };
     730                 :          0 :         }
     731                 :            :         // or if the remote data is inconsistent with what I think, e.g.,
     732                 :            :         // local{r,0} + remote{-,0} -> local{r,0}, i.e., the remote does not
     733                 :            :         // yet agree, so another par-compat iteration will be pursued. but
     734                 :            :         // I do not have to locally run ref-compat.
     735         [ +  - ]:       4484 :         else if (local_lock_case != remote_lock_case ||
     736         [ +  - ]:       4484 :           local_needs_refining != remote_needs_refining ||
     737         [ -  + ]:       4484 :           local_needs_derefining != remote_needs_derefining)
     738                 :            :         {
     739                 :          0 :           ++neigh_extra;
     740                 :            :         }
     741                 :            :       }
     742                 :            :     }
     743                 :            :   }
     744                 :            : 
     745                 :         71 :   m_remoteEdgeData.clear();
     746                 :         71 :   m_extra = extra.size();
     747                 :            :   //std::cout << thisIndex << ": amr correction reqd for nedge: " << m_extra << std::endl;
     748                 :            :   //std::cout << thisIndex << ": amr correction reqd for neighbor edges: " << neigh_extra << std::endl;
     749                 :            :   //std::cout << thisIndex << ": edge counts by correction case: " << ncorrcase[0]
     750                 :            :   //  << ", " << ncorrcase[1] << ", " << ncorrcase[2] << ", " << ncorrcase[3] << std::endl;
     751                 :            :   //std::cout << thisIndex << ": locked edges that req corr: " << nlocked << std::endl;
     752                 :            : 
     753         [ -  + ]:         71 :   if (!extra.empty()) {
     754                 :            :     //std::cout << thisIndex << ": redoing markings" << std::endl;
     755                 :            :     // Do refinement compatibility (ref-compat) for edges that need correction
     756         [ -  - ]:          0 :     m_refiner.mark_error_refinement_corr( extra );
     757                 :          0 :     ++m_ncit;
     758                 :            :     // Update our extra-edge store based on refiner
     759         [ -  - ]:          0 :     updateEdgeData();
     760                 :          0 :     m_remoteEdges.clear();
     761                 :            :   }
     762         [ +  - ]:         71 :   else if (neigh_extra == 0) {
     763                 :         71 :     m_ncit = 0;
     764                 :            :   }
     765                 :            : 
     766                 :            :   // Aggregate number of extra edges that still need correction and some
     767                 :            :   // refinement/derefinement statistics
     768                 :         71 :   const auto& tet_store = m_refiner.tet_store;
     769                 :            :   std::vector< std::size_t >
     770                 :         71 :     m{ m_meshid,
     771                 :         71 :        m_extra,
     772                 :         71 :        tet_store.marked_refinements.size(),
     773                 :         71 :        tet_store.marked_derefinements.size(),
     774         [ +  - ]:         71 :        static_cast< std::underlying_type_t< RefMode > >( m_mode ) };
     775         [ +  - ]:         71 :   contribute( m, CkReduction::sum_ulong, m_cbr.get< tag::matched >() );
     776                 :         71 : }
     777                 :            : 
     778                 :            : void
     779                 :        142 : Refiner::updateEdgeData()
     780                 :            : // *****************************************************************************
     781                 :            : // Query AMR lib and update our local store of edge data
     782                 :            : // *****************************************************************************
     783                 :            : {
     784                 :        142 :   m_localEdgeData.clear();
     785                 :        142 :   m_intermediates.clear();
     786                 :            : 
     787                 :            :   // This currently takes ALL edges from the AMR lib, i.e., on the whole
     788                 :            :   // domain. We should eventually only collect edges here that are shared with
     789                 :            :   // other chares.
     790                 :        142 :   const auto& ref_edges = m_refiner.tet_store.edge_store.edges;
     791                 :        142 :   const auto& refinpoel = m_refiner.tet_store.get_active_inpoel();
     792                 :            : 
     793         [ +  + ]:     145750 :   for (std::size_t e=0; e<refinpoel.size()/4; ++e) {
     794                 :     145608 :     auto A = refinpoel[e*4+0];
     795                 :     145608 :     auto B = refinpoel[e*4+1];
     796                 :     145608 :     auto C = refinpoel[e*4+2];
     797                 :     145608 :     auto D = refinpoel[e*4+3];
     798                 :            :     std::array<Edge,6> edges{{ {{A,B}}, {{B,C}}, {{A,C}},
     799                 :     145608 :                                {{A,D}}, {{B,D}}, {{C,D}} }};
     800         [ +  + ]:    1019256 :     for (const auto& ed : edges) {
     801                 :     873648 :       auto ae = AMR::edge_t{{{ std::min(ed[0],ed[1]), std::max(ed[0],ed[1]) }}};
     802         [ +  - ]:     873648 :       auto r = tk::cref_find( ref_edges, ae );
     803         [ +  - ]:     873648 :       const auto ged = Edge{{ m_gid[ tk::cref_find( m_lref, ed[0] ) ],
     804         [ +  - ]:     873648 :                               m_gid[ tk::cref_find( m_lref, ed[1] ) ] }};
     805         [ +  - ]:     873648 :       m_localEdgeData[ ged ] =
     806                 :    1747296 :         { r.needs_refining, r.needs_derefining, r.lock_case };
     807                 :            :     }
     808                 :            :   }
     809                 :            : 
     810                 :            :   // Build intermediates to send. This currently takes ALL intermediates from
     811                 :            :   // the AMR lib, i.e., on the whole domain. We should eventually only collect
     812                 :            :   // edges here that are shared with other chares.
     813         [ +  + ]:        746 :   for (const auto& r : m_refiner.tet_store.intermediate_list) {
     814 [ +  - ][ +  - ]:        604 :     m_intermediates.insert( m_gid[ tk::cref_find( m_lref, r ) ] );
     815                 :            :   }
     816                 :        142 : }
     817                 :            : 
     818                 :            : void
     819                 :         58 : Refiner::updateBndEdgeData()
     820                 :            : // *****************************************************************************
     821                 :            : // Query AMR lib and update our local store of boundary edge data
     822                 :            : // *****************************************************************************
     823                 :            : {
     824                 :            :   // This currently takes ALL edges from the AMR lib, i.e., on the whole
     825                 :            :   // domain. We should eventually only collect edges here that are shared with
     826                 :            :   // other chares.
     827                 :         58 :   const auto& ref_edges = m_refiner.tet_store.edge_store.edges;
     828                 :         58 :   const auto& refinpoel = m_refiner.tet_store.get_active_inpoel();
     829                 :            : 
     830         [ +  + ]:      48163 :   for (std::size_t e=0; e<refinpoel.size()/4; ++e) {
     831                 :      48105 :     auto A = refinpoel[e*4+0];
     832                 :      48105 :     auto B = refinpoel[e*4+1];
     833                 :      48105 :     auto C = refinpoel[e*4+2];
     834                 :      48105 :     auto D = refinpoel[e*4+3];
     835                 :            :     std::array<Edge,6> edges{{ {{A,B}}, {{B,C}}, {{A,C}},
     836                 :      48105 :                                {{A,D}}, {{B,D}}, {{C,D}} }};
     837         [ +  + ]:     336735 :     for (const auto& ed : edges) {
     838                 :     288630 :       auto ae = AMR::edge_t{{{ std::min(ed[0],ed[1]), std::max(ed[0],ed[1]) }}};
     839         [ +  - ]:     288630 :       auto r = tk::cref_find( ref_edges, ae );
     840         [ +  - ]:     288630 :       const auto ged = Edge{{ m_gid[ tk::cref_find( m_lref, ed[0] ) ],
     841         [ +  - ]:     288630 :                               m_gid[ tk::cref_find( m_lref, ed[1] ) ] }};
     842                 :            :       // only update edges that are on chare boundary OR unlocked
     843 [ +  - ][ +  - ]:     577260 :       if (m_localEdgeData.find(ged) == m_localEdgeData.end() ||
                 [ +  - ]
     844 [ +  - ][ +  - ]:     288630 :         std::get<2>(m_localEdgeData[ged]) == AMR::Edge_Lock_Case::unlocked) {
     845         [ +  - ]:     288630 :         m_localEdgeData[ ged ] = { r.needs_refining, r.needs_derefining,
     846                 :     577260 :           r.lock_case };
     847                 :            :       }
     848                 :            :     }
     849                 :            :   }
     850                 :         58 : }
     851                 :            : 
     852                 :            : std::tuple< std::vector< std::string >,
     853                 :            :             std::vector< std::vector< tk::real > >,
     854                 :            :             std::vector< std::string >,
     855                 :            :             std::vector< std::vector< tk::real > > >
     856                 :         98 : Refiner::refinementFields() const
     857                 :            : // *****************************************************************************
     858                 :            : //  Collect mesh output fields from refiner lib
     859                 :            : //! \return Names and fields of mesh refinement data in mesh cells and nodes
     860                 :            : // *****************************************************************************
     861                 :            : {
     862                 :            :   // Find number of nodes in current mesh
     863         [ +  - ]:         98 :   auto npoin = tk::npoin_in_graph( m_inpoel );
     864                 :            :   // Generate edges surrounding points in current mesh
     865         [ +  - ]:         98 :   auto esup = tk::genEsup( m_inpoel, 4 );
     866                 :            : 
     867                 :            :   // Update solution on current mesh
     868         [ +  - ]:         98 :   const auto& u = solution( npoin, esup );
     869 [ -  + ][ -  - ]:         98 :   Assert( u.nunk() == npoin, "Solution uninitialized or wrong size" );
         [ -  - ][ -  - ]
     870                 :            : 
     871                 :            :   // Compute error in edges on current mesh
     872         [ +  - ]:         98 :   auto edgeError = errorsInEdges( npoin, esup, u );
     873                 :            : 
     874                 :            :   // Transfer error from edges to cells for field output
     875         [ +  - ]:         98 :   std::vector< tk::real > error( m_inpoel.size()/4, 0.0 );
     876         [ +  + ]:     133213 :   for (std::size_t e=0; e<m_inpoel.size()/4; ++e) {
     877                 :     133115 :     auto A = m_inpoel[e*4+0];
     878                 :     133115 :     auto B = m_inpoel[e*4+1];
     879                 :     133115 :     auto C = m_inpoel[e*4+2];
     880                 :     133115 :     auto D = m_inpoel[e*4+3];
     881                 :            :     std::array<Edge,6> edges{{ {{A,B}}, {{B,C}}, {{A,C}},
     882                 :     133115 :                                {{A,D}}, {{B,D}}, {{C,D}} }};
     883                 :            :     // sum error from edges to elements
     884 [ +  - ][ +  + ]:     931805 :     for (const auto& ed : edges) error[e] += tk::cref_find( edgeError, ed );
     885                 :     133115 :     error[e] /= 6.0;    // assign edge-average error to element
     886                 :            :   }
     887                 :            : 
     888                 :            :   // Prepare element fields with mesh refinement data
     889                 :            :   std::vector< std::string >
     890 [ +  - ][ +  + ]:        392 :     elemfieldnames{ "refinement level", "cell type", "error" };
                 [ -  - ]
     891                 :         98 :   auto& tet_store = m_refiner.tet_store;
     892                 :            :   std::vector< std::vector< tk::real > > elemfields{
     893                 :            :     tet_store.get_refinement_level_list(),
     894                 :            :     tet_store.get_cell_type_list(),
     895 [ +  - ][ +  + ]:        392 :     error };
                 [ -  - ]
     896                 :            : 
     897                 :            :   using tuple_t = std::tuple< std::vector< std::string >,
     898                 :            :                               std::vector< std::vector< tk::real > >,
     899                 :            :                               std::vector< std::string >,
     900                 :            :                               std::vector< std::vector< tk::real > > >;
     901         [ +  - ]:        196 :   return tuple_t{ elemfieldnames, elemfields, {}, {} };
     902 [ +  - ][ +  - ]:        294 : }
         [ +  - ][ +  - ]
         [ +  - ][ +  - ]
         [ -  - ][ -  - ]
         [ -  - ][ -  - ]
     903                 :            : 
     904                 :            : void
     905                 :         98 : Refiner::writeMesh( const std::string& basefilename,
     906                 :            :                     uint64_t itr,
     907                 :            :                     tk::real t,
     908                 :            :                     CkCallback c ) const
     909                 :            : // *****************************************************************************
     910                 :            : //  Output mesh to file(s)
     911                 :            : //! \param[in] basefilename File name to append to
     912                 :            : //! \param[in] itr Iteration count since a new mesh
     913                 :            : //! \param[in] t "Physical time" to write to file. "Time" here is used to
     914                 :            : //!   designate a new time step at which the mesh is saved.
     915                 :            : //! \param[in] c Function to continue with after the write
     916                 :            : // *****************************************************************************
     917                 :            : {
     918         [ +  - ]:         98 :   auto r = refinementFields();
     919                 :         98 :   auto& elemfieldnames = std::get< 0 >( r );
     920                 :         98 :   auto& elemfields = std::get< 1 >( r );
     921                 :         98 :   auto& nodefieldnames = std::get< 2 >( r );
     922                 :         98 :   auto& nodefields = std::get< 3 >( r );
     923                 :            : 
     924                 :            :   // Prepare solution field names: depvar + component id for all eqs
     925                 :         98 :   auto ncomp = g_cfg.get< tag::problem_ncomp >();
     926 [ +  - ][ +  - ]:         98 :   std::vector< std::string > solfieldnames( ncomp, "u" );
     927 [ -  + ][ -  - ]:         98 :   Assert( solfieldnames.size() == ncomp, "Size mismatch" );
         [ -  - ][ -  - ]
     928                 :            : 
     929                 :         98 :   auto t0 = g_cfg.get< tag::t0 >();
     930                 :            : 
     931                 :            :   // Augment element field names with solution variable names + field ids
     932                 :            :   //nodefieldnames.insert( end(nodefieldnames),
     933                 :            :   //                       begin(solfieldnames), end(solfieldnames) );
     934 [ +  - ][ +  - ]:         98 :   nodefieldnames.push_back( "c1" );
     935                 :            : 
     936                 :            :   // Evaluate initial conditions on current mesh at t0
     937         [ +  - ]:         98 :   tk::Fields u( m_coord[0].size(), ncomp );
     938         [ +  - ]:         98 :   problems::initialize( m_coord, u, t0 );
     939                 :            : 
     940                 :            :   // Extract all scalar components from solution for output to file
     941                 :            :   //for (std::size_t i=0; i<ncomp; ++i)
     942 [ +  - ][ +  - ]:         98 :   nodefields.push_back( u.extract( 5 ) );
     943                 :            : 
     944                 :            :   // Output mesh
     945                 :            :   m_meshwriter[ CkNodeFirst( CkMyNode() ) ].
     946         [ +  - ]:        196 :     write( m_meshid, /*meshoutput = */ true, /*fieldoutput = */ true, itr, 1, t,
     947         [ +  - ]:         98 :            thisIndex, basefilename, m_inpoel, m_coord, m_bface,
     948 [ +  - ][ +  - ]:        196 :            tk::remap(m_bnode,m_lid), tk::remap(m_triinpoel,m_lid),
     949                 :            :            elemfieldnames, nodefieldnames, {}, {}, elemfields, nodefields, {},
     950                 :            :            {}, {}, c );
     951                 :         98 : }
     952                 :            : 
     953                 :            : void
     954                 :         71 : Refiner::perform()
     955                 :            : // *****************************************************************************
     956                 :            : // Perform mesh refinement and decide how to continue
     957                 :            : //! \details First the mesh refiner object is called to perform a single step
     958                 :            : //!   of mesh refinement. Then, if this function is called during a step
     959                 :            : //!   (potentially multiple levels of) initial AMR, it evaluates whether to do
     960                 :            : //!   another one. If it is called during time stepping, this concludes the
     961                 :            : //!   single mesh refinement step and the new mesh is sent to the PDE worker
     962                 :            : //!   (Discretization).
     963                 :            : // *****************************************************************************
     964                 :            : {
     965                 :            :   // Save old tets and their ids before performing refinement. Outref is always
     966                 :            :   // followed by outderef, so to the outside world, the mesh is uchanged, thus
     967                 :            :   // no update.
     968                 :         71 :   m_oldTets.clear();
     969         [ +  + ]:      82257 :   for (const auto& [ id, tet ] : m_refiner.tet_store.tets) {
     970         [ +  - ]:      82186 :     m_oldTets.insert( tet );
     971                 :            :   }
     972                 :         71 :   m_oldntets = m_oldTets.size();
     973                 :            : 
     974         [ +  - ]:         71 :   if (m_mode == RefMode::T0REF) {
     975                 :            : 
     976                 :            :     // Refine mesh based on next initial refinement type
     977         [ +  - ]:         71 :     if (!m_initref.empty()) {
     978         [ +  - ]:         71 :       auto r = m_initref.back();    // consume (reversed) list from its back
     979         [ +  + ]:         71 :       if (r == "uniform_deref")
     980         [ +  - ]:         24 :         m_refiner.perform_derefinement();
     981                 :            :       else
     982         [ +  - ]:         47 :         m_refiner.perform_refinement();
     983                 :         71 :     }
     984                 :            : 
     985                 :            :   } else {
     986                 :            : 
     987                 :            :     // TODO: does not work yet, fix as above
     988                 :          0 :     m_refiner.perform_refinement();
     989                 :          0 :     m_refiner.perform_derefinement();
     990                 :            :   }
     991                 :            : 
     992                 :            :   // Remove temporary edge-locks resulting from the parallel compatibility
     993                 :         71 :   m_refiner.remove_edge_locks(1);
     994                 :         71 :   m_refiner.remove_edge_temp_locks();
     995                 :            : 
     996                 :            :   //auto& tet_store = m_refiner.tet_store;
     997                 :            :   //std::cout << "before ref: " << tet_store.marked_refinements.size() << ", " << tet_store.marked_derefinements.size() << ", " << tet_store.size() << ", " << tet_store.get_active_inpoel().size() << '\n';
     998                 :            :   //std::cout << "after ref: " << tet_store.marked_refinements.size() << ", " << tet_store.marked_derefinements.size() << ", " << tet_store.size() << ", " << tet_store.get_active_inpoel().size() << '\n';
     999                 :            :   //std::cout << "after deref: " << tet_store.marked_refinements.size() << ", " << tet_store.marked_derefinements.size() << ", " << tet_store.size() << ", " << tet_store.get_active_inpoel().size() << '\n';
    1000                 :            : 
    1001                 :            :   // Update volume and boundary mesh
    1002                 :         71 :   updateMesh();
    1003                 :            : 
    1004                 :            :   // Save mesh at every initial refinement step (mainly for debugging). Will
    1005                 :            :   // replace with just a 'next()' in production.
    1006         [ +  - ]:         71 :   if (m_mode == RefMode::T0REF) {
    1007                 :            : 
    1008                 :         71 :     auto l = m_ninitref - m_initref.size() + 1;  // num initref steps completed
    1009                 :         71 :     auto t0 = g_cfg.get< tag::t0 >();
    1010                 :            :     // Generate times equally subdividing t0-1...t0 to ninitref steps
    1011                 :         71 :     auto t =
    1012                 :         71 :       t0 - 1.0 + static_cast<tk::real>(l)/static_cast<tk::real>(m_ninitref);
    1013                 :         71 :     auto itr = l;
    1014                 :            :     // Output mesh after refinement step
    1015 [ +  - ][ +  - ]:         71 :     writeMesh( "t0ref", itr, t,
    1016 [ +  - ][ +  - ]:        142 :                CkCallback( CkIndex_Refiner::next(), thisProxy[thisIndex] ) );
                 [ +  - ]
    1017                 :            : 
    1018                 :            :   } else {
    1019                 :            : 
    1020                 :          0 :     next();
    1021                 :            : 
    1022                 :            :   }
    1023                 :         71 : }
    1024                 :            : 
    1025                 :            : void
    1026                 :         71 : Refiner::next()
    1027                 :            : // *****************************************************************************
    1028                 :            : // Continue after finishing a refinement step
    1029                 :            : // *****************************************************************************
    1030                 :            : {
    1031         [ +  - ]:         71 :   if (m_mode == RefMode::T0REF) {
    1032                 :            : 
    1033                 :            :     // Remove initial mesh refinement step from list
    1034         [ +  - ]:         71 :     if (!m_initref.empty()) m_initref.pop_back();
    1035                 :            :     // Continue to next initial AMR step or finish
    1036         [ +  + ]:         71 :     if (!m_initref.empty()) t0ref(); else endt0ref();
    1037                 :            : 
    1038         [ -  - ]:          0 :   } else if (m_mode == RefMode::DTREF) {
    1039                 :            : 
    1040                 :            :     // Send new mesh, solution, and communication data back to PDE worker
    1041                 :          0 :     const auto& solver = g_cfg.get< tag::solver >();
    1042         [ -  - ]:          0 :     if (solver == "riecg") {
    1043 [ -  - ][ -  - ]:          0 :       m_riecg[ thisIndex ].ckLocal()->resizePostAMR( m_ginpoel,
    1044                 :          0 :         m_el, m_coord, m_addedNodes, m_addedTets, m_removedNodes,
    1045         [ -  - ]:          0 :         m_nodeCommMap, m_bface, m_bnode, m_triinpoel );
    1046                 :            :     }
    1047         [ -  - ]:          0 :     else if (solver == "laxcg") {
    1048 [ -  - ][ -  - ]:          0 :       m_laxcg[ thisIndex ].ckLocal()->resizePostAMR( m_ginpoel,
    1049                 :          0 :         m_el, m_coord, m_addedNodes, m_addedTets, m_removedNodes,
    1050         [ -  - ]:          0 :         m_nodeCommMap, m_bface, m_bnode, m_triinpoel );
    1051                 :            :     }
    1052         [ -  - ]:          0 :     else if (solver == "zalcg") {
    1053 [ -  - ][ -  - ]:          0 :       m_zalcg[ thisIndex ].ckLocal()->resizePostAMR( m_ginpoel,
    1054                 :          0 :         m_el, m_coord, m_addedNodes, m_addedTets, m_removedNodes,
    1055         [ -  - ]:          0 :         m_nodeCommMap, m_bface, m_bnode, m_triinpoel );
    1056                 :            :     }
    1057         [ -  - ]:          0 :     else if (solver == "kozcg") {
    1058 [ -  - ][ -  - ]:          0 :       m_kozcg[ thisIndex ].ckLocal()->resizePostAMR( m_ginpoel,
    1059                 :          0 :         m_el, m_coord, m_addedNodes, m_addedTets, m_removedNodes,
    1060         [ -  - ]:          0 :         m_nodeCommMap, m_bface, m_bnode, m_triinpoel );
    1061                 :            :     }
    1062         [ -  - ]:          0 :     else if (solver == "chocg") {
    1063 [ -  - ][ -  - ]:          0 :       m_chocg[ thisIndex ].ckLocal()->resizePostAMR( m_ginpoel,
    1064                 :          0 :         m_el, m_coord, m_addedNodes, m_addedTets, m_removedNodes,
    1065         [ -  - ]:          0 :         m_nodeCommMap, m_bface, m_bnode, m_triinpoel );
    1066                 :            :     }
    1067         [ -  - ]:          0 :     else if (solver == "lohcg") {
    1068 [ -  - ][ -  - ]:          0 :       m_lohcg[ thisIndex ].ckLocal()->resizePostAMR( m_ginpoel,
    1069                 :          0 :         m_el, m_coord, m_addedNodes, m_addedTets, m_removedNodes,
    1070         [ -  - ]:          0 :         m_nodeCommMap, m_bface, m_bnode, m_triinpoel );
    1071                 :            :     }
    1072                 :            :     else {
    1073 [ -  - ][ -  - ]:          0 :       Throw( "Unknown solver: " + solver );
                 [ -  - ]
    1074                 :            :     }
    1075                 :            : 
    1076 [ -  - ][ -  - ]:          0 :   } else Throw( "RefMode not implemented" );
                 [ -  - ]
    1077                 :         71 : }
    1078                 :            : 
    1079                 :            : void
    1080                 :       2475 : Refiner::endt0ref()
    1081                 :            : // *****************************************************************************
    1082                 :            : // Finish initial mesh refinement
    1083                 :            : //! \details This function is called as after initial mesh refinement has
    1084                 :            : //!   finished. If initial mesh reifnement was not configured by the user, this
    1085                 :            : //!   is the point where we continue after the constructor, by computing the
    1086                 :            : //!   total number of elements across the whole problem.
    1087                 :            : // *****************************************************************************
    1088                 :            : {
    1089                 :            :   // create sorter Charm++ chare array elements using dynamic insertion
    1090         [ +  - ]:       2475 :   m_sorter[ thisIndex ].insert( m_meshid, m_host, m_meshwriter, m_cbs,
    1091                 :       2475 :     m_disc, m_riecg, m_laxcg, m_zalcg, m_kozcg, m_chocg, m_lohcg, m_cgpre,
    1092         [ +  - ]:       2475 :     m_cgmom,
    1093 [ +  - ][ +  - ]:       4950 :     CkCallback(CkIndex_Refiner::reorder(), thisProxy[thisIndex]), m_ginpoel,
    1094         [ +  - ]:       2475 :     m_coordmap, m_el, m_bface, m_triinpoel, m_bnode, m_nchare );
    1095                 :            : 
    1096                 :            :   // Compute final number of cells across whole problem
    1097                 :            :   std::vector< std::size_t >
    1098         [ +  - ]:       2475 :     meshdata{ m_meshid, m_ginpoel.size()/4, m_coord[0].size() };
    1099         [ +  - ]:       2475 :   contribute( meshdata, CkReduction::sum_ulong, m_cbr.get< tag::refined >() );
    1100                 :            : 
    1101                 :            :   // Free up memory if no dtref
    1102         [ +  - ]:       2475 :   if (!g_cfg.get< tag::href_dt >()) {
    1103                 :       2475 :     tk::destroy( m_ginpoel );
    1104                 :       2475 :     tk::destroy( m_el );
    1105                 :       2475 :     tk::destroy( m_coordmap );
    1106                 :       2475 :     tk::destroy( m_coord );
    1107                 :       2475 :     tk::destroy( m_bface );
    1108                 :       2475 :     tk::destroy( m_bnode );
    1109                 :       2475 :     tk::destroy( m_triinpoel );
    1110                 :       2475 :     tk::destroy( m_initref );
    1111                 :       2475 :     tk::destroy( m_ch );
    1112                 :       2475 :     tk::destroy( m_edgech );
    1113                 :       2475 :     tk::destroy( m_chedge );
    1114                 :       2475 :     tk::destroy( m_localEdgeData );
    1115                 :       2475 :     tk::destroy( m_remoteEdgeData );
    1116                 :       2475 :     tk::destroy( m_remoteEdges );
    1117                 :       2475 :     tk::destroy( m_intermediates );
    1118                 :       2475 :     tk::destroy( m_nodeCommMap );
    1119                 :       2475 :     tk::destroy( m_oldTets );
    1120                 :       2475 :     tk::destroy( m_addedNodes );
    1121                 :       2475 :     tk::destroy( m_addedTets );
    1122                 :       2475 :     tk::destroy( m_coarseBndFaces );
    1123                 :       2475 :     tk::destroy( m_coarseBndNodes );
    1124                 :       2475 :     tk::destroy( m_rid );
    1125                 :            :     //tk::destroy( m_oldrid );
    1126                 :       2475 :     tk::destroy( m_lref );
    1127                 :            :   }
    1128                 :       2475 : }
    1129                 :            : 
    1130                 :            : void
    1131                 :         31 : Refiner::uniformRefine()
    1132                 :            : // *****************************************************************************
    1133                 :            : // Do uniform mesh refinement
    1134                 :            : // *****************************************************************************
    1135                 :            : {
    1136                 :            :   // Do uniform refinement
    1137                 :         31 :   m_refiner.mark_uniform_refinement();
    1138                 :            : 
    1139                 :            :   // Update our extra-edge store based on refiner
    1140                 :         31 :   updateEdgeData();
    1141                 :            : 
    1142                 :            :   // Set number of extra edges to be zero, skipping correction (if this is the
    1143                 :            :   // only step in this refinement step)
    1144                 :         31 :   m_extra = 0;
    1145                 :         31 : }
    1146                 :            : 
    1147                 :            : void
    1148                 :         24 : Refiner::uniformDeRefine()
    1149                 :            : // *****************************************************************************
    1150                 :            : // Do uniform mesh derefinement
    1151                 :            : // *****************************************************************************
    1152                 :            : {
    1153                 :            :   // Do uniform derefinement
    1154                 :         24 :   m_refiner.mark_uniform_derefinement();
    1155                 :            : 
    1156                 :            :   // Update our extra-edge store based on refiner
    1157                 :         24 :   updateEdgeData();
    1158                 :            : 
    1159                 :            :   // Set number of extra edges to be zero, skipping correction (if this is the
    1160                 :            :   // only step in this refinement step)
    1161                 :         24 :   m_extra = 0;
    1162                 :         24 : }
    1163                 :            : 
    1164                 :            : Refiner::EdgeError
    1165                 :        114 : Refiner::errorsInEdges(
    1166                 :            :   std::size_t npoin,
    1167                 :            :   const std::pair< std::vector<std::size_t>, std::vector<std::size_t> >& esup,
    1168                 :            :   const tk::Fields& u ) const
    1169                 :            : // *****************************************************************************
    1170                 :            : //  Compute errors in edges
    1171                 :            : //! \param[in] npoin Number nodes in current mesh (partition)
    1172                 :            : //! \param[in] esup Elements surrounding points linked vectors
    1173                 :            : //! \param[in] u Solution evaluated at mesh nodes for all scalar components
    1174                 :            : //! \return A map associating errors (real values between 0.0 and 1.0 incusive)
    1175                 :            : //!   to edges (2 local node IDs)
    1176                 :            : // *****************************************************************************
    1177                 :            : {
    1178         [ +  - ]:        114 :   auto errtype = g_cfg.get< tag::href_error >();
    1179                 :        114 :   const auto& refvar = g_cfg.get< tag::href_refvar >();
    1180         [ +  - ]:        114 :   auto psup = tk::genPsup( m_inpoel, 4, esup );
    1181                 :            : 
    1182                 :            :   // Compute errors in ICs and define refinement criteria for edges
    1183                 :            :   AMR::Error error;
    1184                 :        114 :   EdgeError edgeError;
    1185                 :            : 
    1186         [ +  + ]:      35211 :   for (std::size_t p=0; p<npoin; ++p) { // for all mesh nodes on this chare
    1187         [ +  + ]:     422029 :     for (auto q : tk::Around(psup,p)) { // for all nodes surrounding p
    1188                 :     386932 :       tk::real cmax = 0.0;
    1189         [ +  - ]:     386932 :       AMR::edge_t e(p,q);
    1190         [ +  + ]:     549892 :       for (auto i : refvar) {          // for all refinement variables
    1191         [ +  - ]:     162960 :         auto c = error.scalar( u, e, i-1, m_coord, m_inpoel, esup, errtype );
    1192         [ +  + ]:     162960 :         if (c > cmax) cmax = c;        // find max error at edge
    1193                 :            :       }
    1194         [ +  - ]:     386932 :       edgeError[ {{p,q}} ] = cmax;       // associate error to edge
    1195                 :            :     }
    1196                 :            :   }
    1197                 :            : 
    1198                 :        228 :   return edgeError;
    1199                 :        114 : }
    1200                 :            : 
    1201                 :            : tk::Fields
    1202                 :        114 : Refiner::solution( std::size_t npoin,
    1203                 :            :                    const std::pair< std::vector< std::size_t >,
    1204                 :            :                                     std::vector< std::size_t > >& esup ) const
    1205                 :            : // *****************************************************************************
    1206                 :            : //  Update (or evaluate) solution on current mesh
    1207                 :            : //! \param[in] npoin Number nodes in current mesh (partition)
    1208                 :            : //! \param[in] esup Elements surrounding points linked vectors
    1209                 :            : //! \return Solution updated/evaluated for all scalar components
    1210                 :            : // *****************************************************************************
    1211                 :            : {
    1212                 :            :   // Get solution whose error to evaluate
    1213                 :        114 :   tk::Fields u;
    1214                 :            : 
    1215         [ +  - ]:        114 :   if (m_mode == RefMode::T0REF) {
    1216                 :            : 
    1217                 :            :     // Evaluate initial conditions at mesh nodes
    1218         [ +  - ]:        114 :     u = nodeinit( npoin, esup );
    1219                 :            : 
    1220         [ -  - ]:          0 :   } else if (m_mode == RefMode::DTREF) {
    1221                 :            : 
    1222                 :            :     // Query current solution
    1223                 :          0 :     const auto& solver = g_cfg.get< tag::solver >();
    1224         [ -  - ]:          0 :     if (solver == "riecg") {
    1225 [ -  - ][ -  - ]:          0 :       u = m_riecg[ thisIndex ].ckLocal()->solution();
                 [ -  - ]
    1226                 :            :     }
    1227         [ -  - ]:          0 :     else if (solver == "laxcg") {
    1228 [ -  - ][ -  - ]:          0 :       u = m_laxcg[ thisIndex ].ckLocal()->solution();
                 [ -  - ]
    1229                 :            :     }
    1230         [ -  - ]:          0 :     else if (solver == "zalcg") {
    1231 [ -  - ][ -  - ]:          0 :       u = m_zalcg[ thisIndex ].ckLocal()->solution();
                 [ -  - ]
    1232                 :            :     }
    1233         [ -  - ]:          0 :     else if (solver == "kozcg") {
    1234 [ -  - ][ -  - ]:          0 :       u = m_kozcg[ thisIndex ].ckLocal()->solution();
                 [ -  - ]
    1235                 :            :     }
    1236         [ -  - ]:          0 :     else if (solver == "chocg") {
    1237 [ -  - ][ -  - ]:          0 :       u = m_chocg[ thisIndex ].ckLocal()->solution();
                 [ -  - ]
    1238                 :            :     }
    1239         [ -  - ]:          0 :     else if (solver == "lohcg") {
    1240 [ -  - ][ -  - ]:          0 :       u = m_lohcg[ thisIndex ].ckLocal()->solution();
                 [ -  - ]
    1241                 :            :     }
    1242                 :            :     else {
    1243 [ -  - ][ -  - ]:          0 :       Throw( "Unknown solver: " + solver );
                 [ -  - ]
    1244                 :            :     }
    1245                 :            :  
    1246 [ -  - ][ -  - ]:          0 :   } else Throw( "RefMode not implemented" );
                 [ -  - ]
    1247                 :            : 
    1248                 :        114 :   return u;
    1249                 :          0 : }
    1250                 :            : 
    1251                 :            : void
    1252                 :         16 : Refiner::errorRefine()
    1253                 :            : // *****************************************************************************
    1254                 :            : // Do error-based mesh refinement and derefinement
    1255                 :            : // *****************************************************************************
    1256                 :            : {
    1257                 :            :   // Find number of nodes in old mesh
    1258         [ +  - ]:         16 :   auto npoin = tk::npoin_in_graph( m_inpoel );
    1259                 :            :   // Generate edges surrounding points in old mesh
    1260         [ +  - ]:         16 :   auto esup = tk::genEsup( m_inpoel, 4 );
    1261                 :            : 
    1262                 :            :   // Update solution on current mesh
    1263         [ +  - ]:         16 :   const auto& u = solution( npoin, esup );
    1264 [ -  + ][ -  - ]:         16 :   Assert( u.nunk() == npoin, "Solution uninitialized or wrong size" );
         [ -  - ][ -  - ]
    1265                 :            : 
    1266                 :            :   using AMR::edge_t;
    1267                 :            :   using AMR::edge_tag;
    1268                 :            : 
    1269                 :            :   // Compute error in edges. Tag edge for refinement if error exceeds
    1270                 :            :   // refinement tolerance, tag edge for derefinement if error is below
    1271                 :            :   // derefinement tolerance.
    1272                 :         16 :   auto tolref = 0.2;//g_cfg.get< tag::amr, tag::tolref >();
    1273                 :         16 :   auto tolderef = 0.05;//g_cfg.get< tag::amr, tag::tolderef >();
    1274                 :         16 :   std::vector< std::pair< edge_t, edge_tag > > tagged_edges;
    1275 [ +  - ][ +  + ]:       5551 :   for (const auto& e : errorsInEdges(npoin,esup,u)) {
    1276         [ +  + ]:       5535 :     if (e.second > tolref) {
    1277 [ +  - ][ +  - ]:       7224 :       tagged_edges.push_back( { edge_t( m_rid[e.first[0]], m_rid[e.first[1]] ),
    1278                 :       7224 :                                 edge_tag::REFINE } );
    1279         [ +  + ]:       1923 :     } else if (e.second < tolderef) {
    1280 [ +  - ][ +  - ]:       3464 :       tagged_edges.push_back( { edge_t( m_rid[e.first[0]], m_rid[e.first[1]] ),
    1281                 :       3464 :                                 edge_tag::DEREFINE } );
    1282                 :            :     }
    1283                 :         16 :   }
    1284                 :            : 
    1285                 :            :   // Do error-based refinement
    1286         [ +  - ]:         16 :   m_refiner.mark_error_refinement( tagged_edges );
    1287                 :            : 
    1288                 :            :   // Update our extra-edge store based on refiner
    1289         [ +  - ]:         16 :   updateEdgeData();
    1290                 :            : 
    1291                 :            :   // Set number of extra edges to a nonzero number, triggering correction
    1292                 :         16 :   m_extra = 1;
    1293                 :         16 : }
    1294                 :            : 
    1295                 :            : void
    1296                 :          0 : Refiner::edgelistRefine()
    1297                 :            : // *****************************************************************************
    1298                 :            : // Do mesh refinement based on user explicitly tagging edges
    1299                 :            : // *****************************************************************************
    1300                 :            : {
    1301                 :            :   // Get user-defined node-pairs (edges) to tag for refinement
    1302         [ -  - ]:          0 :   const auto& edgenodelist = std::vector<uint64_t>{0,1};//g_cfg.get< tag::amr, tag::edge >();
    1303                 :            : 
    1304         [ -  - ]:          0 :   if (!edgenodelist.empty()) {  // if user explicitly tagged edges
    1305                 :            :     // Find number of nodes in old mesh
    1306         [ -  - ]:          0 :     auto npoin = tk::npoin_in_graph( m_inpoel );
    1307                 :            :     // Generate edges surrounding points in old mesh
    1308         [ -  - ]:          0 :     auto esup = tk::genEsup( m_inpoel, 4 );
    1309         [ -  - ]:          0 :     auto psup = tk::genPsup( m_inpoel, 4, esup );
    1310                 :            : 
    1311                 :          0 :     EdgeSet useredges;
    1312         [ -  - ]:          0 :     for (std::size_t i=0; i<edgenodelist.size()/2; ++i)
    1313         [ -  - ]:          0 :       useredges.insert( {{ {edgenodelist[i*2+0], edgenodelist[i*2+1]} }} );
    1314                 :            : 
    1315                 :            :     using AMR::edge_t;
    1316                 :            :     using AMR::edge_tag;
    1317                 :            : 
    1318                 :            :     // Tag edges the user configured
    1319                 :          0 :     std::vector< std::pair< edge_t, edge_tag > > tagged_edges;
    1320         [ -  - ]:          0 :     for (std::size_t p=0; p<npoin; ++p)        // for all mesh nodes on this chare
    1321         [ -  - ]:          0 :       for (auto q : tk::Around(psup,p)) {      // for all nodes surrounding p
    1322                 :          0 :         Edge e{{ m_gid[p], m_gid[q] }};
    1323         [ -  - ]:          0 :         auto f = useredges.find(e);
    1324         [ -  - ]:          0 :         if (f != end(useredges)) { // tag edge if on user's list
    1325 [ -  - ][ -  - ]:          0 :           tagged_edges.push_back( { edge_t( m_rid[p], m_rid[q] ),
    1326                 :          0 :                                     edge_tag::REFINE } );
    1327         [ -  - ]:          0 :           useredges.erase( f );
    1328                 :            :         }
    1329                 :            :       }
    1330                 :            : 
    1331                 :            :     // Do error-based refinement
    1332         [ -  - ]:          0 :     m_refiner.mark_error_refinement( tagged_edges );
    1333                 :            : 
    1334                 :            :     // Update our extra-edge store based on refiner
    1335         [ -  - ]:          0 :     updateEdgeData();
    1336                 :            : 
    1337                 :            :     // Set number of extra edges to a nonzero number, triggering correction
    1338                 :          0 :     m_extra = 1;
    1339                 :          0 :   }
    1340                 :          0 : }
    1341                 :            : 
    1342                 :            : void
    1343                 :          0 : Refiner::coordRefine()
    1344                 :            : // *****************************************************************************
    1345                 :            : // Do mesh refinement based on tagging edges based on end-point coordinates
    1346                 :            : // *****************************************************************************
    1347                 :            : {
    1348                 :            :   // Get user-defined half-world coordinates
    1349                 :          0 :   auto xminus = 0.0;
    1350                 :          0 :   auto xplus = 0.0;
    1351                 :          0 :   auto yminus = 0.0;
    1352                 :          0 :   auto yplus = 0.0;
    1353                 :          0 :   auto zminus = 0.0;
    1354                 :          0 :   auto zplus = 0.0;
    1355                 :            : 
    1356                 :            :   // The default is the largest representable double
    1357                 :          0 :   auto eps = 1.0e-12;
    1358                 :          0 :   auto xminus_default = 0.0;
    1359                 :          0 :   auto xplus_default = 0.0;
    1360                 :          0 :   auto yminus_default = 0.0;
    1361                 :          0 :   auto yplus_default = 0.0;
    1362                 :          0 :   auto zminus_default = 0.0;
    1363                 :          0 :   auto zplus_default = 0.0;
    1364                 :            : 
    1365                 :            :   // Decide if user has configured the half-world
    1366                 :          0 :   bool xm = std::abs(xminus - xminus_default) > eps ? true : false;
    1367                 :          0 :   bool xp = std::abs(xplus - xplus_default) > eps ? true : false;
    1368                 :          0 :   bool ym = std::abs(yminus - yminus_default) > eps ? true : false;
    1369                 :          0 :   bool yp = std::abs(yplus - yplus_default) > eps ? true : false;
    1370                 :          0 :   bool zm = std::abs(zminus - zminus_default) > eps ? true : false;
    1371                 :          0 :   bool zp = std::abs(zplus - zplus_default) > eps ? true : false;
    1372                 :            : 
    1373                 :            :   using AMR::edge_t;
    1374                 :            :   using AMR::edge_tag;
    1375                 :            : 
    1376 [ -  - ][ -  - ]:          0 :   if (xm || xp || ym || yp || zm || zp) {       // if any half-world configured
         [ -  - ][ -  - ]
         [ -  - ][ -  - ]
    1377                 :            :     // Find number of nodes in old mesh
    1378         [ -  - ]:          0 :     auto npoin = tk::npoin_in_graph( m_inpoel );
    1379                 :            :     // Generate edges surrounding points in old mesh
    1380         [ -  - ]:          0 :     auto esup = tk::genEsup( m_inpoel, 4 );
    1381         [ -  - ]:          0 :     auto psup = tk::genPsup( m_inpoel, 4, esup );
    1382                 :            :     // Get access to node coordinates
    1383                 :          0 :     const auto& x = m_coord[0];
    1384                 :          0 :     const auto& y = m_coord[1];
    1385                 :          0 :     const auto& z = m_coord[2];
    1386                 :            :     // Compute edges to be tagged for refinement
    1387                 :          0 :     std::vector< std::pair< edge_t, edge_tag > > tagged_edges;
    1388         [ -  - ]:          0 :     for (std::size_t p=0; p<npoin; ++p)    // for all mesh nodes on this chare
    1389         [ -  - ]:          0 :       for (auto q : tk::Around(psup,p)) {  // for all nodes surrounding p
    1390                 :          0 :         Edge e{{p,q}};
    1391                 :            : 
    1392                 :          0 :         bool t = true;
    1393 [ -  - ][ -  - ]:          0 :         if (xm) { if (x[p]>xminus && x[q]>xminus) t = false; }
         [ -  - ][ -  - ]
    1394 [ -  - ][ -  - ]:          0 :         if (xp) { if (x[p]<xplus && x[q]<xplus) t = false; }
         [ -  - ][ -  - ]
    1395 [ -  - ][ -  - ]:          0 :         if (ym) { if (y[p]>yminus && y[q]>yminus) t = false; }
         [ -  - ][ -  - ]
    1396 [ -  - ][ -  - ]:          0 :         if (yp) { if (y[p]<yplus && y[q]<yplus) t = false; }
         [ -  - ][ -  - ]
    1397 [ -  - ][ -  - ]:          0 :         if (zm) { if (z[p]>zminus && z[q]>zminus) t = false; }
         [ -  - ][ -  - ]
    1398 [ -  - ][ -  - ]:          0 :         if (zp) { if (z[p]<zplus && z[q]<zplus) t = false; }
         [ -  - ][ -  - ]
    1399                 :            : 
    1400         [ -  - ]:          0 :         if (t) {
    1401 [ -  - ][ -  - ]:          0 :           tagged_edges.push_back( { edge_t( m_rid[e[0]], m_rid[e[1]] ),
    1402                 :          0 :                                     edge_tag::REFINE } );
    1403                 :            :         }
    1404                 :            :       }
    1405                 :            : 
    1406                 :            :     // Do error-based refinement
    1407         [ -  - ]:          0 :     m_refiner.mark_error_refinement( tagged_edges );
    1408                 :            : 
    1409                 :            :     // Update our extra-edge store based on refiner
    1410         [ -  - ]:          0 :     updateEdgeData();
    1411                 :            : 
    1412                 :            :     // Set number of extra edges to a nonzero number, triggering correction
    1413                 :          0 :     m_extra = 1;
    1414                 :          0 :   }
    1415                 :          0 : }
    1416                 :            : 
    1417                 :            : tk::Fields
    1418                 :        114 : Refiner::nodeinit( std::size_t /*npoin*/,
    1419                 :            :                    const std::pair< std::vector< std::size_t >,
    1420                 :            :                                     std::vector< std::size_t > >& /*esup*/ ) const
    1421                 :            : // *****************************************************************************
    1422                 :            : // Evaluate initial conditions (IC) at mesh nodes
    1423                 :            : // //! \param[in] npoin Number points in mesh (on this chare)
    1424                 :            : // //! \param[in] esup Elements surrounding points as linked lists, see tk::genEsup
    1425                 :            : //! \return Initial conditions (evaluated at t0) at nodes
    1426                 :            : // *****************************************************************************
    1427                 :            : {
    1428                 :        114 :   auto t0 = g_cfg.get< tag::t0 >();
    1429                 :        114 :   auto nprop = g_cfg.get< tag::problem_ncomp >();
    1430                 :            : 
    1431                 :            :   // Will store nodal ICs
    1432                 :        114 :   tk::Fields u( m_coord[0].size(), nprop );
    1433                 :            : 
    1434                 :            :   // Evaluate ICs
    1435                 :            : 
    1436                 :            :   // Evaluate ICs for all scalar components integrated
    1437         [ +  - ]:        114 :   problems::initialize( m_coord, u, t0 );
    1438                 :            : 
    1439 [ -  + ][ -  - ]:        114 :   Assert( u.nunk() == m_coord[0].size(), "Size mismatch" );
         [ -  - ][ -  - ]
    1440 [ -  + ][ -  - ]:        114 :   Assert( u.nprop() == nprop, "Size mismatch" );
         [ -  - ][ -  - ]
    1441                 :            : 
    1442                 :        114 :   return u;
    1443                 :          0 : }
    1444                 :            : 
    1445                 :            : void
    1446                 :         71 : Refiner::updateMesh()
    1447                 :            : // *****************************************************************************
    1448                 :            : // Update old mesh after refinement
    1449                 :            : // *****************************************************************************
    1450                 :            : {
    1451                 :            :   // Get refined mesh connectivity
    1452         [ +  - ]:         71 :   const auto& refinpoel = m_refiner.tet_store.get_active_inpoel();
    1453 [ -  + ][ -  - ]:         71 :   Assert( refinpoel.size()%4 == 0, "Inconsistent refined mesh connectivity" );
         [ -  - ][ -  - ]
    1454                 :            : 
    1455                 :            :   // Generate unique node lists of old and refined mesh using local ids
    1456         [ +  - ]:         71 :   auto rinpoel = m_inpoel;
    1457         [ +  - ]:         71 :   tk::remap( rinpoel, m_rid );
    1458         [ +  - ]:         71 :   std::unordered_set< std::size_t > old( begin(rinpoel), end(rinpoel) );
    1459         [ +  - ]:         71 :   std::unordered_set< std::size_t > ref( begin(refinpoel), end(refinpoel) );
    1460                 :            : 
    1461                 :            :   // Augment refiner id -> local node id map with newly added nodes
    1462                 :         71 :   std::size_t l = m_lref.size();
    1463 [ +  - ][ +  + ]:      31966 :   for (auto r : ref) if (old.find(r) == end(old)) m_lref[r] = l++;
         [ +  - ][ +  + ]
    1464                 :            : 
    1465                 :            :   // Get nodal communication map from Discretization worker
    1466         [ -  + ]:         71 :   if ( m_mode == RefMode::DTREF) {
    1467 [ -  - ][ -  - ]:          0 :     m_nodeCommMap = m_disc[ thisIndex ].ckLocal()->NodeCommMap();
                 [ -  - ]
    1468                 :            :   }
    1469                 :            : 
    1470                 :            :   // Update mesh and solution after refinement
    1471         [ +  - ]:         71 :   newVolMesh( old, ref );
    1472                 :            : 
    1473                 :            :   // Update mesh connectivity from refiner lib, remapping refiner to local ids
    1474 [ +  - ][ +  - ]:         71 :   m_inpoel = m_refiner.tet_store.get_active_inpoel();
    1475                 :            : 
    1476         [ +  - ]:         71 :   tk::remap( m_inpoel, m_lref );
    1477                 :            : 
    1478                 :            :   // Update mesh connectivity with new global node ids
    1479         [ +  - ]:         71 :   m_ginpoel = m_inpoel;
    1480 [ +  - ][ -  + ]:         71 :   Assert( tk::uniquecopy(m_ginpoel).size() == m_coord[0].size(),
         [ -  - ][ -  - ]
                 [ -  - ]
    1481                 :            :           "Size mismatch" );
    1482         [ +  - ]:         71 :   tk::remap( m_ginpoel, m_gid );
    1483                 :            : 
    1484                 :            :   // Update boundary face and node information
    1485         [ +  - ]:         71 :   newBndMesh( ref );
    1486                 :            : 
    1487                 :            :   // Augment node communication map with newly added nodes on chare-boundary
    1488         [ -  + ]:         71 :   if (m_mode == RefMode::DTREF) {
    1489         [ -  - ]:          0 :     for (const auto& [ neighborchare, edges ] : m_remoteEdges) {
    1490         [ -  - ]:          0 :       auto& nodes = tk::ref_find( m_nodeCommMap, neighborchare );
    1491         [ -  - ]:          0 :       for (const auto& e : edges) {
    1492                 :            :         // If parent nodes were part of the node communication map for chare
    1493 [ -  - ][ -  - ]:          0 :         if (nodes.find(e[0]) != end(nodes) && nodes.find(e[1]) != end(nodes)) {
         [ -  - ][ -  - ]
                 [ -  - ]
    1494                 :            :           // Add new node if local id was generated for it
    1495         [ -  - ]:          0 :           auto n = Hash<2>()( e );
    1496 [ -  - ][ -  - ]:          0 :           if (m_lid.find(n) != end(m_lid)) nodes.insert( n );
                 [ -  - ]
    1497                 :            :         }
    1498                 :            :       }
    1499                 :            :     }
    1500                 :            :   }
    1501                 :            : 
    1502                 :            :   // Ensure valid mesh after refinement
    1503 [ +  - ][ -  + ]:         71 :   Assert( tk::positiveJacobians( m_inpoel, m_coord ),
         [ -  - ][ -  - ]
                 [ -  - ]
    1504                 :            :           "Refined mesh cell Jacobian non-positive" );
    1505                 :            : 
    1506 [ +  - ][ -  + ]:         71 :   Assert( tk::conforming( m_inpoel, m_coord, true, m_rid ),
         [ -  - ][ -  - ]
         [ -  - ][ -  - ]
    1507                 :            :           "Chare-"+std::to_string(thisIndex)+
    1508                 :            :           " mesh not conforming after updating mesh after mesh refinement" );
    1509                 :            : 
    1510                 :            :   // Perform leak test on new mesh
    1511 [ +  - ][ +  - ]:         71 :   Assert( !tk::leakyPartition(
         [ +  - ][ -  + ]
         [ -  - ][ -  - ]
                 [ -  - ]
    1512                 :            :             tk::genEsuelTet( m_inpoel, tk::genEsup(m_inpoel,4) ),
    1513                 :            :             m_inpoel, m_coord ),
    1514                 :            :           "Refined mesh partition leaky" );
    1515                 :         71 : }
    1516                 :            : 
    1517                 :            : void
    1518                 :         71 : Refiner::newVolMesh( const std::unordered_set< std::size_t >& old,
    1519                 :            :                      const std::unordered_set< std::size_t >& ref )
    1520                 :            : // *****************************************************************************
    1521                 :            : //  Compute new volume mesh after mesh refinement
    1522                 :            : //! \param[in] old Unique nodes of the old (unrefined) mesh using
    1523                 :            : //!   refiner-lib ids
    1524                 :            : //! \param[in] ref Unique nodes of the refined mesh using refiner-lib ids
    1525                 :            : // *****************************************************************************
    1526                 :            : {
    1527                 :         71 :   const auto& x = m_coord[0];
    1528                 :         71 :   const auto& y = m_coord[1];
    1529                 :         71 :   const auto& z = m_coord[2];
    1530                 :            : 
    1531                 :            :   // Generate coordinates and ids to newly added nodes after refinement
    1532                 :         71 :   std::unordered_map< std::size_t, std::size_t > gid_add;
    1533                 :         71 :   tk::destroy( m_addedNodes );
    1534                 :         71 :   tk::destroy( m_removedNodes );
    1535                 :         71 :   tk::destroy( m_amrNodeMap );
    1536         [ +  + ]:      31966 :   for (auto r : ref) {               // for all unique nodes of the refined mesh
    1537 [ +  - ][ +  + ]:      31895 :     if (old.find(r) == end(old)) {   // if node is newly added
    1538                 :            :       // get (local) parent ids of newly added node
    1539         [ +  - ]:      22670 :       auto p = m_refiner.node_connectivity.get( r );
    1540 [ -  + ][ -  - ]:      22670 :       Assert(p[0] != p[1], "Node without parent edge in newVolMesh");
         [ -  - ][ -  - ]
    1541 [ +  - ][ +  - ]:      22670 :       Assert( old.find(p[0]) != end(old) && old.find(p[1]) != end(old),
         [ +  - ][ +  - ]
         [ -  - ][ -  - ]
                 [ -  - ]
    1542                 :            :               "Parent(s) not in old mesh" );
    1543                 :            :       // local parent ids
    1544 [ +  - ][ +  - ]:      22670 :       decltype(p) lp{{tk::cref_find(m_lref,p[0]), tk::cref_find(m_lref,p[1])}};
    1545                 :            :       // global parent ids
    1546                 :      22670 :       decltype(p) gp{{m_gid[lp[0]], m_gid[lp[1]]}};
    1547                 :            :       // generate new global ID for newly added node
    1548         [ +  - ]:      22670 :       auto g = Hash<2>()( gp );
    1549                 :            : 
    1550                 :            :       // if node added by AMR lib has not yet been added to Refiner's new mesh
    1551 [ +  - ][ +  - ]:      22670 :       if (m_coordmap.find(g) == end(m_coordmap)) {
    1552 [ -  + ][ -  - ]:      22670 :         Assert( g >= old.size(), "Hashed id overwriting old id" );
         [ -  - ][ -  - ]
    1553 [ +  - ][ -  + ]:      22670 :         Assert( m_lid.find(g) == end(m_lid),
         [ -  - ][ -  - ]
                 [ -  - ]
    1554                 :            :                 "Overwriting entry global->local node ID map" );
    1555         [ +  - ]:      22670 :         auto l = tk::cref_find( m_lref, r );
    1556                 :            :         // store newly added node id and their parent ids (local ids)
    1557         [ +  - ]:      22670 :         m_addedNodes[r] = lp;   // key = r for later update to local
    1558                 :            :         // assign new node to refiner->global map
    1559         [ +  - ]:      22670 :         gid_add[r] = g; // key = r for later search
    1560                 :            :         // assign new node to global->local map
    1561         [ +  - ]:      22670 :         m_lid[g] = l;
    1562                 :            :         // generate and store coordinates for newly added node
    1563         [ +  - ]:      45340 :         m_coordmap.insert( {g, {{ (x[lp[0]] + x[lp[1]])/2.0,
    1564                 :      22670 :                                   (y[lp[0]] + y[lp[1]])/2.0,
    1565                 :      22670 :                                   (z[lp[0]] + z[lp[1]])/2.0 }} } );
    1566                 :            :       }
    1567                 :            :     }
    1568                 :            :   }
    1569                 :         71 :   tk::destroy( m_coord );
    1570                 :            : 
    1571                 :            :   // generate a node map based on oldnodes+addednodes
    1572         [ +  - ]:         71 :   std::vector< size_t > nodeVec(m_coordmap.size());
    1573         [ +  + ]:      41397 :   for (size_t j=0; j<nodeVec.size(); ++j) {
    1574                 :      41326 :     nodeVec[j] = j;
    1575                 :            :   }
    1576                 :            : 
    1577                 :            :   // Remove coordinates and ids of removed nodes due to derefinement
    1578                 :         71 :   std::unordered_map< std::size_t, std::size_t > gid_rem;
    1579         [ +  + ]:      18727 :   for (auto o : old) {               // for all unique nodes of the old mesh
    1580 [ +  - ][ +  + ]:      18656 :     if (ref.find(o) == end(ref)) {   // if node is no longer in new mesh
    1581         [ +  - ]:       9431 :       auto l = tk::cref_find( m_lref, o );
    1582                 :       9431 :       auto g = m_gid[l];
    1583                 :            :       // store local-ids of removed nodes
    1584         [ +  - ]:       9431 :       m_removedNodes.insert(l);
    1585                 :            :       // remove derefined nodes from node comm map
    1586         [ -  + ]:       9431 :       for (auto& [neighborchare, sharednodes] : m_nodeCommMap) {
    1587 [ -  - ][ -  - ]:          0 :         if (sharednodes.find(g) != sharednodes.end()) {
    1588         [ -  - ]:          0 :           sharednodes.erase(g);
    1589                 :            :         }
    1590                 :            :       }
    1591         [ +  - ]:       9431 :       gid_rem[l] = g;
    1592         [ +  - ]:       9431 :       m_lid.erase( g );
    1593         [ +  - ]:       9431 :       m_coordmap.erase( g );
    1594                 :            :     }
    1595                 :            :   }
    1596                 :            : 
    1597                 :            :   // update the node map by removing the derefined nodes
    1598 [ -  + ][ -  - ]:         71 :   if (m_mode == RefMode::DTREF && m_removedNodes.size() > 0) {
                 [ -  + ]
    1599                 :            :     // remove derefined nodes
    1600                 :          0 :     size_t remCount = 0;
    1601                 :          0 :     size_t origSize = nodeVec.size();
    1602         [ -  - ]:          0 :     for (size_t j=0; j<origSize; ++j) {
    1603                 :          0 :       auto nd = nodeVec[j-remCount];
    1604                 :            : 
    1605                 :          0 :       bool no_change = false;
    1606                 :          0 :       size_t nodeidx = 0;
    1607         [ -  - ]:          0 :       for (const auto& rn : m_removedNodes) {
    1608         [ -  - ]:          0 :         if (nd < *m_removedNodes.cbegin()) {
    1609                 :          0 :           no_change = true;
    1610                 :          0 :           break;
    1611                 :            :         }
    1612         [ -  - ]:          0 :         else if (nd <= rn) {
    1613                 :          0 :           nodeidx = rn;
    1614                 :          0 :           break;
    1615                 :            :         }
    1616                 :            :       }
    1617                 :            : 
    1618                 :            :       // if node is out-or-range of removed nodes list, continue with next entry
    1619         [ -  - ]:          0 :       if (no_change)
    1620                 :          0 :         continue;
    1621                 :            :       // if not is within range of removed nodes list, erase node appropriately
    1622         [ -  - ]:          0 :       else if (nodeidx == nd) {
    1623                 :            :         //! Difference type for iterator/pointer arithmetics
    1624                 :            :         using diff_type = std::vector< std::size_t >::difference_type;
    1625         [ -  - ]:          0 :         nodeVec.erase(nodeVec.begin()+static_cast< diff_type >(j-remCount));
    1626                 :          0 :         ++remCount;
    1627                 :            :       }
    1628                 :            :     }
    1629                 :            : 
    1630 [ -  - ][ -  - ]:          0 :     Assert(remCount == m_removedNodes.size(), "Incorrect number of nodes removed "
         [ -  - ][ -  - ]
    1631                 :            :       "from node map.");
    1632                 :            :   }
    1633                 :            : 
    1634                 :            :   // invert node vector to get node map
    1635         [ +  + ]:      41397 :   for (size_t i=0; i<nodeVec.size(); ++i) {
    1636         [ +  - ]:      41326 :     m_amrNodeMap[nodeVec[i]] = i;
    1637                 :            :   }
    1638                 :            : 
    1639                 :            :   //// Save previous states of refiner-local node id maps before update
    1640                 :            :   //m_oldrid = m_rid;
    1641                 :            :   //m_oldlref = m_lref;
    1642                 :            : 
    1643                 :            :   // Generate new node id maps for nodes kept
    1644                 :         71 :   tk::destroy( m_lref );
    1645         [ +  - ]:         71 :   std::vector< std::size_t > rid( ref.size() );
    1646         [ +  - ]:         71 :   std::vector< std::size_t > gid( ref.size() );
    1647                 :         71 :   std::size_t l = 0;    // will generate new local node id
    1648         [ +  + ]:      18727 :   for (std::size_t i=0; i<m_gid.size(); ++i) {
    1649 [ +  - ][ +  + ]:      18656 :     if (gid_rem.find(i) == end(gid_rem)) {
    1650                 :       9225 :       gid[l] = m_gid[i];
    1651                 :       9225 :       rid[l] = m_rid[i];
    1652         [ +  - ]:       9225 :       m_lref[ m_rid[i] ] = l;
    1653                 :       9225 :       ++l;
    1654                 :            :     }
    1655                 :            :   }
    1656                 :            :   // Add newly added nodes due to refinement to node id maps
    1657                 :            :   // cppcheck-suppress unreadVariable
    1658         [ +  - ]:         71 :   decltype(m_addedNodes) addedNodes( m_addedNodes.size() );
    1659         [ +  + ]:      22741 :   for (const auto& n : gid_add) {
    1660                 :      22670 :     auto r = n.first;
    1661                 :      22670 :     auto g = n.second;
    1662                 :      22670 :     gid[l] = g;
    1663                 :            :     // cppcheck-suppress unreadVariable
    1664                 :      22670 :     rid[l] = r;
    1665 [ +  - ][ -  + ]:      22670 :     Assert(m_lref.find(r) == m_lref.end(), "Overwriting lref");
         [ -  - ][ -  - ]
                 [ -  - ]
    1666         [ +  - ]:      22670 :     m_lref[r] = l;
    1667         [ +  - ]:      22670 :     auto it = m_addedNodes.find( r );
    1668 [ -  + ][ -  - ]:      22670 :     Assert( it != end(m_addedNodes), "Cannot find added node" );
         [ -  - ][ -  - ]
    1669         [ +  - ]:      22670 :     addedNodes[l] = std::move(it->second);
    1670 [ +  - ][ +  - ]:      22670 :     addedNodes.at(l)[0] = m_amrNodeMap[addedNodes.at(l)[0]];
                 [ +  - ]
    1671 [ +  - ][ +  - ]:      22670 :     addedNodes.at(l)[1] = m_amrNodeMap[addedNodes.at(l)[1]];
                 [ +  - ]
    1672                 :      22670 :     ++l;
    1673                 :            :   }
    1674 [ -  + ][ -  - ]:         71 :   Assert( m_lref.size() == ref.size(), "Size mismatch" );
         [ -  - ][ -  - ]
    1675                 :            :   //for (auto r : ref) {
    1676                 :            :   //  Assert(m_lref.find(r) != m_lref.end(), "Node missing in lref");
    1677                 :            :   //}
    1678                 :            :   //const auto& int_list = m_refiner.tet_store.intermediate_list;
    1679                 :            :   //for (auto in : int_list) {
    1680                 :            :   //  Assert(m_lref.find(in) != m_lref.end(), "Interm node missing in lref: "
    1681                 :            :   //    + std::to_string(in));
    1682                 :            :   //}
    1683                 :         71 :   m_rid = std::move( rid );
    1684 [ -  + ][ -  - ]:         71 :   Assert( m_rid.size() == ref.size(), "Size mismatch" );
         [ -  - ][ -  - ]
    1685                 :         71 :   m_addedNodes = std::move( addedNodes );
    1686                 :            : 
    1687                 :            :   // Update node coordinates, ids, and id maps
    1688                 :         71 :   auto& rx = m_coord[0];
    1689                 :         71 :   auto& ry = m_coord[1];
    1690                 :         71 :   auto& rz = m_coord[2];
    1691         [ +  - ]:         71 :   rx.resize( ref.size() );
    1692         [ +  - ]:         71 :   ry.resize( ref.size() );
    1693         [ +  - ]:         71 :   rz.resize( ref.size() );
    1694         [ +  + ]:      31966 :   for (std::size_t i=0; i<gid.size(); ++i) {
    1695         [ +  - ]:      31895 :     tk::ref_find( m_lid, gid[i] ) = i;
    1696         [ +  - ]:      31895 :     const auto& c = tk::cref_find( m_coordmap, gid[i] );
    1697                 :      31895 :     rx[i] = c[0];
    1698                 :      31895 :     ry[i] = c[1];
    1699                 :      31895 :     rz[i] = c[2];
    1700                 :            :   }
    1701                 :         71 :   m_gid = std::move( gid );
    1702 [ +  - ][ +  - ]:         71 :   Assert( m_gid.size() == m_lid.size() && m_gid.size() == ref.size(),
         [ -  - ][ -  - ]
                 [ -  - ]
    1703                 :            :     "Size mismatch" );
    1704                 :         71 : }
    1705                 :            : 
    1706                 :            : std::unordered_set< std::size_t >
    1707                 :    2552108 : Refiner::ancestors( std::size_t n )
    1708                 :            : // *****************************************************************************
    1709                 :            : // Find the oldest parents of a mesh node in the AMR hierarchy
    1710                 :            : //! \param[in] n Local node id whose ancestors to search
    1711                 :            : //! \return Parents of local node id from the coarsest (original) mesh
    1712                 :            : // *****************************************************************************
    1713                 :            : {
    1714         [ +  - ]:    2552108 :   auto d = m_refiner.node_connectivity.get( m_rid[n] );
    1715         [ +  - ]:    2552108 :   decltype(d) p{{ tk::cref_find( m_lref, d[0] ),
    1716         [ +  - ]:    2552108 :                   tk::cref_find( m_lref, d[1] ) }};
    1717                 :            : 
    1718                 :    2552108 :   std::unordered_set< std::size_t > s;
    1719                 :            : 
    1720 [ +  - ][ +  + ]:    2552108 :   if (p != AMR::node_pair_t{{n,n}}) {
    1721         [ +  - ]:     926230 :     auto q = ancestors( p[0] );
    1722         [ +  - ]:     926230 :     s.insert( begin(q), end(q) );
    1723         [ +  - ]:     926230 :     auto r = ancestors( p[1] );
    1724         [ +  - ]:     926230 :     s.insert( begin(r), end(r) );
    1725                 :     926230 :   } else {
    1726         [ +  - ]:    1625878 :     s.insert( begin(p), end(p) );
    1727                 :            :   }
    1728                 :            : 
    1729                 :    5104216 :   return s;
    1730                 :          0 : }
    1731                 :            : 
    1732                 :            : Refiner::BndFaceData
    1733                 :         71 : Refiner::boundary()
    1734                 :            : // *****************************************************************************
    1735                 :            : //  Generate boundary data structures used to update refined/derefined boundary
    1736                 :            : //  faces and nodes of side sets
    1737                 :            : //! \return A tuple of boundary face data
    1738                 :            : //! \details The output of this function is used to regenerate physical boundary
    1739                 :            : //!   face and node data structures after refinement, see updateBndData().
    1740                 :            : // *****************************************************************************
    1741                 :            : {
    1742                 :            :   // Generate the inverse of AMR's tet store
    1743                 :         71 :   std::unordered_map< Tet, std::size_t, Hash<4>, Eq<4> > invtets;
    1744         [ +  + ]:     145529 :   for (const auto& [key, tet] : m_refiner.tet_store.tets)
    1745         [ +  - ]:     145458 :     invtets[ tet ] = key;
    1746                 :            : 
    1747                 :            :   //std::cout << thisIndex << " invt: " << invtets.size() << '\n';
    1748                 :            :   //std::cout << thisIndex << " active inpoel size: " << m_refiner.tet_store.get_active_inpoel().size() << '\n';
    1749                 :            :   //std::cout << thisIndex << " marked derefinement size: " << m_refiner.tet_store.marked_derefinements.size() << '\n';
    1750                 :            : 
    1751                 :            :   // Generate data structure pcFaceTets for the new (post-AMR) mesh:
    1752                 :            :   // pcFaceTets is a map that associates all triangle boundary faces (physical
    1753                 :            :   // and chare) to the id of the tet adjacent to the said face.
    1754                 :            :   // Key: Face-nodes' global id; Value: tet-id.
    1755                 :         71 :   std::unordered_map< Face, std::size_t, Hash<3>, Eq<3> > pcFaceTets;
    1756 [ +  - ][ +  - ]:         71 :   auto esuel = tk::genEsuelTet( m_inpoel, tk::genEsup(m_inpoel,4) );
    1757         [ +  + ]:     128238 :   for (std::size_t e=0; e<esuel.size()/4; ++e) {
    1758                 :     128167 :     auto m = e*4;
    1759         [ +  + ]:     640835 :     for (std::size_t f=0; f<4; ++f) {
    1760         [ +  + ]:     512668 :       if (esuel[m+f] == -1) {  // if a face does not have an adjacent tet
    1761                 :      38060 :         Face b{{ m_ginpoel[ m+tk::lpofa[f][0] ],
    1762                 :      38060 :                  m_ginpoel[ m+tk::lpofa[f][1] ],
    1763                 :      76120 :                  m_ginpoel[ m+tk::lpofa[f][2] ] }};
    1764 [ +  - ][ +  - ]:      38060 :         Assert( m_inpoel[m+0] < m_rid.size() &&
         [ +  - ][ +  - ]
         [ -  - ][ -  - ]
                 [ -  - ]
    1765                 :            :                 m_inpoel[m+1] < m_rid.size() &&
    1766                 :            :                 m_inpoel[m+2] < m_rid.size() &&
    1767                 :            :                 m_inpoel[m+3] < m_rid.size(), "Indexing out of rid" );
    1768                 :      76120 :         Tet t{{ m_rid[ m_inpoel[m+0] ], m_rid[ m_inpoel[m+1] ],
    1769                 :      76120 :                 m_rid[ m_inpoel[m+2] ], m_rid[ m_inpoel[m+3] ] }};
    1770                 :            :         //Tet t{{ m_inpoel[m+0], m_inpoel[m+1],
    1771                 :            :         //        m_inpoel[m+2], m_inpoel[m+3] }};
    1772                 :            :         // associate tet id to adjacent (physical or chare) boundary face
    1773         [ +  - ]:      38060 :         auto i = invtets.find( t );
    1774 [ +  - ][ -  + ]:      38060 :         Assert(m_refiner.tet_store.is_active(i->second),
         [ -  - ][ -  - ]
                 [ -  - ]
    1775                 :            :           "Inactive element while regenerating boundary data");
    1776         [ +  - ]:      38060 :         if (i != end(invtets)) {
    1777                 :            :           //std::cout << "refacetets: " <<
    1778                 :            :           //  b[0] << "-" << b[1] << "-" << b[2] << std::endl;
    1779         [ +  - ]:      38060 :           pcFaceTets[ b ] = i->second;
    1780                 :            :         } else {
    1781 [ -  - ][ -  - ]:          0 :           Throw("Active element not found in tet_store");
                 [ -  - ]
    1782                 :            :         }
    1783                 :            :       }
    1784                 :            :     }
    1785                 :            :   }
    1786                 :            : 
    1787                 :            :   // Generate child->parent tet and id maps after refinement/derefinement step
    1788                 :            : //  tk::destroy( m_oldparent );
    1789                 :         71 :   m_addedTets.clear();
    1790                 :         71 :   std::size_t p = 0;
    1791                 :            :   // cppcheck-suppress unreadVariable
    1792                 :         71 :   std::size_t c = 0;
    1793                 :         71 :   const auto& tet_store = m_refiner.tet_store;
    1794         [ +  + ]:     145529 :   for (const auto& t : tet_store.tets) {
    1795                 :            :     // query number of children of tet
    1796         [ +  - ]:     145458 :     auto nc = tet_store.data( t.first ).children.size();
    1797         [ +  + ]:     282302 :     for (decltype(nc) i=0; i<nc; ++i ) {      // for all child tets
    1798                 :            :       // get child tet id
    1799         [ +  - ]:     136844 :       auto childtet = tet_store.get_child_id( t.first, i );
    1800         [ +  - ]:     136844 :       auto ct = tet_store.tets.find( childtet );
    1801 [ -  + ][ -  - ]:     136844 :       Assert(ct != tet_store.tets.end(), "Child not found in tet store");
         [ -  - ][ -  - ]
    1802                 :            : //      //auto cA = tk::cref_find( m_lref, ct->second[0] );
    1803                 :            : //      //auto cB = tk::cref_find( m_lref, ct->second[1] );
    1804                 :            : //      //auto cC = tk::cref_find( m_lref, ct->second[2] );
    1805                 :            : //      //auto cD = tk::cref_find( m_lref, ct->second[3] );
    1806                 :            : //      // get nodes of parent tet
    1807                 :            : //      //auto pA = tk::cref_find( m_lref, t.second[0] );
    1808                 :            : //      //auto pB = tk::cref_find( m_lref, t.second[1] );
    1809                 :            : //      //auto pC = tk::cref_find( m_lref, t.second[2] );
    1810                 :            : //      //auto pD = tk::cref_find( m_lref, t.second[3] );
    1811                 :            : //      // assign parent tet to child tet
    1812                 :            : //      //m_oldparent[ {{cA,cB,cC,cD}} ] = {{pA,pB,pC,pD}};
    1813                 :            : //      m_oldparent[ ct->second ] = t.second; //{{pA,pB,pC,pD}};
    1814 [ +  - ][ +  + ]:     136844 :       if (m_oldTets.find(ct->second) == end(m_oldTets)) {
    1815                 :            :         // TODO: the following code can assign negative ids to newly added tets.
    1816                 :            :         // This needs to be corrected before applying to cell-based schemes.
    1817                 :            :         //Assert((p-m_oldntets) > 0, "Negative id assigned to added tet");
    1818         [ +  - ]:     112028 :         m_addedTets[ c++ ] = p - m_oldntets;
    1819                 :            :       }
    1820                 :            :     }
    1821                 :     145458 :     ++p;
    1822                 :            :   }
    1823                 :            : 
    1824                 :            :   //std::cout << thisIndex << " added: " << m_addedTets.size() << '\n';
    1825                 :            :   //std::cout << thisIndex << " parent: " << m_oldparent.size() << '\n';
    1826                 :            :   //std::cout << thisIndex << " pcret: " << pcFaceTets.size() << '\n';
    1827                 :            : 
    1828                 :            :   //for (std::size_t f=0; f<m_triinpoel.size()/3; ++f) {
    1829                 :            :   //  std::cout << "triinpoel: " <<
    1830                 :            :   //    m_triinpoel[f*3+0] << "-" << m_triinpoel[f*3+1] << "-" <<
    1831                 :            :   //    m_triinpoel[f*3+2] << std::endl;
    1832                 :            :   //}
    1833                 :            : 
    1834                 :        142 :   return pcFaceTets;
    1835                 :         71 : }
    1836                 :            : 
    1837                 :            : void
    1838                 :         71 : Refiner::newBndMesh( const std::unordered_set< std::size_t >& ref )
    1839                 :            : // *****************************************************************************
    1840                 :            : // Update boundary data structures after mesh refinement
    1841                 :            : //! \param[in] ref Unique nodes of the refined mesh using refiner-lib ids
    1842                 :            : // *****************************************************************************
    1843                 :            : {
    1844                 :            :   // Generate boundary face data structures used to regenerate boundary face
    1845                 :            :   // and node data after mesh refinement
    1846         [ +  - ]:         71 :   auto pcFaceTets = boundary();
    1847                 :            : 
    1848                 :            :   // Regerate boundary faces and nodes after AMR step
    1849         [ +  - ]:         71 :   updateBndData( ref, pcFaceTets );
    1850                 :         71 : }
    1851                 :            : 
    1852                 :            : void
    1853                 :         71 : Refiner::updateBndData(
    1854                 :            :   [[maybe_unused]] const std::unordered_set< std::size_t >& ref,
    1855                 :            :   const BndFaceData& pcFaceTets )
    1856                 :            : // *****************************************************************************
    1857                 :            : // Regenerate boundary faces and nodes after AMR step
    1858                 :            : //! \param[in] ref Unique nodes of the refined mesh using refiner-lib ids
    1859                 :            : //! \param[in] pcFaceTets Boundary face data
    1860                 :            : // *****************************************************************************
    1861                 :            : {
    1862                 :            :   // storage for boundary faces associated to side-set IDs of the refined mesh
    1863                 :         71 :   tk::destroy( m_bface );
    1864                 :            :   // storage for boundary faces-node connectivity of the refined mesh
    1865                 :         71 :   tk::destroy( m_triinpoel );
    1866                 :            :   // storage for boundary nodes associated to side-set IDs of the refined mesh
    1867                 :         71 :   tk::destroy( m_bnode );
    1868                 :            : 
    1869                 :            :   // will collect unique faces added for each side set
    1870                 :         71 :   std::unordered_map< int, FaceSet > bf;
    1871                 :            : 
    1872                 :            :   // Lambda to search the parents in the coarsest mesh of a mesh node and if
    1873                 :            :   // found, add its global id to boundary node lists associated to the side
    1874                 :            :   // set(s) of its parents. Argument 'n' is the local id of the mesh node id
    1875                 :            :   // whose parents to search.
    1876                 :     585468 :   auto addBndNodes = [&]( std::size_t n ){
    1877         [ +  - ]:     585468 :     auto a = ancestors( n );  // find parents of n in coarse mesh
    1878         [ +  + ]:     585468 :     if (a.size() == 1) {
    1879                 :            :       // node was part of the coarse mesh
    1880 [ -  + ][ -  - ]:     109874 :       Assert(*a.cbegin() == n, "Single ancestor not self");
         [ -  - ][ -  - ]
    1881         [ +  - ]:     109874 :       auto ss = keys( m_coarseBndNodes, m_gid[*a.cbegin()] );
    1882         [ +  + ]:     269898 :       for (auto s : ss)
    1883 [ +  - ][ +  - ]:     160024 :         m_bnode[ s ].push_back( m_gid[n] );
    1884         [ +  + ]:     585468 :     } else if (a.size() == 2) {
    1885                 :            :       // node was added to an edge of a coarse face
    1886         [ +  - ]:     374802 :       std::vector< std::size_t > p( begin(a), end(a) );
    1887         [ +  - ]:     374802 :       auto ss1 = keys( m_coarseBndNodes, m_gid[p[0]] );
    1888         [ +  - ]:     374802 :       auto ss2 = keys( m_coarseBndNodes, m_gid[p[1]] );
    1889         [ +  + ]:     891908 :       for (auto s : ss1) {
    1890                 :            :         // only add 'n' to bnode if all parent nodes are in same side set, else
    1891                 :            :         // 'n' is not a boundary node
    1892 [ +  - ][ +  + ]:     517106 :         if (ss2.find(s) != end(ss2)) {
    1893 [ +  - ][ +  - ]:     404902 :           m_bnode[ s ].push_back( m_gid[n] );
    1894                 :            :         }
    1895                 :            :       }
    1896         [ +  - ]:     475594 :     } else if (a.size() == 3) {
    1897                 :            :       // node was added inside of a coarse face
    1898         [ +  - ]:     100792 :       std::vector< std::size_t > p( begin(a), end(a) );
    1899         [ +  - ]:     100792 :       auto ss1 = keys( m_coarseBndNodes, m_gid[p[0]] );
    1900         [ +  - ]:     100792 :       auto ss2 = keys( m_coarseBndNodes, m_gid[p[1]] );
    1901         [ +  - ]:     100792 :       auto ss3 = keys( m_coarseBndNodes, m_gid[p[2]] );
    1902         [ +  + ]:     275845 :       for (auto s : ss1) {
    1903                 :            :         // only add 'n' to bnode if all parent nodes are in same side set, else
    1904                 :            :         // 'n' is not a boundary node
    1905 [ +  - ][ +  + ]:     175053 :         if (ss2.find(s) != end(ss2) && ss3.find(s) != end(ss3)) {
         [ +  - ][ +  + ]
                 [ +  + ]
    1906 [ +  - ][ +  - ]:      89992 :           m_bnode[ s ].push_back( m_gid[n] );
    1907                 :            :         }
    1908                 :            :       }
    1909                 :     100792 :     }
    1910                 :     585468 :   };
    1911                 :            : 
    1912                 :            :   // face id counter
    1913                 :         71 :   std::size_t facecnt = 0;
    1914                 :            : 
    1915                 :            :   // Lambda to associate a boundary face and connectivity to a side set.
    1916                 :            :   // Argument 's' is the list of faces (ids) to add the new face to. Argument
    1917                 :            :   // 'ss' is the side set id to which the face is added. Argument 'f' is the
    1918                 :            :   // triangle face connectivity to add.
    1919                 :      33968 :   auto addBndFace = [&]( std::vector< std::size_t >& s, int ss, const Face& f )
    1920                 :            :   {
    1921                 :            :     // only add face if it has not yet been aded to this side set
    1922         [ +  - ]:      33968 :     if (bf[ ss ].insert( f ).second) {
    1923         [ +  - ]:      33968 :       s.push_back( facecnt++ );
    1924         [ +  - ]:      33968 :       m_triinpoel.insert( end(m_triinpoel), begin(f), end(f) );
    1925 [ -  + ][ -  - ]:      33968 :       Assert(m_triinpoel.size()/3 == facecnt, "Incorrect size of triinpoel");
         [ -  - ][ -  - ]
    1926                 :            :     }
    1927                 :      33968 :   };
    1928                 :            : 
    1929                 :            :   // Regenerate boundary faces for new mesh along side sets. For all faces
    1930                 :            :   // associated to side sets, we find the ancestors (parents of nodes in the
    1931                 :            :   // original/coarsest mesh) of the nodes comprising the physical and chare
    1932                 :            :   // boundary faces of the new mesh.
    1933                 :            :   //bool faceNoSs = false;
    1934                 :            :   // for all P/C faces in current (post-AMR) mesh
    1935         [ +  + ]:      38131 :   for (const auto& [ face, tetid ] : pcFaceTets) {
    1936                 :            :     // find ancestors of face
    1937                 :      38060 :     std::unordered_set< std::size_t > ans;
    1938         [ +  + ]:     152240 :     for (std::size_t i=0; i<3; ++i) {
    1939 [ +  - ][ +  - ]:     114180 :       auto ai = ancestors(tk::cref_find(m_lid, face[i]));
    1940         [ +  - ]:     114180 :       ans.insert(ai.begin(), ai.end());
    1941                 :     114180 :     }
    1942 [ -  + ][ -  - ]:      38060 :     Assert(ans.size() == 3, "Incorrect number of ancestors in refined face");
         [ -  - ][ -  - ]
    1943                 :            :     Face af;
    1944                 :      38060 :     std::size_t i = 0;
    1945         [ +  + ]:     152240 :     for (auto ai:ans) {
    1946                 :     114180 :       af[i] = m_gid[ai];
    1947                 :     114180 :       ++i;
    1948                 :            :     }
    1949                 :            :     // for all boundary faces in original mesh
    1950                 :            :     //std::size_t fss = 0;
    1951         [ +  + ]:     233216 :     for (const auto& [ss, cfaceset] : m_coarseBndFaces) {
    1952 [ +  - ][ +  + ]:     195156 :       if (cfaceset.find(af) != cfaceset.end()) {
    1953 [ +  - ][ +  - ]:      33968 :         addBndFace(m_bface[ss], ss, face);
    1954                 :            :         //++fss;
    1955                 :            :       }
    1956         [ +  + ]:     780624 :       for (auto j : face) {
    1957 [ +  - ][ +  - ]:     585468 :         addBndNodes(tk::cref_find(m_lid, j));
    1958                 :            :       }
    1959                 :            :     }
    1960                 :            :     //if (fss==0) {
    1961                 :            :     //  std::cout << "Face added to no/multiple side sets; " << fss << std::endl;
    1962                 :            :     //  faceNoSs = true;
    1963                 :            :     //}
    1964                 :      38060 :   }
    1965                 :            : 
    1966                 :            :   // Commented code below, since diagcg can work without sideset/bcs
    1967                 :            :   //Assert(!faceNoSs, "Face/s added to incorrect number of side sets");
    1968                 :            : 
    1969                 :            :   // Make boundary node IDs unique for each physical boundary (side set)
    1970 [ +  - ][ +  + ]:        423 :   for (auto& s : m_bnode) tk::unique( s.second );
    1971                 :            : 
    1972                 :            :   //for (const auto& [ setid, faceids ] : m_bface) {
    1973                 :            :   //  std::cout << "sset: " << setid << std::endl;
    1974                 :            :   //  for (auto f : faceids) {
    1975                 :            :   //    Assert(f<m_triinpoel.size()/3, "Out of bounds access into triinpoel");
    1976                 :            :   //    std::cout << "new bndfaces: " <<
    1977                 :            :   //      m_triinpoel[f*3+0] << "-" << m_triinpoel[f*3+1] << "-" <<
    1978                 :            :   //      m_triinpoel[f*3+2] << std::endl;
    1979                 :            :   //  }
    1980                 :            :   //}
    1981                 :            : 
    1982                 :            :   //for (std::size_t f=0; f<m_triinpoel.size()/3; ++f) {
    1983                 :            :   //  std::cout << "new triinpoel: " <<
    1984                 :            :   //    m_triinpoel[f*3+0] << "-" << m_triinpoel[f*3+1] << "-" <<
    1985                 :            :   //    m_triinpoel[f*3+2] << std::endl;
    1986                 :            :   //}
    1987                 :            : 
    1988                 :            :   //std::cout << thisIndex << " bf: " << tk::sumvalsize( m_bface ) << '\n';
    1989                 :            : 
    1990                 :            :   //std::cout << thisIndex << " bn: " << tk::sumvalsize( m_bnode ) << '\n';
    1991                 :            : 
    1992                 :            :   // Perform leak-test on boundary face data just updated (only in DEBUG)
    1993 [ +  - ][ -  + ]:         71 :   Assert( bndIntegral(), "Partial boundary integral" );
         [ -  - ][ -  - ]
                 [ -  - ]
    1994                 :         71 : }
    1995                 :            : 
    1996                 :            : bool
    1997                 :         71 : Refiner::bndIntegral()
    1998                 :            : // *****************************************************************************
    1999                 :            : //  Compute partial boundary surface integral and sum across all chares
    2000                 :            : //! \return true so we don't trigger assert in client code
    2001                 :            : //! \details This function computes a partial surface integral over the boundary
    2002                 :            : //!   of the faces of this mesh partition then sends its contribution to perform
    2003                 :            : //!   the integral acorss the total problem boundary. After the global sum a
    2004                 :            : //!   non-zero vector result indicates a leak, e.g., a hole in the boundary
    2005                 :            : //!   which indicates an error in the boundary face data structures used to
    2006                 :            : //!   compute the partial surface integrals.
    2007                 :            : // *****************************************************************************
    2008                 :            : {
    2009                 :         71 :   const auto& x = m_coord[0];
    2010                 :         71 :   const auto& y = m_coord[1];
    2011                 :         71 :   const auto& z = m_coord[2];
    2012                 :            : 
    2013         [ +  - ]:         71 :   std::vector< tk::real > s{{ 0.0, 0.0, 0.0 }};
    2014                 :            : 
    2015         [ +  + ]:        423 :   for (const auto& [ setid, faceids ] : m_bface) {
    2016         [ +  + ]:      34320 :     for (auto f : faceids) {
    2017         [ +  - ]:      33968 :       auto A = tk::cref_find( m_lid, m_triinpoel[f*3+0] );
    2018         [ +  - ]:      33968 :       auto B = tk::cref_find( m_lid, m_triinpoel[f*3+1] );
    2019         [ +  - ]:      33968 :       auto C = tk::cref_find( m_lid, m_triinpoel[f*3+2] );
    2020                 :            :       // Compute face area and normal
    2021                 :            :       tk::real nx, ny, nz;
    2022                 :      33968 :       auto a = tk::normal( x[A],x[B],x[C], y[A],y[B],y[C], z[A],z[B],z[C],
    2023                 :            :                            nx, ny, nz );
    2024                 :            :       // Sum up face area * face unit-normal
    2025                 :      33968 :       s[0] += a * nx;
    2026                 :      33968 :       s[1] += a * ny;
    2027                 :      33968 :       s[2] += a * nz;
    2028                 :            :     }
    2029                 :            :   }
    2030                 :            : 
    2031         [ +  - ]:         71 :   s.push_back( -1.0 );  // negative: no call-back after reduction
    2032         [ +  - ]:         71 :   s.push_back( static_cast< tk::real >( m_meshid ) );
    2033                 :            : 
    2034                 :            :   // Send contribution to host summing partial surface integrals
    2035         [ +  - ]:         71 :   contribute( s, CkReduction::sum_double, m_cbr.get< tag::bndint >() );
    2036                 :            : 
    2037                 :         71 :   return true;  // don't trigger the assert in client code
    2038                 :         71 : }
    2039                 :            : 
    2040                 :            : #include "NoWarning/refiner.def.h"

Generated by: LCOV version 1.16