LCOV - code coverage report
Current view: top level - src/grid/common - grid_process_vab.h (source / functions) Hit Total Coverage
Test: CP2K Regtests (git:0de0cc2) Lines: 95 95 100.0 %
Date: 2024-03-28 07:31:50 Functions: 10 10 100.0 %

          Line data    Source code
       1             : /*----------------------------------------------------------------------------*/
       2             : /*  CP2K: A general program to perform molecular dynamics simulations         */
       3             : /*  Copyright 2000-2024 CP2K developers group <https://cp2k.org>              */
       4             : /*                                                                            */
       5             : /*  SPDX-License-Identifier: BSD-3-Clause                                     */
       6             : /*----------------------------------------------------------------------------*/
       7             : 
       8             : #include <stdbool.h>
       9             : 
      10             : #if defined(__CUDACC__) || defined(__HIPCC__)
      11             : #define GRID_DEVICE __device__
      12             : #else
      13             : #define GRID_DEVICE
      14             : #endif
      15             : 
      16             : /*******************************************************************************
      17             :  * \brief Returns matrix element cab[idx(b)][idx(a)].
      18             :  *        This function has to be implemented by the importing compilation unit.
      19             :  *        A simple implementation is just: returns cab[idx(b) * n1 + idx(a)];
      20             :  * \author Ole Schuett
      21             :  ******************************************************************************/
      22             : GRID_DEVICE static inline double cab_get(const cab_store *cab, const orbital a,
      23             :                                          const orbital b);
      24             : 
      25             : /*******************************************************************************
      26             :  * \brief Returns i'th component of force on atom a for compute_tau=false.
      27             :  * \author Ole Schuett
      28             :  ******************************************************************************/
      29             : GRID_DEVICE static inline double
      30   450751794 : get_force_a_normal(const orbital a, const orbital b, const int i,
      31             :                    const double zeta, const cab_store *cab) {
      32   450751794 :   const double aip1 = cab_get(cab, up(i, a), b);
      33   450751794 :   const double aim1 = cab_get(cab, down(i, a), b);
      34   450751794 :   return 2.0 * zeta * aip1 - a.l[i] * aim1;
      35             : }
      36             : 
      37             : /*******************************************************************************
      38             :  * \brief Returns i'th component of force on atom a.
      39             :  * \author Ole Schuett
      40             :  ******************************************************************************/
      41             : GRID_DEVICE static inline double
      42   415473936 : get_force_a(const orbital a, const orbital b, const int i, const double zeta,
      43             :             const double zetb, const cab_store *cab, const bool compute_tau) {
      44   415473936 :   if (!compute_tau) {
      45   412266858 :     return get_force_a_normal(a, b, i, zeta, cab);
      46             :   } else {
      47             :     double force = 0.0;
      48    12828312 :     for (int k = 0; k < 3; k++) {
      49     9621234 :       force += 0.5 * a.l[k] * b.l[k] *
      50     9621234 :                get_force_a_normal(down(k, a), down(k, b), i, zeta, cab);
      51     9621234 :       force -= zeta * b.l[k] *
      52     9621234 :                get_force_a_normal(up(k, a), down(k, b), i, zeta, cab);
      53     9621234 :       force -= a.l[k] * zetb *
      54     9621234 :                get_force_a_normal(down(k, a), up(k, b), i, zeta, cab);
      55     9621234 :       force += 2.0 * zeta * zetb *
      56     9621234 :                get_force_a_normal(up(k, a), up(k, b), i, zeta, cab);
      57             :     }
      58             :     return force;
      59             :   }
      60             : }
      61             : 
      62             : /*******************************************************************************
      63             :  * \brief Returns i'th component of force on atom b for compute_tau=false.
      64             :  * \author Ole Schuett
      65             :  ******************************************************************************/
      66             : GRID_DEVICE static inline double
      67   450725490 : get_force_b_normal(const orbital a, const orbital b, const int i,
      68             :                    const double zetb, const double rab[3],
      69             :                    const cab_store *cab) {
      70   450725490 :   const double axpm0 = cab_get(cab, a, b);
      71   450725490 :   const double aip1 = cab_get(cab, up(i, a), b);
      72   450725490 :   const double bim1 = cab_get(cab, a, down(i, b));
      73   450725490 :   return 2.0 * zetb * (aip1 - rab[i] * axpm0) - b.l[i] * bim1;
      74             : }
      75             : 
      76             : /*******************************************************************************
      77             :  * \brief Returns i'th component of force on atom b.
      78             :  * \author Ole Schuett
      79             :  ******************************************************************************/
      80             : GRID_DEVICE static inline double
      81   415447632 : get_force_b(const orbital a, const orbital b, const int i, const double zeta,
      82             :             const double zetb, const double rab[3], const cab_store *cab,
      83             :             const bool compute_tau) {
      84   415447632 :   if (!compute_tau) {
      85   412240554 :     return get_force_b_normal(a, b, i, zetb, rab, cab);
      86             :   } else {
      87             :     double force = 0.0;
      88    12828312 :     for (int k = 0; k < 3; k++) {
      89     9621234 :       force += 0.5 * a.l[k] * b.l[k] *
      90     9621234 :                get_force_b_normal(down(k, a), down(k, b), i, zetb, rab, cab);
      91     9621234 :       force -= zeta * b.l[k] *
      92     9621234 :                get_force_b_normal(up(k, a), down(k, b), i, zetb, rab, cab);
      93     9621234 :       force -= a.l[k] * zetb *
      94     9621234 :                get_force_b_normal(down(k, a), up(k, b), i, zetb, rab, cab);
      95     9621234 :       force += 2.0 * zeta * zetb *
      96     9621234 :                get_force_b_normal(up(k, a), up(k, b), i, zetb, rab, cab);
      97             :     }
      98             :     return force;
      99             :   }
     100             : }
     101             : 
     102             : /*******************************************************************************
     103             :  * \brief Returns element i,j of virial on atom a for compute_tau=false.
     104             :  * \author Ole Schuett
     105             :  ******************************************************************************/
     106             : GRID_DEVICE static inline double
     107   250076988 : get_virial_a_normal(const orbital a, const orbital b, const int i, const int j,
     108             :                     const double zeta, const cab_store *cab) {
     109   250076988 :   return 2.0 * zeta * cab_get(cab, up(i, up(j, a)), b) -
     110   250076988 :          a.l[j] * cab_get(cab, up(i, down(j, a)), b);
     111             : }
     112             : 
     113             : /*******************************************************************************
     114             :  * \brief Returns element i,j of virial on atom a.
     115             :  * \author Ole Schuett
     116             :  ******************************************************************************/
     117             : GRID_DEVICE static inline double
     118   187270596 : get_virial_a(const orbital a, const orbital b, const int i, const int j,
     119             :              const double zeta, const double zetb, const cab_store *cab,
     120             :              const bool compute_tau) {
     121             : 
     122   187270596 :   if (!compute_tau) {
     123   181560924 :     return get_virial_a_normal(a, b, i, j, zeta, cab);
     124             :   } else {
     125             :     double virial = 0.0;
     126    22838688 :     for (int k = 0; k < 3; k++) {
     127    17129016 :       virial += 0.5 * a.l[k] * b.l[k] *
     128    17129016 :                 get_virial_a_normal(down(k, a), down(k, b), i, j, zeta, cab);
     129    17129016 :       virial -= zeta * b.l[k] *
     130    17129016 :                 get_virial_a_normal(up(k, a), down(k, b), i, j, zeta, cab);
     131    17129016 :       virial -= a.l[k] * zetb *
     132    17129016 :                 get_virial_a_normal(down(k, a), up(k, b), i, j, zeta, cab);
     133    17129016 :       virial += 2.0 * zeta * zetb *
     134    17129016 :                 get_virial_a_normal(up(k, a), up(k, b), i, j, zeta, cab);
     135             :     }
     136             :     return virial;
     137             :   }
     138             : }
     139             : 
     140             : /*******************************************************************************
     141             :  * \brief Returns element i,j of virial on atom b for compute_tau=false.
     142             :  * \author Ole Schuett
     143             :  ******************************************************************************/
     144             : GRID_DEVICE static inline double
     145   250073748 : get_virial_b_normal(const orbital a, const orbital b, const int i, const int j,
     146             :                     const double zetb, const double rab[3],
     147             :                     const cab_store *cab) {
     148             : 
     149   250073748 :   return 2.0 * zetb *
     150   250073748 :              (cab_get(cab, up(i, up(j, a)), b) -
     151   250073748 :               cab_get(cab, up(i, a), b) * rab[j] -
     152   250073748 :               cab_get(cab, up(j, a), b) * rab[i] +
     153   250073748 :               cab_get(cab, a, b) * rab[j] * rab[i]) -
     154   250073748 :          b.l[j] * cab_get(cab, a, up(i, down(j, b)));
     155             : }
     156             : 
     157             : /*******************************************************************************
     158             :  * \brief Returns element i,j of virial on atom b.
     159             :  * \author Ole Schuett
     160             :  ******************************************************************************/
     161             : GRID_DEVICE static inline double
     162   187267356 : get_virial_b(const orbital a, const orbital b, const int i, const int j,
     163             :              const double zeta, const double zetb, const double rab[3],
     164             :              const cab_store *cab, const bool compute_tau) {
     165             : 
     166   187267356 :   if (!compute_tau) {
     167   181557684 :     return get_virial_b_normal(a, b, i, j, zetb, rab, cab);
     168             :   } else {
     169             :     double virial = 0.0;
     170    22838688 :     for (int k = 0; k < 3; k++) {
     171    17129016 :       virial +=
     172    17129016 :           0.5 * a.l[k] * b.l[k] *
     173    17129016 :           get_virial_b_normal(down(k, a), down(k, b), i, j, zetb, rab, cab);
     174    17129016 :       virial -= zeta * b.l[k] *
     175    17129016 :                 get_virial_b_normal(up(k, a), down(k, b), i, j, zetb, rab, cab);
     176    17129016 :       virial -= a.l[k] * zetb *
     177    17129016 :                 get_virial_b_normal(down(k, a), up(k, b), i, j, zetb, rab, cab);
     178    17129016 :       virial += 2.0 * zeta * zetb *
     179    17129016 :                 get_virial_b_normal(up(k, a), up(k, b), i, j, zetb, rab, cab);
     180             :     }
     181             :     return virial;
     182             :   }
     183             : }
     184             : 
     185             : /*******************************************************************************
     186             :  * \brief Returns element i,j of hab matrix.
     187             :  * \author Ole Schuett
     188             :  ******************************************************************************/
     189   883572001 : GRID_DEVICE static inline double get_hab(const orbital a, const orbital b,
     190             :                                          const double zeta, const double zetb,
     191             :                                          const cab_store *cab,
     192             :                                          const bool compute_tau) {
     193   883572001 :   if (!compute_tau) {
     194   873079064 :     return cab_get(cab, a, b);
     195             :   } else {
     196             :     double hab = 0.0;
     197    41971748 :     for (int k = 0; k < 3; k++) {
     198    31478811 :       hab += 0.5 * a.l[k] * b.l[k] * cab_get(cab, down(k, a), down(k, b));
     199    31478811 :       hab -= zeta * b.l[k] * cab_get(cab, up(k, a), down(k, b));
     200    31478811 :       hab -= a.l[k] * zetb * cab_get(cab, down(k, a), up(k, b));
     201    31478811 :       hab += 2.0 * zeta * zetb * cab_get(cab, up(k, a), up(k, b));
     202             :     }
     203             :     return hab;
     204             :   }
     205             : }
     206             : 
     207             : /*******************************************************************************
     208             :  * \brief Differences in angular momentum.
     209             :  * \author Ole Schuett
     210             :  ******************************************************************************/
     211             : typedef struct {
     212             :   int la_max_diff;
     213             :   int la_min_diff;
     214             :   int lb_max_diff;
     215             :   int lb_min_diff;
     216             : } process_ldiffs;
     217             : 
     218             : /*******************************************************************************
     219             :  * \brief Returns difference in angular momentum range for given flags.
     220             :  * \author Ole Schuett
     221             :  ******************************************************************************/
     222    85653760 : static process_ldiffs process_get_ldiffs(bool calculate_forces,
     223             :                                          bool calculate_virial,
     224             :                                          bool compute_tau) {
     225    85653760 :   process_ldiffs ldiffs;
     226             : 
     227    85653760 :   ldiffs.la_max_diff = 0;
     228    85653760 :   ldiffs.lb_max_diff = 0;
     229    85653760 :   ldiffs.la_min_diff = 0;
     230    85653760 :   ldiffs.lb_min_diff = 0;
     231             : 
     232    85653760 :   if (calculate_forces || calculate_virial) {
     233    17728729 :     ldiffs.la_max_diff += 1; // for deriv. of gaussian, unimportant which one
     234    17728729 :     ldiffs.la_min_diff -= 1;
     235    17728729 :     ldiffs.lb_min_diff -= 1;
     236             :   }
     237             : 
     238    85653760 :   if (calculate_virial) {
     239     2732325 :     ldiffs.la_max_diff += 1;
     240     2732325 :     ldiffs.lb_max_diff += 1;
     241             :   }
     242             : 
     243    85653760 :   if (compute_tau) {
     244     1286922 :     ldiffs.la_max_diff += 1;
     245     1286922 :     ldiffs.lb_max_diff += 1;
     246     1286922 :     ldiffs.la_min_diff -= 1;
     247     1286922 :     ldiffs.lb_min_diff -= 1;
     248             :   }
     249             : 
     250    85653760 :   return ldiffs;
     251             : }
     252             : 
     253             : // EOF

Generated by: LCOV version 1.15