This file is indexed.

/usr/include/chemps2/DMRGSCFintegrals.h is in libchemps2-dev 1.8.3-2.

This file is owned by root:root, with mode 0o644.

The actual contents of the file can be viewed below.

  1
  2
  3
  4
  5
  6
  7
  8
  9
 10
 11
 12
 13
 14
 15
 16
 17
 18
 19
 20
 21
 22
 23
 24
 25
 26
 27
 28
 29
 30
 31
 32
 33
 34
 35
 36
 37
 38
 39
 40
 41
 42
 43
 44
 45
 46
 47
 48
 49
 50
 51
 52
 53
 54
 55
 56
 57
 58
 59
 60
 61
 62
 63
 64
 65
 66
 67
 68
 69
 70
 71
 72
 73
 74
 75
 76
 77
 78
 79
 80
 81
 82
 83
 84
 85
 86
 87
 88
 89
 90
 91
 92
 93
 94
 95
 96
 97
 98
 99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
/*
   CheMPS2: a spin-adapted implementation of DMRG for ab initio quantum chemistry
   Copyright (C) 2013-2016 Sebastian Wouters

   This program is free software; you can redistribute it and/or modify
   it under the terms of the GNU General Public License as published by
   the Free Software Foundation; either version 2 of the License, or
   (at your option) any later version.

   This program is distributed in the hope that it will be useful,
   but WITHOUT ANY WARRANTY; without even the implied warranty of
   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
   GNU General Public License for more details.

   You should have received a copy of the GNU General Public License along
   with this program; if not, write to the Free Software Foundation, Inc.,
   51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
*/

#ifndef DMRGSCFINTEGRALS_CHEMPS2_H
#define DMRGSCFINTEGRALS_CHEMPS2_H

#include "DMRGSCFindices.h"

namespace CheMPS2{
/** DMRGSCFintegrals class.
    \author Sebastian Wouters <sebastianwouters@gmail.com>
    \date January 27, 2015
    
    Container class for rotated DMRGSCF integrals. At most two virtual indices are needed to compute the gradient and Hessian, which allows for computational as well as memory savings. The integrals are stored in two objects:
    \f{eqnarray*}{
       Coulomb & : & ( c_1 c_2 | a_1 a_2 ) \\
      Exchange & : & ( c_1 v_1 | c_2 v_2 )
    \f}
    where \f$c\f$ denote core (occupied + active) orbitals, \f$v\f$ virtual orbitals, and \f$a\f$ both core and virtual orbitals. For the Coulomb integrals the fourfold permutation symmetry \f$c_1 \leq c_2\f$ and \f$a_1 \leq a_2\f$ is taken into account and for the exchange integrals the twofold permutation symmetry \f$c_1 \leq c_2\f$.
    
    Consider for example a molecule in C1 symmetry with 100 core (occupied + active) and 300 virtual orbitals. The FourIndex class needs to store \f$\frac{1}{8}(N_{core} + N_{virt})^4\f$ doubles or 25.6 GB. This class needs to store \f$\frac{1}{4} N_{core}^2 (N_{core}+N_{virt})^2 + \frac{1}{2} N_{core}^2 N_{virt}^2\f$ doubles or 6.8 GB. The eightfold permutation symmetry of the \f$( c_1 c_2 | c_3 c_4 )\f$ part of the Coulomb object is not used in this class. For the example, this would however only result in a reduction of \f$\frac{1}{8} N_{core}^4\f$ doubles or 0.1 GB.
*/
   class DMRGSCFintegrals{

      public:
      
         //! Constructor
         /** \param iHandler The DMRGSCFindices which contain information on the core (occupied + active) and virtual spaces */
         DMRGSCFintegrals(DMRGSCFindices * iHandler);
         
         //! Destructor
         virtual ~DMRGSCFintegrals();
         
         //! Set the storage objects to zero
         void clear();
         
         //! Set an element of the Coulomb object ( c1 c2 | a1 a2 )
         /** \param Ic1 The irrep of the first core index
             \param Ic2 The irrep of the second core index
             \param Ia1 The irrep of the first core+virtual index
             \param Ia2 The irrep of the second core+virtual index
             \param c1 The first core index (within the irrep block)
             \param c2 The second core index (within the irrep block)
             \param a1 The first core+virtual index (within the irrep block)
             \param a2 The second core+virtual index (within the irrep block)
             \param val The value to which the element of the object should be set */
         void set_coulomb(const int Ic1, const int Ic2, const int Ia1, const int Ia2, const int c1, const int c2, const int a1, const int a2, const double val);

         //! Add a double to an element of the Coulomb object ( c1 c2 | a1 a2 )
         /** \param Ic1 The irrep of the first core index
             \param Ic2 The irrep of the second core index
             \param Ia1 The irrep of the first core+virtual index
             \param Ia2 The irrep of the second core+virtual index
             \param c1 The first core index (within the irrep block)
             \param c2 The second core index (within the irrep block)
             \param a1 The first core+virtual index (within the irrep block)
             \param a2 The second core+virtual index (within the irrep block)
             \param val The value with which the element should be augmented */
         void add_coulomb(const int Ic1, const int Ic2, const int Ia1, const int Ia2, const int c1, const int c2, const int a1, const int a2, const double val);

         //! Get an element of the Coulomb object ( c1 c2 | a1 a2 )
         /** \param Ic1 The irrep of the first core index
             \param Ic2 The irrep of the second core index
             \param Ia1 The irrep of the first core+virtual index
             \param Ia2 The irrep of the second core+virtual index
             \param c1 The first core index (within the irrep block)
             \param c2 The second core index (within the irrep block)
             \param a1 The first core+virtual index (within the irrep block)
             \param a2 The second core+virtual index (within the irrep block)
             \return The desired value of the Coulomb object */
         double get_coulomb(const int Ic1, const int Ic2, const int Ia1, const int Ia2, const int c1, const int c2, const int a1, const int a2) const;

         //! Set an element of the Exchange object ( c1 v1 | c2 v2 )
         /** \param Ic1 The irrep of the first core index
             \param Ic2 The irrep of the second core index
             \param Iv1 The irrep of the first virtual index
             \param Iv2 The irrep of the second virtual index
             \param c1 The first core index (within the irrep block)
             \param c2 The second core index (within the irrep block)
             \param v1 The first virtual index (within the irrep block; counting starts at NCORE[ Iv1 ])
             \param v2 The second virtual index (within the irrep block; counting starts at NCORE[ Iv2 ])
             \param val The value to which the element of the object should be set */
         void set_exchange(const int Ic1, const int Ic2, const int Iv1, const int Iv2, const int c1, const int c2, const int v1, const int v2, const double val);
         
         //! Add a double to an element of the Exchange object ( c1 v1 | c2 v2 )
         /** \param Ic1 The irrep of the first core index
             \param Ic2 The irrep of the second core index
             \param Iv1 The irrep of the first virtual index
             \param Iv2 The irrep of the second virtual index
             \param c1 The first core index (within the irrep block)
             \param c2 The second core index (within the irrep block)
             \param v1 The first virtual index (within the irrep block; counting starts at NCORE[ Iv1 ])
             \param v2 The second virtual index (within the irrep block; counting starts at NCORE[ Iv2 ])
             \param val The value with which the element should be augmented */
         void add_exchange(const int Ic1, const int Ic2, const int Iv1, const int Iv2, const int c1, const int c2, const int v1, const int v2, const double val);
         
         //! Get an element of the Exchange object ( c1 v1 | c2 v2 )
         /** \param Ic1 The irrep of the first core index
             \param Ic2 The irrep of the second core index
             \param Iv1 The irrep of the first virtual index
             \param Iv2 The irrep of the second virtual index
             \param c1 The first core index (within the irrep block)
             \param c2 The second core index (within the irrep block)
             \param v1 The first virtual index (within the irrep block; counting starts at NCORE[ Iv1 ])
             \param v2 The second virtual index (within the irrep block; counting starts at NCORE[ Iv2 ])
             \return The desired value of the Exchange object */
         double get_exchange(const int Ic1, const int Ic2, const int Iv1, const int Iv2, const int c1, const int c2, const int v1, const int v2) const;
         
         //! Get a two-body matrix element with at most 2 virtual indices, using the FourIndex API
         /** \param I1 The irrep of the first index
             \param I2 The irrep of the second index
             \param I3 The irrep of the third index
             \param I4 The irrep of the fourth index
             \param index1 The first index (within the irrep block)
             \param index2 The second index (within the irrep block)
             \param index3 The third index (within the irrep block)
             \param index4 The fourth index (within the irrep block)
             \return The desired two-body matrix element */
         double FourIndexAPI(const int I1, const int I2, const int I3, const int I4, const int index1, const int index2, const int index3, const int index4) const;
      
      private:
      
         // The number of irreps
         int numberOfIrreps;
         
         // The number of core orbitals per irrep
         int * NCORE;
         
         // The number of virtual orbitals per irrep
         int * NVIRTUAL;
         
         // The total number of orbitals per irrep
         int * NTOTAL;
         
         // Pointers to the Coulomb array:
         //    - coulomb_ptr[ Ic1 x Ic2 != 0 ][ I_c1 ][ I_a1 ][ c1 + size(c1) * c2 ] points to the relevant part of coulomb_array
         //    - coulomb_ptr[ Ic1 x Ic2 == 0 ][ I_c1 ][ I_a1 ][ c1 + (c2*(c2+1))/2 ] points to the relevant part of coulomb_array
         long long **** coulomb_ptr;
         long long coulomb_size;
         double * coulomb_array;
         long long calcNumCoulombElements(const bool allocate);
         long long get_coulomb_ptr( const int Ic1, const int Ic2, const int Ia1, const int Ia2, const int c1, const int c2, const int a1, const int a2 ) const;
         
         // Pointers to the Exchange array:
         //    - exchange_ptr[ Ic1 x Ic2 != 0 ][ I_c1 ][ I_v1 ][ c1 + size(c1) * c2 ] points to the relevant part of exchange_array
         //    - exchange_ptr[ Ic1 x Ic2 == 0 ][ I_c1 ][ I_v1 ][ c1 + (c2*(c2+1))/2 ] points to the relevant part of exchange_array
         long long **** exchange_ptr;
         long long exchange_size;
         double * exchange_array;
         long long calcNumExchangeElements(const bool allocate);
         long long get_exchange_ptr( const int Ic1, const int Ic2, const int Iv1, const int Iv2, const int c1, const int c2, const int v1, const int v2 ) const;

   };
}

#endif