Quinoa all test code coverage report
Current view: top level - Inciter - ALE.cpp (source / functions) Hit Total Coverage
Commit: -128-NOTFOUND Lines: 324 325 99.7 %
Date: 2024-11-22 09:12:55 Functions: 19 20 95.0 %
Legend: Lines: hit not hit | Branches: + taken - not taken # not executed Branches: 326 468 69.7 %

           Branch data     Line data    Source code
       1                 :            : // *****************************************************************************
       2                 :            : /*!
       3                 :            :   \file      src/Inciter/ALE.cpp
       4                 :            :   \copyright 2012-2015 J. Bakosi,
       5                 :            :              2016-2018 Los Alamos National Security, LLC.,
       6                 :            :              2019-2021 Triad National Security, LLC.
       7                 :            :              All rights reserved. See the LICENSE file for details.
       8                 :            :   \brief     Definitions file for distributed ALE mesh motion
       9                 :            :   \details   Definitions file for asynchronous distributed
      10                 :            :              arbitrary Lagrangian-Eulerian (ALE) mesh motion using Charm++.
      11                 :            : */
      12                 :            : // *****************************************************************************
      13                 :            : 
      14                 :            : #include "ALE.hpp"
      15                 :            : #include "DerivedData.hpp"
      16                 :            : #include "Vector.hpp"
      17                 :            : #include "Inciter/InputDeck/InputDeck.hpp"
      18                 :            : 
      19                 :            : namespace inciter {
      20                 :            : 
      21                 :            : extern ctr::InputDeck g_inputdeck;
      22                 :            : 
      23                 :            : } // inciter::
      24                 :            : 
      25                 :            : using inciter::ALE;
      26                 :            : 
      27                 :         31 : ALE::ALE( const tk::CProxy_ConjugateGradients& conjugategradientsproxy,
      28                 :            :           const std::array< std::vector< tk::real >, 3 >& coord,
      29                 :            :           const std::vector< std::size_t >& inpoel,
      30                 :            :           const std::vector< std::size_t >& gid,
      31                 :            :           const std::unordered_map< std::size_t, std::size_t >& lid,
      32                 :         31 :           const tk::NodeCommMap& nodeCommMap ) :
      33                 :            :   m_conjugategradients( conjugategradientsproxy ),
      34                 :            :   m_done(),
      35                 :            :   m_nvort( 0 ),
      36                 :            :   m_ndiv( 0 ),
      37                 :            :   m_npot( 0 ),
      38                 :            :   m_nwf( 0 ),
      39                 :            :   m_nodeCommMap( nodeCommMap ),
      40                 :            :   m_lid( lid ),
      41                 :            :   m_coord0( coord ),
      42                 :            :   m_inpoel( inpoel ),
      43                 :            :   m_vol0(),
      44                 :            :   m_vol(),
      45                 :            :   m_it( 0 ),
      46                 :            :   m_t( 0.0 ),
      47                 :            :   m_w( gid.size(), 3 ),
      48                 :            :   m_wf( gid.size(), 3 ),
      49                 :            :   m_wfc(),
      50                 :            :   m_veldiv(),
      51                 :            :   m_veldivc(),
      52                 :            :   m_gradpot(),
      53                 :            :   m_gradpotc(),
      54                 :            :   m_vorticity(),
      55                 :            :   m_vorticityc(),
      56                 :            :   m_meshveldirbcnodes(),
      57                 :            :   m_meshvelsymbcnodes(),
      58 [ +  - ][ +  - ]:        124 :   m_move( moveCfg() )
         [ +  - ][ +  - ]
         [ +  - ][ +  - ]
         [ +  - ][ +  - ]
      59                 :            : // *****************************************************************************
      60                 :            : //  Constructor
      61                 :            : //! \param[in] conjugategradientsproxy Distributed Conjugrate Gradients linear
      62                 :            : //!   solver proxy (bound to ALE proxy)
      63                 :            : //! \param[in] coord Mesh node coordinates
      64                 :            : //! \param[in] inpoel Mesh element connectivity
      65                 :            : //! \param[in] gid Local->global node id map
      66                 :            : //! \param[in] lid Global->locbal node id map
      67                 :            : //! \param[in] nodeCommMap Node communication map
      68                 :            : // *****************************************************************************
      69                 :            : {
      70                 :         31 :   auto smoother = g_inputdeck.get< tag::ale, tag::smoother >();
      71                 :            : 
      72         [ +  + ]:         31 :   if (smoother == ctr::MeshVelocitySmootherType::LAPLACE)
      73                 :         17 :     m_conjugategradients[ thisIndex ].        // solve for mesh velocity
      74 [ +  - ][ +  - ]:         34 :       insert( Laplacian( 3, coord ), gid, m_lid, m_nodeCommMap );
         [ +  - ][ -  + ]
                 [ -  - ]
      75         [ +  + ]:         14 :   else if (smoother == ctr::MeshVelocitySmootherType::HELMHOLTZ)
      76                 :          4 :     m_conjugategradients[ thisIndex ].        // solve for scalar potential
      77 [ +  - ][ +  - ]:          8 :       insert( Laplacian( 1, coord ), gid, m_lid, m_nodeCommMap );
         [ +  - ][ -  + ]
                 [ -  - ]
      78                 :            : 
      79                 :            :   // Zero ALE mesh velocity
      80                 :            :   m_w.fill( 0.0 );
      81                 :            : 
      82                 :            :   // Activate SDAG wait for initially computing prerequisites for ALE
      83 [ +  - ][ +  - ]:         31 :   thisProxy[ thisIndex ].wait4vel();
      84 [ +  - ][ +  - ]:         31 :   thisProxy[ thisIndex ].wait4pot();
      85 [ +  - ][ +  - ]:         31 :   thisProxy[ thisIndex ].wait4for();
      86                 :         31 : }
      87                 :            : 
      88                 :            : std::tuple< tk::CSR, std::vector< tk::real >, std::vector< tk::real > >
      89                 :         21 : ALE::Laplacian( std::size_t ncomp,
      90                 :            :                 const std::array< std::vector< tk::real >, 3 >& coord ) const
      91                 :            : // *****************************************************************************
      92                 :            : // Generate {A,x,b} for Laplacian mesh velocity smoother
      93                 :            : //! \param[in] ncomp Number of scalar components
      94                 :            : //! \param[in] coord Mesh node coordinates
      95                 :            : //! \return {A,x,b} with a Laplacian, unknown, and rhs initialized with zeros
      96                 :            : //! \see Waltz, et al. "A three-dimensional finite element arbitrary
      97                 :            : //!   Lagrangian-Eulerian method for shock hydrodynamics on unstructured
      98                 :            : //!   grids", Computers& Fluids, 2013, and Bakosi, et al. "Improved ALE mesh
      99                 :            : //!   velocities for complex flows, International Journal for Numerical Methods
     100                 :            : //!   in Fluids, 2017.
     101                 :            : // *****************************************************************************
     102                 :            : {
     103 [ +  - ][ +  - ]:         63 :   tk::CSR A( ncomp, tk::genPsup(m_inpoel,4,tk::genEsup(m_inpoel,4)) );
     104                 :            : 
     105                 :            :   const auto& X = coord[0];
     106                 :            :   const auto& Y = coord[1];
     107                 :            :   const auto& Z = coord[2];
     108                 :            : 
     109                 :            :   // fill matrix with Laplacian
     110         [ +  + ]:      93293 :   for (std::size_t e=0; e<m_inpoel.size()/4; ++e) {
     111                 :            :     // access node IDs
     112                 :            :     const std::array< std::size_t, 4 >
     113                 :      93272 :       N{{ m_inpoel[e*4+0], m_inpoel[e*4+1], m_inpoel[e*4+2], m_inpoel[e*4+3] }};
     114                 :            :     // compute element Jacobi determinant
     115                 :            :     const std::array< tk::real, 3 >
     116                 :      93272 :       ba{{ X[N[1]]-X[N[0]], Y[N[1]]-Y[N[0]], Z[N[1]]-Z[N[0]] }},
     117                 :      93272 :       ca{{ X[N[2]]-X[N[0]], Y[N[2]]-Y[N[0]], Z[N[2]]-Z[N[0]] }},
     118                 :      93272 :       da{{ X[N[3]]-X[N[0]], Y[N[3]]-Y[N[0]], Z[N[3]]-Z[N[0]] }};
     119                 :            :     const auto J = tk::triple( ba, ca, da );        // J = 6V
     120                 :            :     Assert( J > 0, "Element Jacobian non-positive" );
     121                 :            : 
     122                 :            :     // shape function derivatives, nnode*ndim [4][3]
     123                 :            :     std::array< std::array< tk::real, 3 >, 4 > grad;
     124                 :            :     grad[1] = tk::crossdiv( ca, da, J );
     125                 :      93272 :     grad[2] = tk::crossdiv( da, ba, J );
     126                 :      93272 :     grad[3] = tk::crossdiv( ba, ca, J );
     127         [ +  + ]:     373088 :     for (std::size_t i=0; i<3; ++i)
     128                 :     279816 :       grad[0][i] = -grad[1][i]-grad[2][i]-grad[3][i];
     129                 :            : 
     130         [ +  + ]:     466360 :     for (std::size_t a=0; a<4; ++a)
     131         [ +  + ]:    1492352 :       for (std::size_t k=0; k<3; ++k)
     132         [ +  + ]:    5596320 :          for (std::size_t b=0; b<4; ++b)
     133         [ +  + ]:   17838144 :            for (std::size_t i=0; i<ncomp; ++i)
     134         [ +  - ]:   13361088 :              A(N[a],N[b],i) -= J/6 * grad[a][k] * grad[b][k];
     135                 :            :   }
     136                 :            : 
     137         [ +  - ]:         21 :   auto npoin = coord[0].size();
     138 [ +  - ][ +  - ]:         21 :   std::vector< tk::real > b(npoin*ncomp,0.0), x(npoin*ncomp,0.0);
                 [ -  - ]
     139                 :            : 
     140                 :         21 :   return { std::move(A), std::move(x), std::move(b) };
     141                 :            : }
     142                 :            : 
     143                 :            : decltype(ALE::m_move)
     144                 :         31 : ALE::moveCfg()
     145                 :            : // *****************************************************************************
     146                 :            : // Initialize user-defined functions for ALE moving sides
     147                 :            : //! \details This function fills in only part of the data structure
     148                 :            : //!   returned, containing the user-defined functions in discrete form that will
     149                 :            : //!   be sampled in time. The node lists will be initialized later.
     150                 :            : // *****************************************************************************
     151                 :            : {
     152                 :            :   decltype(m_move) cfg;
     153                 :            : 
     154         [ +  + ]:         39 :   for (const auto& m : g_inputdeck.get< tag::ale, tag::move >()) {
     155                 :            :     const auto& fn = m.get< tag::fn >();
     156                 :            :     Assert( fn.size() % 4 == 0, "Incomplete user-defined function" );
     157         [ +  - ]:          8 :     cfg.emplace_back();
     158                 :            :     // store user-defined function type
     159                 :          8 :     std::get<0>(cfg.back()) = m.get< tag::fntype >();
     160                 :            :     // store user-defined function discrete data
     161         [ +  + ]:       1204 :     for (std::size_t i=0; i<fn.size()/4; ++i)
     162                 :            :       std::get<1>(cfg.back()).
     163         [ +  - ]:       1196 :         push_back( {{ fn[i*4+0], fn[i*4+1], fn[i*4+2], fn[i*4+3] }} );
     164                 :            :   }
     165                 :            : 
     166                 :         31 :   return cfg;
     167                 :            : }
     168                 :            : 
     169                 :            : void
     170                 :       1111 : ALE::meshvelBnd(
     171                 :            :   const std::map< int, std::vector< std::size_t > >& bface,
     172                 :            :   const std::map< int, std::vector< std::size_t > >& bnode,
     173                 :            :   const std::vector< std::size_t >& triinpoel )
     174                 :            : // *****************************************************************************
     175                 :            : //  Query mesh velocity boundary conditions node lists and node list at which
     176                 :            : //  ALE moves boundaries
     177                 :            : //! \param[in] bface Boundary-faces mapped to side sets used in the input file
     178                 :            : //! \param[in] bnode Boundary-node lists mapped to side sets used in input file
     179                 :            : //! \param[in] triinpoel Boundary-face connectivity where BCs set
     180                 :            : // *****************************************************************************
     181                 :            : {
     182                 :            :   // Prepare unique set of mesh velocity Dirichlet BC nodes
     183                 :       1111 :   tk::destroy( m_meshveldirbcnodes );
     184                 :            :   std::unordered_map<int, std::unordered_set< std::size_t >> meshveldirbcnodes;
     185         [ +  + ]:       3333 :   for (const auto& s : g_inputdeck.template get< tag::ale, tag::dirichlet >()) {
     186                 :       2222 :     auto k = bface.find(static_cast<int>(s));
     187         [ +  + ]:       2222 :     if (k != end(bface)) {
     188         [ +  - ]:       1054 :       auto& n = meshveldirbcnodes[ k->first ];  // associate set id
     189         [ +  + ]:      65844 :       for (auto f : k->second) {                // face ids on side set
     190         [ +  - ]:      64790 :         n.insert( triinpoel[f*3+0] );
     191         [ +  - ]:      64790 :         n.insert( triinpoel[f*3+1] );
     192         [ +  - ]:      64790 :         n.insert( triinpoel[f*3+2] );
     193                 :            :       }
     194                 :            :     }
     195                 :            :   }
     196         [ +  + ]:       2165 :   for (const auto& [s,nodes] : meshveldirbcnodes)
     197                 :            :     m_meshveldirbcnodes.insert( begin(nodes), end(nodes) );
     198                 :            : 
     199                 :            :   // Prepare unique set of mesh velocity symmetry BC nodes. Note that somewhat
     200                 :            :   // counter-intuitively, we interrogate the boundary nodes instead of boundary
     201                 :            :   // faces here. This is because if we query the boundary faces, then we will
     202                 :            :   // get the mathematically correctly defined finite discrete surfaces
     203                 :            :   // (triangles) where mesh velocity symmetry BCs are configured by the user.
     204                 :            :   // However, in parallel, decomposing the domain and the boundary in various
     205                 :            :   // ways can produce situations on the boundary where boundary nodes are part
     206                 :            :   // of the given side set for mesh velocity symmetry BCs but not a full
     207                 :            :   // triangle face because, not all 3 nodes lie on the boundary. Thus
     208                 :            :   // interrogating the boundary nodes will be a superset and will include those
     209                 :            :   // nodes that are part of imposing symmetry BCs on nodes of faces that are
     210                 :            :   // only partial due to domain decomposition.
     211                 :       1111 :   tk::destroy( m_meshvelsymbcnodes );
     212                 :            :   std::unordered_map<int, std::unordered_set< std::size_t >> meshvelsymbcnodes;
     213         [ +  + ]:       3343 :   for (const auto& s : g_inputdeck.template get< tag::ale, tag::symmetry >()) {
     214                 :       2232 :     auto k = bnode.find(static_cast<int>(s));
     215         [ +  + ]:       2232 :     if (k != end(bnode)) {
     216         [ +  - ]:       1674 :       auto& n = meshvelsymbcnodes[ k->first ];  // associate set id
     217         [ +  + ]:     918065 :       for (auto g : k->second) {                // node ids on side set
     218                 :            :         n.insert( tk::cref_find(m_lid,g) );     // store local ids
     219                 :            :       }
     220                 :            :     }
     221                 :            :   }
     222         [ +  + ]:       2785 :   for (const auto& [s,nodes] : meshvelsymbcnodes)
     223                 :            :     m_meshvelsymbcnodes.insert( begin(nodes), end(nodes) );
     224                 :            : 
     225                 :            :   // Prepare unique sets of boundary nodes at which ALE moves the boundary
     226                 :            :   // based on user-defined functions.
     227                 :            :   std::unordered_map< int, std::unordered_set< std::size_t > > movenodes;
     228                 :            :   std::size_t i = 0;
     229         [ +  + ]:       1359 :   for (const auto& m : g_inputdeck.get< tag::ale, tag::move >()) {
     230         [ +  + ]:        496 :     for (const auto& s : m.get< tag::sideset >()) {
     231                 :        248 :       auto k = bnode.find(static_cast<int>(s));
     232         [ +  + ]:        248 :       if (k != end(bnode)) {
     233         [ +  - ]:        186 :         auto& n = movenodes[ k->first ];        // associate set id
     234         [ +  + ]:       9145 :         for (auto g : k->second) {              // node ids on side set
     235                 :            :           n.insert( tk::cref_find(m_lid,g) );   // store local ids
     236                 :            :         }
     237                 :            :       }
     238                 :            :     }
     239                 :            :     // store all nodes from multiple side sets moved by this usrdef fn
     240                 :        248 :     auto& n = std::get<2>(m_move[i]);
     241                 :            :     n.clear();
     242         [ +  + ]:        434 :     for (const auto& [s,nodes] : movenodes) n.insert(begin(nodes), end(nodes));
     243                 :            :     // increment move ... end configuration block counter
     244                 :        248 :     ++i;
     245                 :            :   }
     246                 :       1111 : }
     247                 :            : 
     248                 :            : bool
     249                 :      67301 : ALE::move( std::size_t i ) const
     250                 :            : // *****************************************************************************
     251                 :            : // Find Dirichlet BCs on mesh velocity with prescribed movement
     252                 :            : //! \param[in] i Local node id to check
     253                 :            : //! \return True of node falls on a boundary that is prescribed to move
     254                 :            : // *****************************************************************************
     255                 :            : {
     256         [ +  + ]:      67301 :   for (const auto& m : m_move)
     257         [ -  + ]:       2728 :     if (std::get<2>(m).find(i) != end(std::get<2>(m)))
     258                 :            :       return true;
     259                 :            : 
     260                 :            :   return false;
     261                 :            : }
     262                 :            : 
     263                 :            : bool
     264                 :        801 : ALE::converged() const
     265                 :            : // *****************************************************************************
     266                 :            : //  Query the solution of the Conjugrate Gradients linear solver
     267                 :            : //! \return Solution to the Conjugate Gradients linear solve
     268                 :            : // *****************************************************************************
     269                 :            : {
     270         [ -  + ]:       1602 :   return m_conjugategradients[ thisIndex ].ckLocal()->converged();
     271                 :            : }
     272                 :            : 
     273                 :            : void
     274                 :       1111 : ALE::start(
     275                 :            :   const tk::UnsMesh::Coords vel,
     276                 :            :   const std::vector< tk::real >& soundspeed,
     277                 :            :   CkCallback done,
     278                 :            :   const std::array< std::vector< tk::real >, 3 >& coord,
     279                 :            :   const tk::UnsMesh::Coords coordn,
     280                 :            :   const std::vector< tk::real >& vol0,
     281                 :            :   const std::vector< tk::real >& vol,
     282                 :            :   const std::unordered_map< int,
     283                 :            :     std::unordered_map< std::size_t, std::array< tk::real, 4 > > >& bnorm,
     284                 :            :   std::size_t initial,
     285                 :            :   std::size_t it,
     286                 :            :   tk::real t,
     287                 :            :   tk::real adt )
     288                 :            : // *****************************************************************************
     289                 :            : // Start computing new mesh velocity for ALE mesh motion
     290                 :            : //! \param[in] vel Fluid velocity at mesh nodes
     291                 :            : //! \param[in] soundspeed Speed of sound at mesh nodes
     292                 :            : //! \param[in] done Function to continue with when mesh velocity has been
     293                 :            : //!   computed
     294                 :            : //! \param[in] coord Mesh node coordinates
     295                 :            : //! \param[in] coordn Mesh node coordinates at the previous time step
     296                 :            : //! \param[in] vol0 Nodal mesh volumes at t=t0
     297                 :            : //! \param[in] vol Nodal mesh volumes
     298                 :            : //! \param[in] bnorm Face normals in boundary points associated to side sets
     299                 :            : //! \param[in] initial Nonzero during the first time step stage, zero otherwise
     300                 :            : //! \param[in] it Iteration count
     301                 :            : //! \param[in] t Physics time
     302                 :            : //! \param[in] adt alpha*dt of the RK time step
     303                 :            : // *****************************************************************************
     304                 :            : {
     305                 :       1111 :   m_done = done;
     306                 :            : 
     307                 :            :   m_coord = coord;
     308                 :       1111 :   m_soundspeed = soundspeed;
     309                 :       1111 :   m_vol0 = vol0;
     310                 :       1111 :   m_vol = vol;
     311                 :            :   m_bnorm = bnorm;
     312                 :       1111 :   m_it = it;
     313                 :       1111 :   m_t = t;
     314                 :       1111 :   m_adt = adt;
     315                 :            : 
     316                 :            :   // assign mesh velocity
     317                 :       1111 :   auto meshveltype = g_inputdeck.get< tag::ale, tag::mesh_velocity >();
     318         [ +  + ]:       1111 :   if (meshveltype == ctr::MeshVelocityType::SINE) {
     319                 :            : 
     320                 :            :     // prescribe mesh velocity with a sine function during setup
     321         [ +  + ]:        279 :     if (initial)
     322         [ +  + ]:       1812 :       for (std::size_t i=0; i<m_w.nunk(); ++i)
     323                 :       1803 :         m_w(i,0) = std::pow( std::sin(coord[0][i]*M_PI), 2.0 );
     324                 :            : 
     325         [ +  + ]:        832 :   } else if (meshveltype == ctr::MeshVelocityType::FLUID) {
     326                 :            : 
     327                 :            :     // equate mesh velocity with fluid velocity
     328         [ +  + ]:       1726 :     for (auto j : g_inputdeck.get< tag::ale, tag::mesh_motion >())
     329         [ +  + ]:    1108793 :       for (std::size_t i=0; i<vel[j].size(); ++i)
     330                 :    1107651 :         m_w(i,j) = vel[j][i];
     331                 :            : 
     332         [ +  - ]:        248 :   } else if (meshveltype == ctr::MeshVelocityType::USER_DEFINED) {
     333                 :            : 
     334                 :            :     // assign mesh velocity to sidesets from user-defined functions
     335         [ +  + ]:        496 :     for (const auto& m : m_move)
     336         [ +  + ]:        248 :       if (std::get<0>(m) == tk::ctr::UserTableType::VELOCITY) {
     337                 :        124 :         auto meshvel = tk::sample<3>( t, std::get<1>(m) );
     338         [ +  + ]:       1922 :         for (auto i : std::get<2>(m))
     339         [ +  + ]:       5394 :           for (auto j : g_inputdeck.get< tag::ale, tag::mesh_motion >())
     340                 :       3596 :             m_w(i,j) = meshvel[j];
     341         [ +  - ]:        124 :       } else if (std::get<0>(m) == tk::ctr::UserTableType::POSITION) {
     342                 :            :         auto eps = std::numeric_limits< tk::real >::epsilon();
     343         [ +  + ]:        124 :         if (adt > eps) {      // dt == 0 during setup
     344                 :        120 :           auto pos = tk::sample<3>( t+adt, std::get<1>(m) );
     345         [ +  + ]:       1440 :           for (auto i : std::get<2>(m))
     346         [ +  + ]:       3960 :             for (auto j : g_inputdeck.get< tag::ale, tag::mesh_motion >())
     347                 :       2640 :               m_w(i,j) = (m_coord0[j][i] + pos[j] - coordn[j][i]) / adt;
     348                 :            :         }
     349                 :            :       }
     350                 :            : 
     351                 :            :   }
     352                 :            : 
     353                 :            :   // start computing the fluid vorticity
     354         [ +  + ]:       2222 :   m_vorticity = tk::curl( coord, m_inpoel, vel );
     355                 :            :   // communicate vorticity sums to other chares on chare-boundary
     356         [ +  + ]:       1111 :   if (m_nodeCommMap.empty()) {
     357                 :        154 :     comvort_complete();
     358                 :            :   } else {
     359         [ +  + ]:       3927 :     for (const auto& [c,n] : m_nodeCommMap) {
     360         [ +  - ]:       2970 :       std::vector< std::array< tk::real, 3 > > v( n.size() );
     361                 :            :       std::size_t j = 0;
     362         [ +  + ]:     300090 :       for (auto i : n) {
     363                 :     297120 :         auto lid = tk::cref_find( m_lid, i );
     364                 :     297120 :         v[j][0] = m_vorticity[0][lid];
     365                 :     297120 :         v[j][1] = m_vorticity[1][lid];
     366                 :     297120 :         v[j][2] = m_vorticity[2][lid];
     367                 :     297120 :         ++j;
     368                 :            :       }
     369 [ +  - ][ +  - ]:       5940 :       thisProxy[c].comvort( std::vector<std::size_t>(begin(n),end(n)), v );
         [ +  - ][ -  + ]
         [ +  - ][ -  - ]
     370                 :            :     }
     371                 :            :   }
     372                 :       1111 :   ownvort_complete();
     373                 :            : 
     374                 :            :   // start computing the fluid velocity divergence
     375         [ +  + ]:       2222 :   m_veldiv = tk::div( coord, m_inpoel, vel );
     376                 :            :   // communicate vorticity sums to other chares on chare-boundary
     377         [ +  + ]:       1111 :   if (m_nodeCommMap.empty()) {
     378                 :        154 :     comdiv_complete();
     379                 :            :   } else {
     380         [ +  + ]:       3927 :     for (const auto& [c,n] : m_nodeCommMap) {
     381         [ +  - ]:       2970 :       std::vector< tk::real > v( n.size() );
     382                 :            :       std::size_t j = 0;
     383         [ +  + ]:     300090 :       for (auto i : n) v[j++] = m_veldiv[ tk::cref_find( m_lid, i ) ];
     384 [ +  - ][ +  - ]:       5940 :       thisProxy[c].comdiv( std::vector<std::size_t>(begin(n),end(n)), v );
         [ +  - ][ -  + ]
         [ +  - ][ -  - ]
     385                 :            :     }
     386                 :            :   }
     387                 :       1111 :   owndiv_complete();
     388                 :       1111 : }
     389                 :            : 
     390                 :            : void
     391                 :       2970 : ALE::comvort( const std::vector< std::size_t >& gid,
     392                 :            :               const std::vector< std::array< tk::real, 3 > >& v )
     393                 :            : // *****************************************************************************
     394                 :            : //  Receive contributions to vorticity on chare-boundaries
     395                 :            : //! \param[in] gid Global mesh node IDs at which we receive contributions
     396                 :            : //! \param[in] v Partial contributions to chare-boundary nodes
     397                 :            : // *****************************************************************************
     398                 :            : {
     399                 :            :   Assert( v.size() == gid.size(), "Size mismatch" );
     400                 :            :   using tk::operator+=;
     401         [ +  + ]:     300090 :   for (std::size_t i=0; i<gid.size(); ++i) m_vorticityc[ gid[i] ] += v[i];
     402                 :            : 
     403                 :            :   // When we have heard from all chares we communicate with, this chare is done
     404         [ +  + ]:       2970 :   if (++m_nvort == m_nodeCommMap.size()) {
     405                 :        957 :     m_nvort = 0;
     406                 :        957 :     comvort_complete();
     407                 :            :   }
     408                 :       2970 : }
     409                 :            : 
     410                 :            : void
     411                 :       2970 : ALE::comdiv( const std::vector< std::size_t >& gid,
     412                 :            :              const std::vector< tk::real >& v )
     413                 :            : // *****************************************************************************
     414                 :            : //  Receive contributions to velocity divergence on chare-boundaries
     415                 :            : //! \param[in] gid Global mesh node IDs at which we receive RHS contributions
     416                 :            : //! \param[in] v Partial contributions to chare-boundary nodes
     417                 :            : // *****************************************************************************
     418                 :            : {
     419                 :            :   Assert( v.size() == gid.size(), "Size mismatch" );
     420                 :            :   using tk::operator+=;
     421         [ +  + ]:     300090 :   for (std::size_t i=0; i<gid.size(); ++i) m_veldivc[ gid[i] ] += v[i];
     422                 :            : 
     423                 :            :   // When we have heard from all chares we communicate with, this chare is done
     424         [ +  + ]:       2970 :   if (++m_ndiv == m_nodeCommMap.size()) {
     425                 :        957 :     m_ndiv = 0;
     426                 :        957 :     comdiv_complete();
     427                 :            :   }
     428                 :       2970 : }
     429                 :            : 
     430                 :            : void
     431                 :       1111 : ALE::mergevel()
     432                 :            : // *****************************************************************************
     433                 :            : // Finalize computing fluid vorticity and velocity divergence for ALE
     434                 :            : // *****************************************************************************
     435                 :            : {
     436                 :            :   // combine own and communicated contributions to vorticity
     437         [ +  + ]:     259140 :   for (const auto& [g,v] : m_vorticityc) {
     438                 :     258029 :     auto lid = tk::cref_find( m_lid, g );
     439                 :     258029 :     m_vorticity[0][lid] += v[0];
     440                 :     258029 :     m_vorticity[1][lid] += v[1];
     441                 :     258029 :     m_vorticity[2][lid] += v[2];
     442                 :            :   }
     443                 :            :   // clear fluid vorticity receive buffer
     444                 :       1111 :   tk::destroy(m_vorticityc);
     445                 :            : 
     446                 :            :   // finish computing vorticity dividing the weak sum by the nodal volumes
     447         [ +  + ]:       4444 :   for (std::size_t j=0; j<3; ++j)
     448         [ +  + ]:    3975612 :     for (std::size_t p=0; p<m_vorticity[j].size(); ++p)
     449                 :    3972279 :       m_vorticity[j][p] /= m_vol[p];
     450                 :            : 
     451                 :            :   // compute vorticity magnitude
     452         [ +  + ]:    1325204 :   for (std::size_t p=0; p<m_vorticity[0].size(); ++p)
     453                 :    1324093 :     m_vorticity[0][p] =
     454                 :    1324093 :       tk::length( m_vorticity[0][p], m_vorticity[1][p], m_vorticity[2][p] );
     455                 :            : 
     456                 :            :   // get rid of the y and z vorticity components, since we just overwrote the
     457                 :            :   // x component with the magnitude
     458                 :            :   tk::destroy( m_vorticity[1] );
     459                 :            :   tk::destroy( m_vorticity[2] );
     460                 :            : 
     461                 :            :   // compute max vorticity magnitude
     462                 :            :   auto maxv =
     463         [ +  - ]:       2222 :     *std::max_element( m_vorticity[0].cbegin(), m_vorticity[0].cend() );
     464                 :            : 
     465                 :            :   // combine own and communicated contributions to velocidy divergence
     466         [ +  + ]:     259140 :   for (const auto& [g,v] : m_veldivc)
     467                 :     258029 :     m_veldiv[ tk::cref_find( m_lid, g ) ] += v;
     468                 :            :   // clear velocity divergence receive buffer
     469                 :       1111 :   tk::destroy(m_veldivc);
     470                 :            : 
     471                 :            :   // finish computing velocity divergence dividing weak sum by the nodal volumes
     472         [ +  + ]:    1325204 :   for (std::size_t p=0; p<m_veldiv.size(); ++p) m_veldiv[p] /= m_vol[p];
     473                 :            : 
     474                 :            :   // Compute max vorticity magnitude across all chares
     475         [ +  - ]:       1111 :   contribute( sizeof(tk::real), &maxv, CkReduction::max_double,
     476                 :       1111 :               CkCallback(CkReductionTarget(ALE,meshvelbc), thisProxy) );
     477                 :       1111 : }
     478                 :            : 
     479                 :            : void
     480                 :       1111 : ALE::meshvelbc( tk::real maxv )
     481                 :            : // *****************************************************************************
     482                 :            : // Apply mesh velocity smoother boundary conditions for ALE mesh motion
     483                 :            : //! \param[in] maxv The largest vorticity magnitude across the whole problem
     484                 :            : // *****************************************************************************
     485                 :            : {
     486                 :            :   std::size_t ignorebc = false;
     487                 :            : 
     488                 :            :   // smooth mesh velocity if needed
     489                 :       1111 :   auto smoother = g_inputdeck.get< tag::ale, tag::smoother >();
     490                 :            : 
     491         [ +  + ]:       1111 :   if (smoother == ctr::MeshVelocitySmootherType::LAPLACE) {
     492                 :            : 
     493                 :            :     // scale mesh velocity with a function of the fluid vorticity
     494         [ +  + ]:        677 :     if (maxv > 1.0e-8) {
     495                 :        652 :       auto mult = g_inputdeck.get< tag::ale, tag::vortmult >();
     496         [ +  + ]:       1780 :       for (auto j : g_inputdeck.get< tag::ale, tag::mesh_motion >())
     497         [ +  + ]:    1593997 :         for (std::size_t p=0; p<m_vorticity[0].size(); ++p)
     498         [ +  + ]:    3185645 :           m_w(p,j) *= std::max( 0.0, 1.0 - mult*m_vorticity[0][p]/maxv );
     499                 :            :     }
     500                 :            : 
     501                 :            :     // Set mesh velocity smoother linear solve boundary conditions
     502                 :            :     std::unordered_map< std::size_t,
     503                 :            :       std::vector< std::pair< bool, tk::real > > > wbc;
     504                 :            : 
     505                 :            :     // Dirichlet BCs on mesh velocity with prescribed movement
     506         [ +  + ]:        925 :     for (const auto& m : m_move)
     507         [ +  + ]:        248 :       if (std::get<0>(m) == tk::ctr::UserTableType::VELOCITY) {
     508                 :        124 :         auto meshvel = tk::sample<3>( m_t, std::get<1>(m) );
     509         [ +  + ]:       1922 :         for (auto i : std::get<2>(m))
     510         [ +  + ]:       5394 :           for (auto j : g_inputdeck.get< tag::ale, tag::mesh_motion >())
     511                 :       3596 :             m_w(i,j) = meshvel[j];
     512         [ +  - ]:        124 :       } else if (std::get<0>(m) == tk::ctr::UserTableType::POSITION) {
     513                 :            :         auto eps = std::numeric_limits< tk::real >::epsilon();
     514         [ +  + ]:        124 :         if (m_adt > eps) {
     515                 :        120 :           ignorebc = m_it > 0;
     516         [ +  + ]:       1440 :           for (auto i : std::get<2>(m))
     517         [ +  - ]:       1320 :             if (m_meshveldirbcnodes.find(i) != end(m_meshveldirbcnodes))
     518         [ +  - ]:       2640 :               wbc[i] = {{ {false,0}, {false,0}, {false,0} }};
     519                 :            :         }
     520                 :            :       }
     521                 :            : 
     522                 :            :     // Dirichlet BCs where user specified mesh velocity BCs
     523         [ +  + ]:      30375 :     for (auto i : m_meshveldirbcnodes)
     524 [ +  - ][ +  + ]:      58032 :       if (not move(i)) wbc[i] = {{ {true,0}, {true,0}, {true,0} }};
                 [ +  - ]
     525                 :            : 
     526                 :            :     // initialize mesh velocity smoother linear solver
     527                 :        677 :     m_conjugategradients[ thisIndex ].ckLocal()->
     528 [ +  - ][ +  - ]:       2708 :       init( m_w.flat(), {}, wbc, ignorebc,
         [ +  - ][ -  + ]
                 [ -  - ]
     529 [ +  - ][ +  - ]:       2031 :             CkCallback(CkIndex_ALE::applied(nullptr), thisProxy[thisIndex]) );
         [ -  + ][ -  + ]
         [ -  - ][ -  - ]
     530                 :            : 
     531         [ +  + ]:        434 :   } else if (smoother == ctr::MeshVelocitySmootherType::HELMHOLTZ) {
     532                 :            : 
     533                 :            :     // Set scalar potential linear solve boundary conditions
     534                 :            :     std::unordered_map< std::size_t,
     535                 :            :       std::vector< std::pair< bool, tk::real > > > pbc;
     536                 :            : 
     537                 :            :     // Dirichlet BCs where user specified mesh velocity BCs
     538 [ +  + ][ +  - ]:       8029 :     for (auto i : m_meshveldirbcnodes) pbc[i] = {{ {true,0} }};
     539                 :            : 
     540                 :            :     // prepare velocity divergence as weak sum required for Helmholtz solve
     541         [ +  - ]:        124 :     decltype(m_veldiv) wveldiv = m_veldiv;
     542         [ +  + ]:      10013 :     for (std::size_t p=0; p<wveldiv.size(); ++p) wveldiv[p] *= m_vol[p];
     543                 :            : 
     544                 :            :     // initialize Helmholtz decomposition linear solver
     545                 :        124 :     m_conjugategradients[ thisIndex ].ckLocal()->
     546 [ +  - ][ +  - ]:        496 :       init( {}, wveldiv, pbc, ignorebc,
         [ +  - ][ -  - ]
     547 [ +  - ][ +  - ]:        372 :             CkCallback(CkIndex_ALE::applied(nullptr), thisProxy[thisIndex]) );
         [ -  + ][ -  + ]
         [ -  - ][ -  - ]
     548                 :            : 
     549                 :            :   } else {
     550                 :            : 
     551                 :            :     // continue
     552                 :        310 :     applied();
     553                 :            : 
     554                 :            :   }
     555                 :       1111 : }
     556                 :            : 
     557                 :            : void
     558                 :       1111 : ALE::applied( [[maybe_unused]] CkDataMsg* msg )
     559                 :            : // *****************************************************************************
     560                 :            : // Solve mesh velocity linear solve for ALE mesh motion
     561                 :            : // *****************************************************************************
     562                 :            : {
     563                 :            :   //if (msg != nullptr) {
     564                 :            :   //  auto *norm = static_cast< tk::real * >( msg->getData() );
     565                 :            :   //  std::cout << "applied: " << *norm << '\n';
     566                 :            :   //}
     567                 :            : 
     568                 :       1111 :   auto smoother = g_inputdeck.get< tag::ale, tag::smoother >();
     569                 :            : 
     570         [ +  + ]:       1111 :   if (smoother == ctr::MeshVelocitySmootherType::LAPLACE) {
     571                 :            : 
     572         [ +  - ]:       2031 :     m_conjugategradients[ thisIndex ].ckLocal()->solve(
     573                 :            :        g_inputdeck.get< tag::ale, tag::maxit >(),
     574                 :            :        g_inputdeck.get< tag::ale, tag::tolerance >(),
     575 [ +  - ][ +  - ]:       2031 :        CkCallback(CkIndex_ALE::solved(nullptr), thisProxy[thisIndex]) );
         [ -  + ][ -  + ]
         [ -  - ][ -  - ]
     576                 :            : 
     577         [ +  + ]:        434 :   } else if (smoother == ctr::MeshVelocitySmootherType::HELMHOLTZ) {
     578                 :            : 
     579         [ +  - ]:        372 :     m_conjugategradients[ thisIndex ].ckLocal()->solve(
     580                 :            :        g_inputdeck.get< tag::ale, tag::maxit >(),
     581                 :            :        g_inputdeck.get< tag::ale, tag::tolerance >(),
     582 [ +  - ][ +  - ]:        372 :        CkCallback(CkIndex_ALE::helmholtz(nullptr), thisProxy[thisIndex]) );
         [ -  + ][ -  + ]
         [ -  - ][ -  - ]
     583                 :            : 
     584                 :            :   } else {
     585                 :            : 
     586                 :        310 :     solved();
     587                 :            : 
     588                 :            :   }
     589                 :       1111 : }
     590                 :            : 
     591                 :            : void
     592                 :        124 : ALE::helmholtz( [[maybe_unused]] CkDataMsg* msg )
     593                 :            : // *****************************************************************************
     594                 :            : //  Compute the gradient of the scalar potential for ALE
     595                 :            : // *****************************************************************************
     596                 :            : {
     597                 :            :   //if (msg != nullptr) {
     598                 :            :   //  auto *norm = static_cast< tk::real * >( msg->getData() );
     599                 :            :   //  std::cout << "solved: " << *norm << '\n';
     600                 :            :   //}
     601                 :            : 
     602                 :            :   // compute gradient of scalar potential for ALE (own portion)
     603 [ +  - ][ +  - ]:        248 :   m_gradpot = tk::grad( m_coord, m_inpoel,
     604 [ -  + ][ -  + ]:        372 :                  m_conjugategradients[ thisIndex ].ckLocal()->solution() );
                 [ -  - ]
     605                 :            : 
     606                 :            :   // communicate scalar potential sums to other chares on chare-boundary
     607         [ -  + ]:        124 :   if (m_nodeCommMap.empty())
     608                 :          0 :     compot_complete();
     609                 :            :   else
     610         [ +  + ]:        496 :     for (const auto& [c,n] : m_nodeCommMap) {
     611         [ +  - ]:        372 :       std::vector< std::array< tk::real, 3 > > v( n.size() );
     612                 :            :       std::size_t j = 0;
     613         [ +  + ]:       6386 :       for (auto i : n) {
     614                 :       6014 :         auto lid = tk::cref_find( m_lid, i );
     615                 :       6014 :         v[j][0] = m_gradpot[0][lid];
     616                 :       6014 :         v[j][1] = m_gradpot[1][lid];
     617                 :       6014 :         v[j][2] = m_gradpot[2][lid];
     618                 :       6014 :         ++j;
     619                 :            :       }
     620 [ +  - ][ +  - ]:        744 :       thisProxy[c].compot( std::vector<std::size_t>(begin(n),end(n)), v );
         [ +  - ][ -  + ]
         [ +  - ][ -  - ]
     621                 :            :     }
     622                 :        124 :   ownpot_complete();
     623                 :        124 : }
     624                 :            : 
     625                 :            : void
     626                 :        372 : ALE::compot( const std::vector< std::size_t >& gid,
     627                 :            :                const std::vector< std::array< tk::real, 3 > >& v )
     628                 :            : // *****************************************************************************
     629                 :            : //  Receive contributions to scalar potential gradient on chare-boundaries
     630                 :            : //! \param[in] gid Global mesh node IDs at which we receive contributions
     631                 :            : //! \param[in] v Partial contributions to chare-boundary nodes
     632                 :            : // *****************************************************************************
     633                 :            : {
     634                 :            :   Assert( v.size() == gid.size(), "Size mismatch" );
     635                 :            :   using tk::operator+=;
     636         [ +  + ]:       6386 :   for (std::size_t i=0; i<gid.size(); ++i) m_gradpotc[ gid[i] ] += v[i];
     637                 :            : 
     638                 :            :   // When we have heard from all chares we communicate with, this chare is done
     639         [ +  + ]:        372 :   if (++m_npot == m_nodeCommMap.size()) {
     640                 :        124 :     m_npot = 0;
     641                 :        124 :     compot_complete();
     642                 :            :   }
     643                 :        372 : }
     644                 :            : 
     645                 :            : void
     646                 :        124 : ALE::gradpot()
     647                 :            : // *****************************************************************************
     648                 :            : // Finalize computing gradient of the scalar potential for ALE
     649                 :            : // *****************************************************************************
     650                 :            : {
     651                 :            :   // combine own and communicated contributions to scalar potential gradient
     652         [ +  + ]:       4712 :   for (const auto& [g,v] : m_gradpotc) {
     653                 :       4588 :     auto lid = tk::cref_find( m_lid, g );
     654                 :       4588 :     m_gradpot[0][lid] += v[0];
     655                 :       4588 :     m_gradpot[1][lid] += v[1];
     656                 :       4588 :     m_gradpot[2][lid] += v[2];
     657                 :            :   }
     658                 :            :   // clear receive buffer
     659                 :        124 :   tk::destroy(m_gradpotc);
     660                 :            : 
     661                 :            :   // finish computing the gradient dividing weak sum by the nodal volumes
     662         [ +  + ]:        496 :   for (auto j : g_inputdeck.get< tag::ale, tag::mesh_motion >())
     663         [ +  + ]:      30039 :     for (std::size_t p=0; p<m_gradpot[j].size(); ++p)
     664                 :      29667 :       m_gradpot[j][p] /= m_vol[p];
     665                 :            : 
     666                 :        124 :   solved();
     667                 :        124 : }
     668                 :            : 
     669                 :            : void
     670                 :       1111 : ALE::solved( [[maybe_unused]] CkDataMsg* msg )
     671                 :            : // *****************************************************************************
     672                 :            : //  Mesh smoother linear solver converged
     673                 :            : // *****************************************************************************
     674                 :            : {
     675                 :            :   //if (msg != nullptr) {
     676                 :            :   //  auto *normres = static_cast< tk::real * >( msg->getData() );
     677                 :            :   //  std::cout << "solved: " << *normres << '\n';
     678                 :            :   //}
     679                 :            : 
     680                 :       1111 :   auto smoother = g_inputdeck.get< tag::ale, tag::smoother >();
     681                 :            : 
     682         [ +  + ]:       1111 :   if (smoother == ctr::MeshVelocitySmootherType::LAPLACE) {
     683                 :            : 
     684                 :            :     // Read out linear solution
     685                 :       1354 :     auto w = m_conjugategradients[ thisIndex ].ckLocal()->solution();
     686                 :            : 
     687                 :            :     // Assign mesh velocity from linear solution skipping dimensions that are
     688                 :            :     // not allowed to move
     689         [ +  + ]:       1850 :     for (auto j : g_inputdeck.get< tag::ale, tag::mesh_motion >())
     690         [ +  + ]:    1653928 :       for (std::size_t i=0; i<m_w.nunk(); ++i)
     691                 :    1652755 :         m_w(i,j) = w[i*m_w.nprop()+j];
     692                 :            : 
     693         [ +  + ]:        434 :   } else if (smoother == ctr::MeshVelocitySmootherType::HELMHOLTZ) {
     694                 :            : 
     695                 :        124 :     auto a1 = g_inputdeck.get< tag::ale, tag::vortmult >();
     696         [ +  + ]:        496 :     for (auto j : g_inputdeck.get< tag::ale, tag::mesh_motion >())
     697         [ +  + ]:      30039 :       for (std::size_t p=0; p<m_w.nunk(); ++p)
     698                 :      29667 :         m_w(p,j) += a1 * (m_gradpot[j][p] - m_w(p,j));
     699                 :            : 
     700                 :            :   }
     701                 :            : 
     702                 :            :   // continue to applying a mesh force to the mesh velocity
     703                 :       1111 :   startforce();
     704                 :       1111 : }
     705                 :            : 
     706                 :            : void
     707                 :       1111 : ALE::startforce()
     708                 :            : // *****************************************************************************
     709                 :            : //  Compute mesh force for the ALE mesh velocity
     710                 :            : //! \details Compute mesh forces. See Sec.4 in Bakosi, Waltz, Morgan, Improved
     711                 :            : //!   ALE mesh velocities for complex flows, International Journal for Numerical
     712                 :            : //!   Methods in Fluids, 2017.
     713                 :            : // *****************************************************************************
     714                 :            : {
     715                 :            :   const auto& x = m_coord[0];
     716                 :            :   const auto& y = m_coord[1];
     717                 :            :   const auto& z = m_coord[2];
     718                 :            : 
     719                 :            :   // compute pseudo-pressure gradient for mesh force
     720                 :            :   const auto& f = g_inputdeck.get< tag::ale, tag::meshforce >();
     721                 :            :   m_wf.fill( 0.0 );
     722         [ +  + ]:    5107021 :   for (std::size_t e=0; e<m_inpoel.size()/4; ++e) {
     723                 :            :     // access node IDs
     724                 :            :     const std::array< std::size_t, 4 >
     725                 :    5105910 :       N{{m_inpoel[e*4+0], m_inpoel[e*4+1], m_inpoel[e*4+2], m_inpoel[e*4+3]}};
     726                 :            :     // compute element Jacobi determinant
     727                 :            :     const std::array< tk::real, 3 >
     728                 :    5105910 :       ba{{ x[N[1]]-x[N[0]], y[N[1]]-y[N[0]], z[N[1]]-z[N[0]] }},
     729                 :    5105910 :       ca{{ x[N[2]]-x[N[0]], y[N[2]]-y[N[0]], z[N[2]]-z[N[0]] }},
     730                 :    5105910 :       da{{ x[N[3]]-x[N[0]], y[N[3]]-y[N[0]], z[N[3]]-z[N[0]] }};
     731                 :            :     const auto J = tk::triple( ba, ca, da );        // J = 6V
     732                 :    5105910 :     auto J24 = J/24.0;
     733                 :            :     Assert( J > 0, "Element Jacobian non-positive" );
     734                 :            : 
     735                 :            :     // shape function derivatives, nnode*ndim [4][3]
     736                 :            :     std::array< std::array< tk::real, 3 >, 4 > grad;
     737                 :    5105910 :     grad[1] = tk::crossdiv( ca, da, J );
     738                 :    5105910 :     grad[2] = tk::crossdiv( da, ba, J );
     739                 :    5105910 :     grad[3] = tk::crossdiv( ba, ca, J );
     740         [ +  + ]:   20423640 :     for (std::size_t i=0; i<3; ++i)
     741                 :   15317730 :       grad[0][i] = -grad[1][i]-grad[2][i]-grad[3][i];
     742                 :            : 
     743                 :            :     // max sound speed across nodes of element
     744                 :    5105910 :     tk::real c = 1.0e-3;        // floor on sound speed
     745 [ +  + ][ +  + ]:   25529550 :     for (std::size_t a=0; a<4; ++a) c = std::max( c, m_soundspeed[N[a]] );
     746                 :            : 
     747                 :            :     // mesh force in nodes
     748                 :    5105910 :     auto V = J/6.0;
     749                 :    5105910 :     auto L = std::cbrt(V);  // element length scale, L=V^(1/3)
     750                 :    5105910 :     auto dv = m_vol0[e] / V;
     751                 :    5105910 :     std::array< tk::real, 4 > q{0,0,0,0};
     752         [ +  + ]:   25529550 :     for (std::size_t a=0; a<4; ++a) {
     753                 :   20423640 :       auto du = L * m_veldiv[N[a]];
     754                 :   20423640 :       q[a] = - du*(f[0]*c + f[1]*std::abs(du) + f[2]*du*du)
     755                 :   20423640 :              + f[3]*c*c*std::abs(dv-1.0);
     756                 :            :     }
     757                 :            : 
     758                 :            :     // scatter add pseudo-pressure gradient
     759         [ +  + ]:   25529550 :     for (std::size_t a=0; a<4; ++a)
     760         [ +  + ]:  102118200 :       for (std::size_t b=0; b<4; ++b)
     761         [ +  + ]:  326778240 :         for (std::size_t i=0; i<3; ++i)
     762                 :  245083680 :           m_wf(N[a],i) += J24 * grad[b][i] * q[b];
     763                 :            :   }
     764                 :            : 
     765                 :            :   // communicate mesh force sums to other chares on chare-boundary
     766         [ +  + ]:       1111 :   if (m_nodeCommMap.empty()) {
     767                 :        154 :     comfor_complete();
     768                 :            :   } else {
     769         [ +  + ]:       3927 :     for (const auto& [c,n] : m_nodeCommMap) {
     770         [ +  - ]:       2970 :       std::vector< std::array< tk::real, 3 > > w( n.size() );
     771                 :            :       std::size_t j = 0;
     772         [ +  + ]:     300090 :       for (auto i : n) {
     773                 :     297120 :         auto lid = tk::cref_find( m_lid, i );
     774                 :     297120 :         w[j][0] = m_wf(lid,0);
     775                 :     297120 :         w[j][1] = m_wf(lid,1);
     776                 :     297120 :         w[j][2] = m_wf(lid,2);
     777                 :     297120 :         ++j;
     778                 :            :       }
     779 [ +  - ][ +  - ]:       5940 :       thisProxy[c].comfor(std::vector<std::size_t>(begin(n),end(n)), w);
         [ +  - ][ -  + ]
         [ +  - ][ -  - ]
     780                 :            :     }
     781                 :            :   }
     782                 :       1111 :   ownfor_complete();
     783                 :       1111 : }
     784                 :            : 
     785                 :            : void
     786                 :       2970 : ALE::comfor( const std::vector< std::size_t >& gid,
     787                 :            :              const std::vector< std::array< tk::real, 3 > >& w )
     788                 :            : // *****************************************************************************
     789                 :            : //  Receive contributions to ALE mesh force on chare-boundaries
     790                 :            : //! \param[in] gid Global mesh node IDs at which we receive contributions
     791                 :            : //! \param[in] w Partial contributions to chare-boundary nodes
     792                 :            : // *****************************************************************************
     793                 :            : {
     794                 :            :   Assert( w.size() == gid.size(), "Size mismatch" );
     795                 :            :   using tk::operator+=;
     796         [ +  + ]:     300090 :   for (std::size_t i=0; i<gid.size(); ++i) m_wfc[ gid[i] ] += w[i];
     797                 :            : 
     798                 :            :   // When we have heard from all chares we communicate with, this chare is done
     799         [ +  + ]:       2970 :   if (++m_nwf == m_nodeCommMap.size()) {
     800                 :        957 :     m_nwf = 0;
     801                 :        957 :     comfor_complete();
     802                 :            :   }
     803                 :       2970 : }
     804                 :            : 
     805                 :            : void
     806                 :       1111 : ALE::meshforce()
     807                 :            : // *****************************************************************************
     808                 :            : // Apply mesh force
     809                 :            : // *****************************************************************************
     810                 :            : {
     811                 :            :   // combine own and communicated contributions to mesh force
     812         [ +  + ]:     259140 :   for (const auto& [g,w] : m_wfc) {
     813                 :     258029 :     auto lid = tk::cref_find( m_lid, g );
     814                 :     258029 :     m_wf(lid,0) += w[0];
     815                 :     258029 :     m_wf(lid,1) += w[1];
     816                 :     258029 :     m_wf(lid,2) += w[2];
     817                 :            :   }
     818                 :            :   // clear receive buffer
     819                 :       1111 :   tk::destroy(m_wfc);
     820                 :            : 
     821                 :            :   // finish computing the mesh force by dviding weak sum by the nodal volumes
     822         [ +  + ]:       4444 :   for (std::size_t j=0; j<3; ++j)
     823         [ +  + ]:    3975612 :     for (std::size_t p=0; p<m_wf.nunk(); ++p)
     824                 :    3972279 :       m_wf(p,j) /= m_vol[p];
     825                 :            : 
     826                 :            :   // advance mesh velocity in time due to pseudo-pressure gradient mesh force
     827         [ +  + ]:       3586 :   for (auto j : g_inputdeck.get< tag::ale, tag::mesh_motion >())
     828         [ +  + ]:    1874803 :     for (std::size_t i=0; i<m_w.nunk(); ++i)
     829                 :            :        // This is likely incorrect. It should be m_w = m_w0 + ...
     830                 :    1872328 :        m_w(i,j) += m_adt * m_wf(i,j);
     831                 :            : 
     832                 :            :   // Enforce mesh velocity Dirichlet BCs where user specfied but did not
     833                 :            :   // prescribe a move
     834         [ +  + ]:      38714 :   for (auto i : m_meshveldirbcnodes)
     835         [ +  + ]:      37603 :     if (not move(i)) m_w(i,0) = m_w(i,1) = m_w(i,2) = 0.0;
     836                 :            : 
     837                 :            :   // On meshvel symmetry BCs remove normal component of mesh velocity
     838                 :            :   const auto& sbc = g_inputdeck.get< tag::ale, tag::symmetry >();
     839         [ +  + ]:     297843 :   for (auto p : m_meshvelsymbcnodes) {
     840         [ +  + ]:    2077124 :     for (const auto& s : sbc) {
     841                 :    1780392 :       auto j = m_bnorm.find(static_cast<int>(s));
     842         [ +  + ]:    1780392 :       if (j != end(m_bnorm)) {
     843                 :            :         auto i = j->second.find(p);
     844         [ +  + ]:    1380660 :         if (i != end(j->second)) {
     845                 :            :           std::array< tk::real, 3 >
     846                 :     320550 :             n{ i->second[0], i->second[1], i->second[2] },
     847                 :     320550 :             v{ m_w(p,0), m_w(p,1), m_w(p,2) };
     848                 :            :           auto v_dot_n = tk::dot( v, n );
     849                 :            :           // symbc: remove normal component of mesh velocity
     850                 :     320550 :           m_w(p,0) -= v_dot_n * n[0];
     851                 :     320550 :           m_w(p,1) -= v_dot_n * n[1];
     852                 :     320550 :           m_w(p,2) -= v_dot_n * n[2];
     853                 :            :         }
     854                 :            :       }
     855                 :            :     }
     856                 :            :   }
     857                 :            : 
     858                 :            :   // Activate SDAG wait for re-computing prerequisites for ALE
     859         [ +  - ]:       1111 :   thisProxy[ thisIndex ].wait4vel();
     860         [ +  - ]:       1111 :   thisProxy[ thisIndex ].wait4pot();
     861         [ +  - ]:       1111 :   thisProxy[ thisIndex ].wait4for();
     862                 :            : 
     863                 :       1111 :   m_done.send();
     864                 :       1111 : }
     865                 :            : 
     866                 :            : #include "NoWarning/ale.def.h"

Generated by: LCOV version 1.14