Sherman-Morrison-Woodbury
Table of Contents
1 Headers
#include "qmckl.h" #include "assert.h" #ifdef HAVE_CONFIG_H #include "config.h" #endif #include <math.h> int main() { qmckl_context context; context = qmckl_context_create(); qmckl_exit_code rc;
2 Naïve Sherman-Morrison
2.1 qmckl_sherman_morrison_naive
This is the simplest of the available Sherman-Morrison-Woodbury kernels. It applies rank-1 updates one by one in the order that is given. It only checks if the denominator in the Sherman-Morrison formula is not too close to zero when an update is evaluated. It will exit with an error code of the denominator is too close to zero.
The formula for any update \(u_j\) (index \(j\) is suppresed for clarity) that is applied is \[ (S + uv^T)^{-1} = S^{-1} - \frac{S^{-1} uv^T S^{-1}}{1 + v^T S^{-1} u} \]
where \(S\) is the Slater-matrix, \(u\) and \(v^T\) are the column and row vectors containing the updates, \(S^{-1}\) is the inverse of the Slater-matrix.
Even though the Slater-matrix \(S\) with all updates applied at once is invertable, during the course of applying updates to the inverse Slater-matrix \(S^{-1}\) one-by-one it can happen that one of the intermediate inverse matrices \(S^{-1}\) becomes singular. Therefore a global threshold value \(\epsilon\) is defined that is used to evaluate each individual update \(u_j\) when it is applied.
This value sets the lower bound for which the denominator \(1+v_j^TS^{-1}u_j\) is considered to be too small and will most probably result in a singular matrix \(S\), or at least in an inverse of \(S\) of very poor numerical quality. Therefore, when \(1+v_j^TS^{-1}u_j \geq \epsilon\), the update is applied as usual and the kernel exits with return code \texttt{QMCKL_SUCCESS}. If \(1+v_j^TS^{-1}u_j \leq \epsilon\) the update is rejected and the kernel exits with return code \texttt{QMCKL_FAILURE}.
If the determinant of the Slater-matrix is passed, it will be updated to the determinant resulting from applying the updates to the original matrix.
qmcklcontext | context | in | Global state |
uint64t | LDS | in | Leading dimension of Slaterinv |
uint64t | Dim | in | Dimension of Slaterinv |
uint64t | Nupdates | in | Number of rank-1 updates to be applied to Slaterinv |
double | Updates[Nupdates*Dim] | in | Array containing the updates |
uint64t | Updatesindex[Nupdates] | in | Array containing the rank-1 updates |
double | breakdown | in | Break-down parameter on which to fail or not |
double | Slaterinv[LDS*Dim] | inout | Array containing the inverse of a Slater-matrix |
double* | determinant | inout | Determinant of the Slater-matrix |
2.1.1 Requirements
context
is notQMCKL_NULL_CONTEXT
LDS >= 2
Dim >= 2
N_updates >= 1
Updates
is allocated with \(N_updates \times Dim\) elementsUpdates_index
is allocated with \(N_updates\) elementsbreakdown
is a small number such that \(0 < breakdown << 1\)Slater_inv
is allocated with \(Dim \times Dim\) elements
2.1.2 C header
qmckl_exit_code qmckl_sherman_morrison_naive ( const context qmckl_context, const LDS uint64_t, const Dim uint64_t, const N_updates uint64_t, const Updates* double, const Updates_index* uint64_t, const breakdown double, Slater_inv* double, determinant* double* );
#include <stdbool.h> #include <math.h> #include "qmckl.h" #include "config.h" // Order important because // __GNUC__ also set in ICC, ICX and CLANG // __clang__ also set in ICX #if defined(__INTEL_COMPILER) #define IVDEP _Pragma("ivdep") #define ALIGNED _Pragma("vector aligned") #elif defined(__INTEL_LLVM_COMPILER) #define IVDEP _Pragma("ivdep") #define ALIGNED _Pragma("vector aligned") #elif defined(__clang__) #define IVDEP _Pragma("clang loop vectorize(enable)") #define ALIGNED #elif defined(__GNUC__) #define IVDEP _Pragma("GCC ivdep") #define ALIGNED #endif qmckl_exit_code qmckl_sherman_morrison_naive_hpc( const qmckl_context context, const uint64_t LDS, const uint64_t Dim, const uint64_t N_updates, const double* __restrict Updates, const uint64_t* __restrict Updates_index, const double breakdown, double* __restrict Slater_inv, double* __restrict determinant) { if (qmckl_context_check(context) == QMCKL_NULL_CONTEXT) { return qmckl_failwith( context, QMCKL_NULL_CONTEXT, "qmckl_sherman_morrison_naive_hpc", NULL); } double __attribute__((aligned(8))) C[Dim]; double __attribute__((aligned(8))) D[LDS]; uint64_t l = 0; // For each update while (l < N_updates) { // C = S^{-1} x u_l for (uint64_t i = 0; i < Dim; i++) { C[i] = 0.0f; IVDEP ALIGNED for (uint64_t j = 0; j < LDS; j++) { C[i] += Slater_inv[i * LDS + j] * Updates[l * LDS + j]; } } // Denominator: v_l^T * C const int cui = Updates_index[l] - 1; double den = 1.0f + C[cui]; if (fabs(den) < breakdown) { return QMCKL_FAILURE; } double iden = 1.0f / den; // Update det(A) if (determinant) *determinant *= den; // selecting column: v_l^T * S_inv IVDEP ALIGNED for (uint64_t j = 0; j < LDS; j++) { D[j] = Slater_inv[cui * LDS + j]; } // A^{-1} = A^{-1} - C x D / den for (uint64_t i = 0; i < Dim; i++) { IVDEP ALIGNED for (uint64_t j = 0; j < LDS; j++) { const double update = C[i] * D[j] * iden; Slater_inv[i * LDS + j] -= update; } } l += 1; } return QMCKL_SUCCESS; }
static inline qmckl_exit_code qmckl_sherman_morrison_naive_{Dim}( const qmckl_context context, const uint64_t N_updates, const double* __restrict Updates, const uint64_t* __restrict Updates_index, const double breakdown, double* __restrict Slater_inv, double* __restrict determinant) { if (qmckl_context_check(context) == QMCKL_NULL_CONTEXT) { return qmckl_failwith( context, QMCKL_NULL_CONTEXT, "qmckl_sherman_morrison_naive_{Dim}", NULL); } #define D{Dim}_P ((1+({Dim}-1)/SIMD_LENGTH)*SIMD_LENGTH) double __attribute__((aligned(8))) C[{Dim}]; double __attribute__((aligned(8))) D[D{Dim}_P]; uint64_t l = 0; // For each update while (l < N_updates) { // C = A^{-1} x U_l for (uint64_t i = 0; i < {Dim}; i++) { C[i] = 0; IVDEP ALIGNED for (uint64_t j = 0; j < D{Dim}_P; j++) { C[i] += Slater_inv[i * D{Dim}_P + j] * Updates[l * D{Dim}_P + j]; } } // Denominator const int cui = Updates_index[l] - 1; double den = 1.0f + C[cui]; if (fabs(den) < breakdown) { return QMCKL_FAILURE; } double iden = 1.0f / den; // Update det(A) if (determinant) *determinant *= den; // selecting column: D = v_l^T * S_inv IVDEP ALIGNED for (uint64_t j = 0; j < D{Dim}_P; j++) { D[j] = Slater_inv[cui * D{Dim}_P + j]; } // A^{-1} = A^{-1} - C x D / den for (uint64_t i = 0; i < {Dim}; i++) { IVDEP ALIGNED for (uint64_t j = 0; j < D{Dim}_P; j++) { double update = C[i] * D[j] * iden; Slater_inv[i * D{Dim}_P + j] -= update; } } l += 1; } return QMCKL_SUCCESS; }
static inline qmckl_exit_code qmckl_sherman_morrison_naive_2( const qmckl_context context, const uint64_t N_updates, const double* __restrict Updates, const uint64_t* __restrict Updates_index, const double breakdown, double* __restrict Slater_inv, double* __restrict determinant) { if (qmckl_context_check(context) == QMCKL_NULL_CONTEXT) { return qmckl_failwith( context, QMCKL_NULL_CONTEXT, "qmckl_sherman_morrison_naive_2", NULL); } #define D2_P ((1+(2-1)/SIMD_LENGTH)*SIMD_LENGTH) double __attribute__((aligned(8))) C[2]; double __attribute__((aligned(8))) D[D2_P]; uint64_t l = 0; // For each update while (l < N_updates) { // C = A^{-1} x U_l for (uint64_t i = 0; i < 2; i++) { C[i] = 0; IVDEP ALIGNED for (uint64_t j = 0; j < D2_P; j++) { C[i] += Slater_inv[i * D2_P + j] * Updates[l * D2_P + j]; } } // Denominator const int cui = Updates_index[l] - 1; double den = 1.0f + C[cui]; if (fabs(den) < breakdown) { return QMCKL_FAILURE; } double iden = 1.0f / den; // Update det(A) if (determinant) *determinant *= den; // selecting column: D = v_l^T * S_inv IVDEP ALIGNED for (uint64_t j = 0; j < D2_P; j++) { D[j] = Slater_inv[cui * D2_P + j]; } // A^{-1} = A^{-1} - C x D / den for (uint64_t i = 0; i < 2; i++) { IVDEP ALIGNED for (uint64_t j = 0; j < D2_P; j++) { double update = C[i] * D[j] * iden; Slater_inv[i * D2_P + j] -= update; } } l += 1; } return QMCKL_SUCCESS; } static inline qmckl_exit_code qmckl_sherman_morrison_naive_3( const qmckl_context context, const uint64_t N_updates, const double* __restrict Updates, const uint64_t* __restrict Updates_index, const double breakdown, double* __restrict Slater_inv, double* __restrict determinant) { if (qmckl_context_check(context) == QMCKL_NULL_CONTEXT) { return qmckl_failwith( context, QMCKL_NULL_CONTEXT, "qmckl_sherman_morrison_naive_3", NULL); } #define D3_P ((1+(3-1)/SIMD_LENGTH)*SIMD_LENGTH) double __attribute__((aligned(8))) C[3]; double __attribute__((aligned(8))) D[D3_P]; uint64_t l = 0; // For each update while (l < N_updates) { // C = A^{-1} x U_l for (uint64_t i = 0; i < 3; i++) { C[i] = 0; IVDEP ALIGNED for (uint64_t j = 0; j < D3_P; j++) { C[i] += Slater_inv[i * D3_P + j] * Updates[l * D3_P + j]; } } // Denominator const int cui = Updates_index[l] - 1; double den = 1.0f + C[cui]; if (fabs(den) < breakdown) { return QMCKL_FAILURE; } double iden = 1.0f / den; // Update det(A) if (determinant) *determinant *= den; // selecting column: D = v_l^T * S_inv IVDEP ALIGNED for (uint64_t j = 0; j < D3_P; j++) { D[j] = Slater_inv[cui * D3_P + j]; } // A^{-1} = A^{-1} - C x D / den for (uint64_t i = 0; i < 3; i++) { IVDEP ALIGNED for (uint64_t j = 0; j < D3_P; j++) { double update = C[i] * D[j] * iden; Slater_inv[i * D3_P + j] -= update; } } l += 1; } return QMCKL_SUCCESS; } static inline qmckl_exit_code qmckl_sherman_morrison_naive_4( const qmckl_context context, const uint64_t N_updates, const double* __restrict Updates, const uint64_t* __restrict Updates_index, const double breakdown, double* __restrict Slater_inv, double* __restrict determinant) { if (qmckl_context_check(context) == QMCKL_NULL_CONTEXT) { return qmckl_failwith( context, QMCKL_NULL_CONTEXT, "qmckl_sherman_morrison_naive_4", NULL); } #define D4_P ((1+(4-1)/SIMD_LENGTH)*SIMD_LENGTH) double __attribute__((aligned(8))) C[4]; double __attribute__((aligned(8))) D[D4_P]; uint64_t l = 0; // For each update while (l < N_updates) { // C = A^{-1} x U_l for (uint64_t i = 0; i < 4; i++) { C[i] = 0; IVDEP ALIGNED for (uint64_t j = 0; j < D4_P; j++) { C[i] += Slater_inv[i * D4_P + j] * Updates[l * D4_P + j]; } } // Denominator const int cui = Updates_index[l] - 1; double den = 1.0f + C[cui]; if (fabs(den) < breakdown) { return QMCKL_FAILURE; } double iden = 1.0f / den; // Update det(A) if (determinant) *determinant *= den; // selecting column: D = v_l^T * S_inv IVDEP ALIGNED for (uint64_t j = 0; j < D4_P; j++) { D[j] = Slater_inv[cui * D4_P + j]; } // A^{-1} = A^{-1} - C x D / den for (uint64_t i = 0; i < 4; i++) { IVDEP ALIGNED for (uint64_t j = 0; j < D4_P; j++) { double update = C[i] * D[j] * iden; Slater_inv[i * D4_P + j] -= update; } } l += 1; } return QMCKL_SUCCESS; } static inline qmckl_exit_code qmckl_sherman_morrison_naive_5( const qmckl_context context, const uint64_t N_updates, const double* __restrict Updates, const uint64_t* __restrict Updates_index, const double breakdown, double* __restrict Slater_inv, double* __restrict determinant) { if (qmckl_context_check(context) == QMCKL_NULL_CONTEXT) { return qmckl_failwith( context, QMCKL_NULL_CONTEXT, "qmckl_sherman_morrison_naive_5", NULL); } #define D5_P ((1+(5-1)/SIMD_LENGTH)*SIMD_LENGTH) double __attribute__((aligned(8))) C[5]; double __attribute__((aligned(8))) D[D5_P]; uint64_t l = 0; // For each update while (l < N_updates) { // C = A^{-1} x U_l for (uint64_t i = 0; i < 5; i++) { C[i] = 0; IVDEP ALIGNED for (uint64_t j = 0; j < D5_P; j++) { C[i] += Slater_inv[i * D5_P + j] * Updates[l * D5_P + j]; } } // Denominator const int cui = Updates_index[l] - 1; double den = 1.0f + C[cui]; if (fabs(den) < breakdown) { return QMCKL_FAILURE; } double iden = 1.0f / den; // Update det(A) if (determinant) *determinant *= den; // selecting column: D = v_l^T * S_inv IVDEP ALIGNED for (uint64_t j = 0; j < D5_P; j++) { D[j] = Slater_inv[cui * D5_P + j]; } // A^{-1} = A^{-1} - C x D / den for (uint64_t i = 0; i < 5; i++) { IVDEP ALIGNED for (uint64_t j = 0; j < D5_P; j++) { double update = C[i] * D[j] * iden; Slater_inv[i * D5_P + j] -= update; } } l += 1; } return QMCKL_SUCCESS; } static inline qmckl_exit_code qmckl_sherman_morrison_naive_6( const qmckl_context context, const uint64_t N_updates, const double* __restrict Updates, const uint64_t* __restrict Updates_index, const double breakdown, double* __restrict Slater_inv, double* __restrict determinant) { if (qmckl_context_check(context) == QMCKL_NULL_CONTEXT) { return qmckl_failwith( context, QMCKL_NULL_CONTEXT, "qmckl_sherman_morrison_naive_6", NULL); } #define D6_P ((1+(6-1)/SIMD_LENGTH)*SIMD_LENGTH) double __attribute__((aligned(8))) C[6]; double __attribute__((aligned(8))) D[D6_P]; uint64_t l = 0; // For each update while (l < N_updates) { // C = A^{-1} x U_l for (uint64_t i = 0; i < 6; i++) { C[i] = 0; IVDEP ALIGNED for (uint64_t j = 0; j < D6_P; j++) { C[i] += Slater_inv[i * D6_P + j] * Updates[l * D6_P + j]; } } // Denominator const int cui = Updates_index[l] - 1; double den = 1.0f + C[cui]; if (fabs(den) < breakdown) { return QMCKL_FAILURE; } double iden = 1.0f / den; // Update det(A) if (determinant) *determinant *= den; // selecting column: D = v_l^T * S_inv IVDEP ALIGNED for (uint64_t j = 0; j < D6_P; j++) { D[j] = Slater_inv[cui * D6_P + j]; } // A^{-1} = A^{-1} - C x D / den for (uint64_t i = 0; i < 6; i++) { IVDEP ALIGNED for (uint64_t j = 0; j < D6_P; j++) { double update = C[i] * D[j] * iden; Slater_inv[i * D6_P + j] -= update; } } l += 1; } return QMCKL_SUCCESS; } static inline qmckl_exit_code qmckl_sherman_morrison_naive_7( const qmckl_context context, const uint64_t N_updates, const double* __restrict Updates, const uint64_t* __restrict Updates_index, const double breakdown, double* __restrict Slater_inv, double* __restrict determinant) { if (qmckl_context_check(context) == QMCKL_NULL_CONTEXT) { return qmckl_failwith( context, QMCKL_NULL_CONTEXT, "qmckl_sherman_morrison_naive_7", NULL); } #define D7_P ((1+(7-1)/SIMD_LENGTH)*SIMD_LENGTH) double __attribute__((aligned(8))) C[7]; double __attribute__((aligned(8))) D[D7_P]; uint64_t l = 0; // For each update while (l < N_updates) { // C = A^{-1} x U_l for (uint64_t i = 0; i < 7; i++) { C[i] = 0; IVDEP ALIGNED for (uint64_t j = 0; j < D7_P; j++) { C[i] += Slater_inv[i * D7_P + j] * Updates[l * D7_P + j]; } } // Denominator const int cui = Updates_index[l] - 1; double den = 1.0f + C[cui]; if (fabs(den) < breakdown) { return QMCKL_FAILURE; } double iden = 1.0f / den; // Update det(A) if (determinant) *determinant *= den; // selecting column: D = v_l^T * S_inv IVDEP ALIGNED for (uint64_t j = 0; j < D7_P; j++) { D[j] = Slater_inv[cui * D7_P + j]; } // A^{-1} = A^{-1} - C x D / den for (uint64_t i = 0; i < 7; i++) { IVDEP ALIGNED for (uint64_t j = 0; j < D7_P; j++) { double update = C[i] * D[j] * iden; Slater_inv[i * D7_P + j] -= update; } } l += 1; } return QMCKL_SUCCESS; } static inline qmckl_exit_code qmckl_sherman_morrison_naive_8( const qmckl_context context, const uint64_t N_updates, const double* __restrict Updates, const uint64_t* __restrict Updates_index, const double breakdown, double* __restrict Slater_inv, double* __restrict determinant) { if (qmckl_context_check(context) == QMCKL_NULL_CONTEXT) { return qmckl_failwith( context, QMCKL_NULL_CONTEXT, "qmckl_sherman_morrison_naive_8", NULL); } #define D8_P ((1+(8-1)/SIMD_LENGTH)*SIMD_LENGTH) double __attribute__((aligned(8))) C[8]; double __attribute__((aligned(8))) D[D8_P]; uint64_t l = 0; // For each update while (l < N_updates) { // C = A^{-1} x U_l for (uint64_t i = 0; i < 8; i++) { C[i] = 0; IVDEP ALIGNED for (uint64_t j = 0; j < D8_P; j++) { C[i] += Slater_inv[i * D8_P + j] * Updates[l * D8_P + j]; } } // Denominator const int cui = Updates_index[l] - 1; double den = 1.0f + C[cui]; if (fabs(den) < breakdown) { return QMCKL_FAILURE; } double iden = 1.0f / den; // Update det(A) if (determinant) *determinant *= den; // selecting column: D = v_l^T * S_inv IVDEP ALIGNED for (uint64_t j = 0; j < D8_P; j++) { D[j] = Slater_inv[cui * D8_P + j]; } // A^{-1} = A^{-1} - C x D / den for (uint64_t i = 0; i < 8; i++) { IVDEP ALIGNED for (uint64_t j = 0; j < D8_P; j++) { double update = C[i] * D[j] * iden; Slater_inv[i * D8_P + j] -= update; } } l += 1; } return QMCKL_SUCCESS; } static inline qmckl_exit_code qmckl_sherman_morrison_naive_9( const qmckl_context context, const uint64_t N_updates, const double* __restrict Updates, const uint64_t* __restrict Updates_index, const double breakdown, double* __restrict Slater_inv, double* __restrict determinant) { if (qmckl_context_check(context) == QMCKL_NULL_CONTEXT) { return qmckl_failwith( context, QMCKL_NULL_CONTEXT, "qmckl_sherman_morrison_naive_9", NULL); } #define D9_P ((1+(9-1)/SIMD_LENGTH)*SIMD_LENGTH) double __attribute__((aligned(8))) C[9]; double __attribute__((aligned(8))) D[D9_P]; uint64_t l = 0; // For each update while (l < N_updates) { // C = A^{-1} x U_l for (uint64_t i = 0; i < 9; i++) { C[i] = 0; IVDEP ALIGNED for (uint64_t j = 0; j < D9_P; j++) { C[i] += Slater_inv[i * D9_P + j] * Updates[l * D9_P + j]; } } // Denominator const int cui = Updates_index[l] - 1; double den = 1.0f + C[cui]; if (fabs(den) < breakdown) { return QMCKL_FAILURE; } double iden = 1.0f / den; // Update det(A) if (determinant) *determinant *= den; // selecting column: D = v_l^T * S_inv IVDEP ALIGNED for (uint64_t j = 0; j < D9_P; j++) { D[j] = Slater_inv[cui * D9_P + j]; } // A^{-1} = A^{-1} - C x D / den for (uint64_t i = 0; i < 9; i++) { IVDEP ALIGNED for (uint64_t j = 0; j < D9_P; j++) { double update = C[i] * D[j] * iden; Slater_inv[i * D9_P + j] -= update; } } l += 1; } return QMCKL_SUCCESS; } static inline qmckl_exit_code qmckl_sherman_morrison_naive_10( const qmckl_context context, const uint64_t N_updates, const double* __restrict Updates, const uint64_t* __restrict Updates_index, const double breakdown, double* __restrict Slater_inv, double* __restrict determinant) { if (qmckl_context_check(context) == QMCKL_NULL_CONTEXT) { return qmckl_failwith( context, QMCKL_NULL_CONTEXT, "qmckl_sherman_morrison_naive_10", NULL); } #define D10_P ((1+(10-1)/SIMD_LENGTH)*SIMD_LENGTH) double __attribute__((aligned(8))) C[10]; double __attribute__((aligned(8))) D[D10_P]; uint64_t l = 0; // For each update while (l < N_updates) { // C = A^{-1} x U_l for (uint64_t i = 0; i < 10; i++) { C[i] = 0; IVDEP ALIGNED for (uint64_t j = 0; j < D10_P; j++) { C[i] += Slater_inv[i * D10_P + j] * Updates[l * D10_P + j]; } } // Denominator const int cui = Updates_index[l] - 1; double den = 1.0f + C[cui]; if (fabs(den) < breakdown) { return QMCKL_FAILURE; } double iden = 1.0f / den; // Update det(A) if (determinant) *determinant *= den; // selecting column: D = v_l^T * S_inv IVDEP ALIGNED for (uint64_t j = 0; j < D10_P; j++) { D[j] = Slater_inv[cui * D10_P + j]; } // A^{-1} = A^{-1} - C x D / den for (uint64_t i = 0; i < 10; i++) { IVDEP ALIGNED for (uint64_t j = 0; j < D10_P; j++) { double update = C[i] * D[j] * iden; Slater_inv[i * D10_P + j] -= update; } } l += 1; } return QMCKL_SUCCESS; } static inline qmckl_exit_code qmckl_sherman_morrison_naive_11( const qmckl_context context, const uint64_t N_updates, const double* __restrict Updates, const uint64_t* __restrict Updates_index, const double breakdown, double* __restrict Slater_inv, double* __restrict determinant) { if (qmckl_context_check(context) == QMCKL_NULL_CONTEXT) { return qmckl_failwith( context, QMCKL_NULL_CONTEXT, "qmckl_sherman_morrison_naive_11", NULL); } #define D11_P ((1+(11-1)/SIMD_LENGTH)*SIMD_LENGTH) double __attribute__((aligned(8))) C[11]; double __attribute__((aligned(8))) D[D11_P]; uint64_t l = 0; // For each update while (l < N_updates) { // C = A^{-1} x U_l for (uint64_t i = 0; i < 11; i++) { C[i] = 0; IVDEP ALIGNED for (uint64_t j = 0; j < D11_P; j++) { C[i] += Slater_inv[i * D11_P + j] * Updates[l * D11_P + j]; } } // Denominator const int cui = Updates_index[l] - 1; double den = 1.0f + C[cui]; if (fabs(den) < breakdown) { return QMCKL_FAILURE; } double iden = 1.0f / den; // Update det(A) if (determinant) *determinant *= den; // selecting column: D = v_l^T * S_inv IVDEP ALIGNED for (uint64_t j = 0; j < D11_P; j++) { D[j] = Slater_inv[cui * D11_P + j]; } // A^{-1} = A^{-1} - C x D / den for (uint64_t i = 0; i < 11; i++) { IVDEP ALIGNED for (uint64_t j = 0; j < D11_P; j++) { double update = C[i] * D[j] * iden; Slater_inv[i * D11_P + j] -= update; } } l += 1; } return QMCKL_SUCCESS; } static inline qmckl_exit_code qmckl_sherman_morrison_naive_12( const qmckl_context context, const uint64_t N_updates, const double* __restrict Updates, const uint64_t* __restrict Updates_index, const double breakdown, double* __restrict Slater_inv, double* __restrict determinant) { if (qmckl_context_check(context) == QMCKL_NULL_CONTEXT) { return qmckl_failwith( context, QMCKL_NULL_CONTEXT, "qmckl_sherman_morrison_naive_12", NULL); } #define D12_P ((1+(12-1)/SIMD_LENGTH)*SIMD_LENGTH) double __attribute__((aligned(8))) C[12]; double __attribute__((aligned(8))) D[D12_P]; uint64_t l = 0; // For each update while (l < N_updates) { // C = A^{-1} x U_l for (uint64_t i = 0; i < 12; i++) { C[i] = 0; IVDEP ALIGNED for (uint64_t j = 0; j < D12_P; j++) { C[i] += Slater_inv[i * D12_P + j] * Updates[l * D12_P + j]; } } // Denominator const int cui = Updates_index[l] - 1; double den = 1.0f + C[cui]; if (fabs(den) < breakdown) { return QMCKL_FAILURE; } double iden = 1.0f / den; // Update det(A) if (determinant) *determinant *= den; // selecting column: D = v_l^T * S_inv IVDEP ALIGNED for (uint64_t j = 0; j < D12_P; j++) { D[j] = Slater_inv[cui * D12_P + j]; } // A^{-1} = A^{-1} - C x D / den for (uint64_t i = 0; i < 12; i++) { IVDEP ALIGNED for (uint64_t j = 0; j < D12_P; j++) { double update = C[i] * D[j] * iden; Slater_inv[i * D12_P + j] -= update; } } l += 1; } return QMCKL_SUCCESS; } static inline qmckl_exit_code qmckl_sherman_morrison_naive_13( const qmckl_context context, const uint64_t N_updates, const double* __restrict Updates, const uint64_t* __restrict Updates_index, const double breakdown, double* __restrict Slater_inv, double* __restrict determinant) { if (qmckl_context_check(context) == QMCKL_NULL_CONTEXT) { return qmckl_failwith( context, QMCKL_NULL_CONTEXT, "qmckl_sherman_morrison_naive_13", NULL); } #define D13_P ((1+(13-1)/SIMD_LENGTH)*SIMD_LENGTH) double __attribute__((aligned(8))) C[13]; double __attribute__((aligned(8))) D[D13_P]; uint64_t l = 0; // For each update while (l < N_updates) { // C = A^{-1} x U_l for (uint64_t i = 0; i < 13; i++) { C[i] = 0; IVDEP ALIGNED for (uint64_t j = 0; j < D13_P; j++) { C[i] += Slater_inv[i * D13_P + j] * Updates[l * D13_P + j]; } } // Denominator const int cui = Updates_index[l] - 1; double den = 1.0f + C[cui]; if (fabs(den) < breakdown) { return QMCKL_FAILURE; } double iden = 1.0f / den; // Update det(A) if (determinant) *determinant *= den; // selecting column: D = v_l^T * S_inv IVDEP ALIGNED for (uint64_t j = 0; j < D13_P; j++) { D[j] = Slater_inv[cui * D13_P + j]; } // A^{-1} = A^{-1} - C x D / den for (uint64_t i = 0; i < 13; i++) { IVDEP ALIGNED for (uint64_t j = 0; j < D13_P; j++) { double update = C[i] * D[j] * iden; Slater_inv[i * D13_P + j] -= update; } } l += 1; } return QMCKL_SUCCESS; } static inline qmckl_exit_code qmckl_sherman_morrison_naive_14( const qmckl_context context, const uint64_t N_updates, const double* __restrict Updates, const uint64_t* __restrict Updates_index, const double breakdown, double* __restrict Slater_inv, double* __restrict determinant) { if (qmckl_context_check(context) == QMCKL_NULL_CONTEXT) { return qmckl_failwith( context, QMCKL_NULL_CONTEXT, "qmckl_sherman_morrison_naive_14", NULL); } #define D14_P ((1+(14-1)/SIMD_LENGTH)*SIMD_LENGTH) double __attribute__((aligned(8))) C[14]; double __attribute__((aligned(8))) D[D14_P]; uint64_t l = 0; // For each update while (l < N_updates) { // C = A^{-1} x U_l for (uint64_t i = 0; i < 14; i++) { C[i] = 0; IVDEP ALIGNED for (uint64_t j = 0; j < D14_P; j++) { C[i] += Slater_inv[i * D14_P + j] * Updates[l * D14_P + j]; } } // Denominator const int cui = Updates_index[l] - 1; double den = 1.0f + C[cui]; if (fabs(den) < breakdown) { return QMCKL_FAILURE; } double iden = 1.0f / den; // Update det(A) if (determinant) *determinant *= den; // selecting column: D = v_l^T * S_inv IVDEP ALIGNED for (uint64_t j = 0; j < D14_P; j++) { D[j] = Slater_inv[cui * D14_P + j]; } // A^{-1} = A^{-1} - C x D / den for (uint64_t i = 0; i < 14; i++) { IVDEP ALIGNED for (uint64_t j = 0; j < D14_P; j++) { double update = C[i] * D[j] * iden; Slater_inv[i * D14_P + j] -= update; } } l += 1; } return QMCKL_SUCCESS; } static inline qmckl_exit_code qmckl_sherman_morrison_naive_15( const qmckl_context context, const uint64_t N_updates, const double* __restrict Updates, const uint64_t* __restrict Updates_index, const double breakdown, double* __restrict Slater_inv, double* __restrict determinant) { if (qmckl_context_check(context) == QMCKL_NULL_CONTEXT) { return qmckl_failwith( context, QMCKL_NULL_CONTEXT, "qmckl_sherman_morrison_naive_15", NULL); } #define D15_P ((1+(15-1)/SIMD_LENGTH)*SIMD_LENGTH) double __attribute__((aligned(8))) C[15]; double __attribute__((aligned(8))) D[D15_P]; uint64_t l = 0; // For each update while (l < N_updates) { // C = A^{-1} x U_l for (uint64_t i = 0; i < 15; i++) { C[i] = 0; IVDEP ALIGNED for (uint64_t j = 0; j < D15_P; j++) { C[i] += Slater_inv[i * D15_P + j] * Updates[l * D15_P + j]; } } // Denominator const int cui = Updates_index[l] - 1; double den = 1.0f + C[cui]; if (fabs(den) < breakdown) { return QMCKL_FAILURE; } double iden = 1.0f / den; // Update det(A) if (determinant) *determinant *= den; // selecting column: D = v_l^T * S_inv IVDEP ALIGNED for (uint64_t j = 0; j < D15_P; j++) { D[j] = Slater_inv[cui * D15_P + j]; } // A^{-1} = A^{-1} - C x D / den for (uint64_t i = 0; i < 15; i++) { IVDEP ALIGNED for (uint64_t j = 0; j < D15_P; j++) { double update = C[i] * D[j] * iden; Slater_inv[i * D15_P + j] -= update; } } l += 1; } return QMCKL_SUCCESS; } static inline qmckl_exit_code qmckl_sherman_morrison_naive_16( const qmckl_context context, const uint64_t N_updates, const double* __restrict Updates, const uint64_t* __restrict Updates_index, const double breakdown, double* __restrict Slater_inv, double* __restrict determinant) { if (qmckl_context_check(context) == QMCKL_NULL_CONTEXT) { return qmckl_failwith( context, QMCKL_NULL_CONTEXT, "qmckl_sherman_morrison_naive_16", NULL); } #define D16_P ((1+(16-1)/SIMD_LENGTH)*SIMD_LENGTH) double __attribute__((aligned(8))) C[16]; double __attribute__((aligned(8))) D[D16_P]; uint64_t l = 0; // For each update while (l < N_updates) { // C = A^{-1} x U_l for (uint64_t i = 0; i < 16; i++) { C[i] = 0; IVDEP ALIGNED for (uint64_t j = 0; j < D16_P; j++) { C[i] += Slater_inv[i * D16_P + j] * Updates[l * D16_P + j]; } } // Denominator const int cui = Updates_index[l] - 1; double den = 1.0f + C[cui]; if (fabs(den) < breakdown) { return QMCKL_FAILURE; } double iden = 1.0f / den; // Update det(A) if (determinant) *determinant *= den; // selecting column: D = v_l^T * S_inv IVDEP ALIGNED for (uint64_t j = 0; j < D16_P; j++) { D[j] = Slater_inv[cui * D16_P + j]; } // A^{-1} = A^{-1} - C x D / den for (uint64_t i = 0; i < 16; i++) { IVDEP ALIGNED for (uint64_t j = 0; j < D16_P; j++) { double update = C[i] * D[j] * iden; Slater_inv[i * D16_P + j] -= update; } } l += 1; } return QMCKL_SUCCESS; } static inline qmckl_exit_code qmckl_sherman_morrison_naive_17( const qmckl_context context, const uint64_t N_updates, const double* __restrict Updates, const uint64_t* __restrict Updates_index, const double breakdown, double* __restrict Slater_inv, double* __restrict determinant) { if (qmckl_context_check(context) == QMCKL_NULL_CONTEXT) { return qmckl_failwith( context, QMCKL_NULL_CONTEXT, "qmckl_sherman_morrison_naive_17", NULL); } #define D17_P ((1+(17-1)/SIMD_LENGTH)*SIMD_LENGTH) double __attribute__((aligned(8))) C[17]; double __attribute__((aligned(8))) D[D17_P]; uint64_t l = 0; // For each update while (l < N_updates) { // C = A^{-1} x U_l for (uint64_t i = 0; i < 17; i++) { C[i] = 0; IVDEP ALIGNED for (uint64_t j = 0; j < D17_P; j++) { C[i] += Slater_inv[i * D17_P + j] * Updates[l * D17_P + j]; } } // Denominator const int cui = Updates_index[l] - 1; double den = 1.0f + C[cui]; if (fabs(den) < breakdown) { return QMCKL_FAILURE; } double iden = 1.0f / den; // Update det(A) if (determinant) *determinant *= den; // selecting column: D = v_l^T * S_inv IVDEP ALIGNED for (uint64_t j = 0; j < D17_P; j++) { D[j] = Slater_inv[cui * D17_P + j]; } // A^{-1} = A^{-1} - C x D / den for (uint64_t i = 0; i < 17; i++) { IVDEP ALIGNED for (uint64_t j = 0; j < D17_P; j++) { double update = C[i] * D[j] * iden; Slater_inv[i * D17_P + j] -= update; } } l += 1; } return QMCKL_SUCCESS; } static inline qmckl_exit_code qmckl_sherman_morrison_naive_18( const qmckl_context context, const uint64_t N_updates, const double* __restrict Updates, const uint64_t* __restrict Updates_index, const double breakdown, double* __restrict Slater_inv, double* __restrict determinant) { if (qmckl_context_check(context) == QMCKL_NULL_CONTEXT) { return qmckl_failwith( context, QMCKL_NULL_CONTEXT, "qmckl_sherman_morrison_naive_18", NULL); } #define D18_P ((1+(18-1)/SIMD_LENGTH)*SIMD_LENGTH) double __attribute__((aligned(8))) C[18]; double __attribute__((aligned(8))) D[D18_P]; uint64_t l = 0; // For each update while (l < N_updates) { // C = A^{-1} x U_l for (uint64_t i = 0; i < 18; i++) { C[i] = 0; IVDEP ALIGNED for (uint64_t j = 0; j < D18_P; j++) { C[i] += Slater_inv[i * D18_P + j] * Updates[l * D18_P + j]; } } // Denominator const int cui = Updates_index[l] - 1; double den = 1.0f + C[cui]; if (fabs(den) < breakdown) { return QMCKL_FAILURE; } double iden = 1.0f / den; // Update det(A) if (determinant) *determinant *= den; // selecting column: D = v_l^T * S_inv IVDEP ALIGNED for (uint64_t j = 0; j < D18_P; j++) { D[j] = Slater_inv[cui * D18_P + j]; } // A^{-1} = A^{-1} - C x D / den for (uint64_t i = 0; i < 18; i++) { IVDEP ALIGNED for (uint64_t j = 0; j < D18_P; j++) { double update = C[i] * D[j] * iden; Slater_inv[i * D18_P + j] -= update; } } l += 1; } return QMCKL_SUCCESS; } static inline qmckl_exit_code qmckl_sherman_morrison_naive_19( const qmckl_context context, const uint64_t N_updates, const double* __restrict Updates, const uint64_t* __restrict Updates_index, const double breakdown, double* __restrict Slater_inv, double* __restrict determinant) { if (qmckl_context_check(context) == QMCKL_NULL_CONTEXT) { return qmckl_failwith( context, QMCKL_NULL_CONTEXT, "qmckl_sherman_morrison_naive_19", NULL); } #define D19_P ((1+(19-1)/SIMD_LENGTH)*SIMD_LENGTH) double __attribute__((aligned(8))) C[19]; double __attribute__((aligned(8))) D[D19_P]; uint64_t l = 0; // For each update while (l < N_updates) { // C = A^{-1} x U_l for (uint64_t i = 0; i < 19; i++) { C[i] = 0; IVDEP ALIGNED for (uint64_t j = 0; j < D19_P; j++) { C[i] += Slater_inv[i * D19_P + j] * Updates[l * D19_P + j]; } } // Denominator const int cui = Updates_index[l] - 1; double den = 1.0f + C[cui]; if (fabs(den) < breakdown) { return QMCKL_FAILURE; } double iden = 1.0f / den; // Update det(A) if (determinant) *determinant *= den; // selecting column: D = v_l^T * S_inv IVDEP ALIGNED for (uint64_t j = 0; j < D19_P; j++) { D[j] = Slater_inv[cui * D19_P + j]; } // A^{-1} = A^{-1} - C x D / den for (uint64_t i = 0; i < 19; i++) { IVDEP ALIGNED for (uint64_t j = 0; j < D19_P; j++) { double update = C[i] * D[j] * iden; Slater_inv[i * D19_P + j] -= update; } } l += 1; } return QMCKL_SUCCESS; } static inline qmckl_exit_code qmckl_sherman_morrison_naive_20( const qmckl_context context, const uint64_t N_updates, const double* __restrict Updates, const uint64_t* __restrict Updates_index, const double breakdown, double* __restrict Slater_inv, double* __restrict determinant) { if (qmckl_context_check(context) == QMCKL_NULL_CONTEXT) { return qmckl_failwith( context, QMCKL_NULL_CONTEXT, "qmckl_sherman_morrison_naive_20", NULL); } #define D20_P ((1+(20-1)/SIMD_LENGTH)*SIMD_LENGTH) double __attribute__((aligned(8))) C[20]; double __attribute__((aligned(8))) D[D20_P]; uint64_t l = 0; // For each update while (l < N_updates) { // C = A^{-1} x U_l for (uint64_t i = 0; i < 20; i++) { C[i] = 0; IVDEP ALIGNED for (uint64_t j = 0; j < D20_P; j++) { C[i] += Slater_inv[i * D20_P + j] * Updates[l * D20_P + j]; } } // Denominator const int cui = Updates_index[l] - 1; double den = 1.0f + C[cui]; if (fabs(den) < breakdown) { return QMCKL_FAILURE; } double iden = 1.0f / den; // Update det(A) if (determinant) *determinant *= den; // selecting column: D = v_l^T * S_inv IVDEP ALIGNED for (uint64_t j = 0; j < D20_P; j++) { D[j] = Slater_inv[cui * D20_P + j]; } // A^{-1} = A^{-1} - C x D / den for (uint64_t i = 0; i < 20; i++) { IVDEP ALIGNED for (uint64_t j = 0; j < D20_P; j++) { double update = C[i] * D[j] * iden; Slater_inv[i * D20_P + j] -= update; } } l += 1; } return QMCKL_SUCCESS; } static inline qmckl_exit_code qmckl_sherman_morrison_naive_21( const qmckl_context context, const uint64_t N_updates, const double* __restrict Updates, const uint64_t* __restrict Updates_index, const double breakdown, double* __restrict Slater_inv, double* __restrict determinant) { if (qmckl_context_check(context) == QMCKL_NULL_CONTEXT) { return qmckl_failwith( context, QMCKL_NULL_CONTEXT, "qmckl_sherman_morrison_naive_21", NULL); } #define D21_P ((1+(21-1)/SIMD_LENGTH)*SIMD_LENGTH) double __attribute__((aligned(8))) C[21]; double __attribute__((aligned(8))) D[D21_P]; uint64_t l = 0; // For each update while (l < N_updates) { // C = A^{-1} x U_l for (uint64_t i = 0; i < 21; i++) { C[i] = 0; IVDEP ALIGNED for (uint64_t j = 0; j < D21_P; j++) { C[i] += Slater_inv[i * D21_P + j] * Updates[l * D21_P + j]; } } // Denominator const int cui = Updates_index[l] - 1; double den = 1.0f + C[cui]; if (fabs(den) < breakdown) { return QMCKL_FAILURE; } double iden = 1.0f / den; // Update det(A) if (determinant) *determinant *= den; // selecting column: D = v_l^T * S_inv IVDEP ALIGNED for (uint64_t j = 0; j < D21_P; j++) { D[j] = Slater_inv[cui * D21_P + j]; } // A^{-1} = A^{-1} - C x D / den for (uint64_t i = 0; i < 21; i++) { IVDEP ALIGNED for (uint64_t j = 0; j < D21_P; j++) { double update = C[i] * D[j] * iden; Slater_inv[i * D21_P + j] -= update; } } l += 1; } return QMCKL_SUCCESS; } qmckl_exit_code qmckl_sherman_morrison_naive(const qmckl_context context, const uint64_t LDS, const uint64_t Dim, const uint64_t N_updates, const double* Updates, const uint64_t* Updates_index, const double breakdown, double* Slater_inv, double* determinant) { if (qmckl_context_check(context) == QMCKL_NULL_CONTEXT) { return qmckl_failwith(context, QMCKL_NULL_CONTEXT, "qmckl_sherman_morrison_naive", NULL); } if (LDS == (1+(Dim-1)/SIMD_LENGTH)*SIMD_LENGTH) { // Most cases switch (Dim) { case 2: return qmckl_sherman_morrison_naive_2(context, N_updates, Updates, Updates_index, breakdown, Slater_inv, determinant); case 3: return qmckl_sherman_morrison_naive_3(context, N_updates, Updates, Updates_index, breakdown, Slater_inv, determinant); case 4: return qmckl_sherman_morrison_naive_4(context, N_updates, Updates, Updates_index, breakdown, Slater_inv, determinant); case 5: return qmckl_sherman_morrison_naive_5(context, N_updates, Updates, Updates_index, breakdown, Slater_inv, determinant); case 6: return qmckl_sherman_morrison_naive_6(context, N_updates, Updates, Updates_index, breakdown, Slater_inv, determinant); case 7: return qmckl_sherman_morrison_naive_7(context, N_updates, Updates, Updates_index, breakdown, Slater_inv, determinant); case 8: return qmckl_sherman_morrison_naive_8(context, N_updates, Updates, Updates_index, breakdown, Slater_inv, determinant); case 9: return qmckl_sherman_morrison_naive_9(context, N_updates, Updates, Updates_index, breakdown, Slater_inv, determinant); case 10: return qmckl_sherman_morrison_naive_10(context, N_updates, Updates, Updates_index, breakdown, Slater_inv, determinant); case 11: return qmckl_sherman_morrison_naive_11(context, N_updates, Updates, Updates_index, breakdown, Slater_inv, determinant); case 12: return qmckl_sherman_morrison_naive_12(context, N_updates, Updates, Updates_index, breakdown, Slater_inv, determinant); case 13: return qmckl_sherman_morrison_naive_13(context, N_updates, Updates, Updates_index, breakdown, Slater_inv, determinant); case 14: return qmckl_sherman_morrison_naive_14(context, N_updates, Updates, Updates_index, breakdown, Slater_inv, determinant); case 15: return qmckl_sherman_morrison_naive_15(context, N_updates, Updates, Updates_index, breakdown, Slater_inv, determinant); case 16: return qmckl_sherman_morrison_naive_16(context, N_updates, Updates, Updates_index, breakdown, Slater_inv, determinant); case 17: return qmckl_sherman_morrison_naive_17(context, N_updates, Updates, Updates_index, breakdown, Slater_inv, determinant); case 18: return qmckl_sherman_morrison_naive_18(context, N_updates, Updates, Updates_index, breakdown, Slater_inv, determinant); case 19: return qmckl_sherman_morrison_naive_19(context, N_updates, Updates, Updates_index, breakdown, Slater_inv, determinant); case 20: return qmckl_sherman_morrison_naive_20(context, N_updates, Updates, Updates_index, breakdown, Slater_inv, determinant); case 21: return qmckl_sherman_morrison_naive_21(context, N_updates, Updates, Updates_index, breakdown, Slater_inv, determinant); } } else { // When SIMD_LENGTH > 1, called with LDS == Dim AND Dim != (1+(Dim-1)/SIMD_LENGTH)*SIMD_LENGTH) return qmckl_sherman_morrison_naive_hpc(context, LDS, Dim, N_updates, Updates, Updates_index, breakdown, Slater_inv, determinant); } return QMCKL_FAILURE; }
2.1.3 Performance
This function performs best when there is only 1 rank-1 update in the update cycle. It is not useful to use Sherman-Morrison with update splitting for these cycles since splitting can never resolve a situation where applying the update causes singular behaviour.
3 Woodbury 2x2
3.1 qmckl_woodbury_2x2
The Woodbury 2x2 kernel. It is used to apply two rank-1 updates at once. The formula used in this algorithm is called the Woodbury Matrix Identity \[ (S + U V)^{-1} = S^{-1} - C B^{-1} D \] where \(S\) is the Slater-matrix \(U\) and \(V\) are the matrices containing the updates and the canonical basis matrix \(S^{-1}\) is the inverse of the Slater-matrix \(C:= S^{-1}U\), a Dim \(\times 2\) matrix \(B := 1 + VC\), the \(2 \times 2\) matrix that is going to be inverted \(D := VS^{-1}\), a \(2 \times Dim\) matrix
If the determinant of the Slater-matrix is passed, it will be updated to the determinant resulting from applying the updates to the original matrix.
qmcklcontext | context | in | Global state |
uint64t | LDS | in | Leading dimension of Slaterinv |
uint64t | Dim | in | Dimension of Slaterinv |
double | Updates[2*Dim] | in | Array containing the updates |
uint64t | Updatesindex[2] | in | Array containing the rank-1 updates |
double | breakdown | in | Break-down parameter on which to fail or not |
double | Slaterinv[LDS*Dim] | inout | Array containing the inverse of a Slater-matrix |
double* | determinant | inout | Determinant of Slater-matrix |
3.1.1 Requirements
context
is notqmckl_null_context
LDS >= 2
Dim >= 2
Updates
is allocated with \(2 \times Dim\) elementsUpdates_index
is allocated with \(2\) elementsbreakdown
is a small number such that \(0 < breakdown << 1\)Slater_inv
is allocated with \(Dim \times Dim\) elements
3.1.2 C header
qmckl_exit_code qmckl_woodbury_2x2 ( const context qmckl_context, const LDS uint64_t, const Dim uint64_t, const Updates* double, const Updates_index* uint64_t, const breakdown double, Slater_inv* double, determinant* double* );
3.1.3 C source
qmckl_exit_code qmckl_woodbury_2x2_hpc(const qmckl_context context, const uint64_t LDS, const uint64_t Dim, const double* __restrict Updates, const uint64_t* __restrict Updates_index, const double breakdown, double* __restrict Slater_inv, double* __restrict determinant) { /* C := S^{-1} * U, dim x 2 B := 1 + V * C, 2 x 2 D := V * S^{-1}, 2 x dim */ if (qmckl_context_check(context) == QMCKL_NULL_CONTEXT) { return qmckl_failwith(context, QMCKL_NULL_CONTEXT, "qmckl_woodbury_2x2_hpc", NULL); } const uint64_t row1 = (Updates_index[0] - 1); const uint64_t row2 = (Updates_index[1] - 1); // Compute C = (S^T)^{-1}U : Dim x 2 double __attribute__((aligned(8))) C[2 * Dim]; for (uint64_t i = 0; i < Dim; i++) { C[i * 2] = 0; C[i * 2 + 1] = 0; IVDEP ALIGNED for (uint64_t k = 0; k < LDS; k++) { C[i * 2] += Slater_inv[i * LDS + k] * Updates[k]; C[i * 2 + 1] += Slater_inv[i * LDS + k] * Updates[LDS + k]; } } // Compute B = 1 + VC : 2 x 2 const double B0 = C[row1 * 2] + 1; const double B1 = C[row1 * 2 + 1]; const double B2 = C[row2 * 2]; const double B3 = C[row2 * 2 + 1] + 1; // Check if determinant of inverted matrix is not zero double det = B0 * B3 - B1 * B2; if (fabs(det) < breakdown) { return QMCKL_FAILURE; } // Update det(S) when passed if (determinant) *determinant *= det; // Compute B^{-1} with explicit formula for 2 x 2 inversion double __attribute__((aligned(8))) Binv[4], idet = 1.0 / det; Binv[0] = idet * B3; Binv[1] = -1.0 * idet * B1; Binv[2] = -1.0 * idet * B2; Binv[3] = idet * B0; // tmp = B^{-1}D : 2 x LDS double __attribute__((aligned(8))) tmp[2 * LDS]; double* r1dim = &(Slater_inv[row1 * LDS]); double* r2dim = &(Slater_inv[row2 * LDS]); IVDEP ALIGNED for (uint64_t j = 0; j < LDS; j++) { tmp[j] = Binv[0] * r1dim[j] + Binv[1] * r2dim[j]; tmp[LDS + j] = Binv[2] * r1dim[j] + Binv[3] * r2dim[j]; } // Compute (S^T)^{-1} - C * tmp : Dim x LDS for (uint64_t i = 0; i < Dim; i++) { IVDEP ALIGNED for (uint64_t j = 0; j < LDS; j++) { Slater_inv[i * LDS + j] -= C[i * 2] * tmp[j]; Slater_inv[i * LDS + j] -= C[i * 2 + 1] * tmp[LDS + j]; } } return QMCKL_SUCCESS; }
static inline qmckl_exit_code qmckl_woodbury_2x2_{Dim}( const qmckl_context context, const double* __restrict Updates, const uint64_t* __restrict Updates_index, const double breakdown, double* __restrict Slater_inv, double* __restrict determinant) { /* C := S^{-1} * U, dim x 2 B := 1 + V * C, 2 x 2 D := V * S^{-1}, 2 x dim */ if (qmckl_context_check(context) == QMCKL_NULL_CONTEXT) { return qmckl_failwith(context, QMCKL_NULL_CONTEXT, "qmckl_woodbury_2x2_{Dim}", NULL); } const uint64_t row1 = (Updates_index[0] - 1); const uint64_t row2 = (Updates_index[1] - 1); // Compute C = (S^T)^{-1}U : {Dim} x 2 double __attribute__((aligned(8))) C[2 * {Dim}]; for (uint64_t i = 0; i < {Dim}; i++) { C[i * 2] = 0; C[i * 2 + 1] = 0; IVDEP ALIGNED for (uint64_t k = 0; k < D{Dim}_P; k++) { C[i * 2] += Slater_inv[i * D{Dim}_P + k] * Updates[k]; C[i * 2 + 1] += Slater_inv[i * D{Dim}_P + k] * Updates[D{Dim}_P + k]; } } // Compute B = 1 + VC : 2 x 2 const double B0 = C[row1 * 2] + 1; const double B1 = C[row1 * 2 + 1]; const double B2 = C[row2 * 2]; const double B3 = C[row2 * 2 + 1] + 1; // Check if determinant of inverted matrix is not zero double det = B0 * B3 - B1 * B2; if (fabs(det) < breakdown) { return QMCKL_FAILURE; } // Update det(S) when passed if (determinant) *determinant *= det; // Compute B^{-1} with explicit formula for 2 x 2 inversion double __attribute__((aligned(8))) Binv[4], idet = 1.0 / det; Binv[0] = idet * B3; Binv[1] = -1.0 * idet * B1; Binv[2] = -1.0 * idet * B2; Binv[3] = idet * B0; // tmp = B^{-1}D : 2 x D{Dim}_P double __attribute__((aligned(8))) tmp[2 * D{Dim}_P]; double* r1dim = &(Slater_inv[row1 * D{Dim}_P]); double* r2dim = &(Slater_inv[row2 * D{Dim}_P]); IVDEP ALIGNED for (uint64_t j = 0; j < D{Dim}_P; j++) { tmp[j] = Binv[0] * r1dim[j] + Binv[1] * r2dim[j]; tmp[D{Dim}_P + j] = Binv[2] * r1dim[j] + Binv[3] * r2dim[j]; } // Compute (S^T)^{-1} - C * tmp : {Dim} x D{Dim}_P for (uint64_t i = 0; i < {Dim}; i++) { IVDEP ALIGNED for (uint64_t j = 0; j < D{Dim}_P; j++) { Slater_inv[i * D{Dim}_P + j] -= C[i * 2] * tmp[j]; Slater_inv[i * D{Dim}_P + j] -= C[i * 2 + 1] * tmp[D{Dim}_P + j]; } } return QMCKL_SUCCESS; }
static inline qmckl_exit_code qmckl_woodbury_2x2_2( const qmckl_context context, const double* __restrict Updates, const uint64_t* __restrict Updates_index, const double breakdown, double* __restrict Slater_inv, double* __restrict determinant) { /* C := S^{-1} * U, dim x 2 B := 1 + V * C, 2 x 2 D := V * S^{-1}, 2 x dim */ if (qmckl_context_check(context) == QMCKL_NULL_CONTEXT) { return qmckl_failwith(context, QMCKL_NULL_CONTEXT, "qmckl_woodbury_2x2_2", NULL); } const uint64_t row1 = (Updates_index[0] - 1); const uint64_t row2 = (Updates_index[1] - 1); // Compute C = (S^T)^{-1}U : 2 x 2 double __attribute__((aligned(8))) C[2 * 2]; for (uint64_t i = 0; i < 2; i++) { C[i * 2] = 0; C[i * 2 + 1] = 0; IVDEP ALIGNED for (uint64_t k = 0; k < D2_P; k++) { C[i * 2] += Slater_inv[i * D2_P + k] * Updates[k]; C[i * 2 + 1] += Slater_inv[i * D2_P + k] * Updates[D2_P + k]; } } // Compute B = 1 + VC : 2 x 2 const double B0 = C[row1 * 2] + 1; const double B1 = C[row1 * 2 + 1]; const double B2 = C[row2 * 2]; const double B3 = C[row2 * 2 + 1] + 1; // Check if determinant of inverted matrix is not zero double det = B0 * B3 - B1 * B2; if (fabs(det) < breakdown) { return QMCKL_FAILURE; } // Update det(S) when passed if (determinant) *determinant *= det; // Compute B^{-1} with explicit formula for 2 x 2 inversion double __attribute__((aligned(8))) Binv[4], idet = 1.0 / det; Binv[0] = idet * B3; Binv[1] = -1.0 * idet * B1; Binv[2] = -1.0 * idet * B2; Binv[3] = idet * B0; // tmp = B^{-1}D : 2 x D2_P double __attribute__((aligned(8))) tmp[2 * D2_P]; double* r1dim = &(Slater_inv[row1 * D2_P]); double* r2dim = &(Slater_inv[row2 * D2_P]); IVDEP ALIGNED for (uint64_t j = 0; j < D2_P; j++) { tmp[j] = Binv[0] * r1dim[j] + Binv[1] * r2dim[j]; tmp[D2_P + j] = Binv[2] * r1dim[j] + Binv[3] * r2dim[j]; } // Compute (S^T)^{-1} - C * tmp : 2 x D2_P for (uint64_t i = 0; i < 2; i++) { IVDEP ALIGNED for (uint64_t j = 0; j < D2_P; j++) { Slater_inv[i * D2_P + j] -= C[i * 2] * tmp[j]; Slater_inv[i * D2_P + j] -= C[i * 2 + 1] * tmp[D2_P + j]; } } return QMCKL_SUCCESS; } static inline qmckl_exit_code qmckl_woodbury_2x2_3( const qmckl_context context, const double* __restrict Updates, const uint64_t* __restrict Updates_index, const double breakdown, double* __restrict Slater_inv, double* __restrict determinant) { /* C := S^{-1} * U, dim x 2 B := 1 + V * C, 2 x 2 D := V * S^{-1}, 2 x dim */ if (qmckl_context_check(context) == QMCKL_NULL_CONTEXT) { return qmckl_failwith(context, QMCKL_NULL_CONTEXT, "qmckl_woodbury_2x2_3", NULL); } const uint64_t row1 = (Updates_index[0] - 1); const uint64_t row2 = (Updates_index[1] - 1); // Compute C = (S^T)^{-1}U : 3 x 2 double __attribute__((aligned(8))) C[2 * 3]; for (uint64_t i = 0; i < 3; i++) { C[i * 2] = 0; C[i * 2 + 1] = 0; IVDEP ALIGNED for (uint64_t k = 0; k < D3_P; k++) { C[i * 2] += Slater_inv[i * D3_P + k] * Updates[k]; C[i * 2 + 1] += Slater_inv[i * D3_P + k] * Updates[D3_P + k]; } } // Compute B = 1 + VC : 2 x 2 const double B0 = C[row1 * 2] + 1; const double B1 = C[row1 * 2 + 1]; const double B2 = C[row2 * 2]; const double B3 = C[row2 * 2 + 1] + 1; // Check if determinant of inverted matrix is not zero double det = B0 * B3 - B1 * B2; if (fabs(det) < breakdown) { return QMCKL_FAILURE; } // Update det(S) when passed if (determinant) *determinant *= det; // Compute B^{-1} with explicit formula for 2 x 2 inversion double __attribute__((aligned(8))) Binv[4], idet = 1.0 / det; Binv[0] = idet * B3; Binv[1] = -1.0 * idet * B1; Binv[2] = -1.0 * idet * B2; Binv[3] = idet * B0; // tmp = B^{-1}D : 2 x D3_P double __attribute__((aligned(8))) tmp[2 * D3_P]; double* r1dim = &(Slater_inv[row1 * D3_P]); double* r2dim = &(Slater_inv[row2 * D3_P]); IVDEP ALIGNED for (uint64_t j = 0; j < D3_P; j++) { tmp[j] = Binv[0] * r1dim[j] + Binv[1] * r2dim[j]; tmp[D3_P + j] = Binv[2] * r1dim[j] + Binv[3] * r2dim[j]; } // Compute (S^T)^{-1} - C * tmp : 3 x D3_P for (uint64_t i = 0; i < 3; i++) { IVDEP ALIGNED for (uint64_t j = 0; j < D3_P; j++) { Slater_inv[i * D3_P + j] -= C[i * 2] * tmp[j]; Slater_inv[i * D3_P + j] -= C[i * 2 + 1] * tmp[D3_P + j]; } } return QMCKL_SUCCESS; } static inline qmckl_exit_code qmckl_woodbury_2x2_4( const qmckl_context context, const double* __restrict Updates, const uint64_t* __restrict Updates_index, const double breakdown, double* __restrict Slater_inv, double* __restrict determinant) { /* C := S^{-1} * U, dim x 2 B := 1 + V * C, 2 x 2 D := V * S^{-1}, 2 x dim */ if (qmckl_context_check(context) == QMCKL_NULL_CONTEXT) { return qmckl_failwith(context, QMCKL_NULL_CONTEXT, "qmckl_woodbury_2x2_4", NULL); } const uint64_t row1 = (Updates_index[0] - 1); const uint64_t row2 = (Updates_index[1] - 1); // Compute C = (S^T)^{-1}U : 4 x 2 double __attribute__((aligned(8))) C[2 * 4]; for (uint64_t i = 0; i < 4; i++) { C[i * 2] = 0; C[i * 2 + 1] = 0; IVDEP ALIGNED for (uint64_t k = 0; k < D4_P; k++) { C[i * 2] += Slater_inv[i * D4_P + k] * Updates[k]; C[i * 2 + 1] += Slater_inv[i * D4_P + k] * Updates[D4_P + k]; } } // Compute B = 1 + VC : 2 x 2 const double B0 = C[row1 * 2] + 1; const double B1 = C[row1 * 2 + 1]; const double B2 = C[row2 * 2]; const double B3 = C[row2 * 2 + 1] + 1; // Check if determinant of inverted matrix is not zero double det = B0 * B3 - B1 * B2; if (fabs(det) < breakdown) { return QMCKL_FAILURE; } // Update det(S) when passed if (determinant) *determinant *= det; // Compute B^{-1} with explicit formula for 2 x 2 inversion double __attribute__((aligned(8))) Binv[4], idet = 1.0 / det; Binv[0] = idet * B3; Binv[1] = -1.0 * idet * B1; Binv[2] = -1.0 * idet * B2; Binv[3] = idet * B0; // tmp = B^{-1}D : 2 x D4_P double __attribute__((aligned(8))) tmp[2 * D4_P]; double* r1dim = &(Slater_inv[row1 * D4_P]); double* r2dim = &(Slater_inv[row2 * D4_P]); IVDEP ALIGNED for (uint64_t j = 0; j < D4_P; j++) { tmp[j] = Binv[0] * r1dim[j] + Binv[1] * r2dim[j]; tmp[D4_P + j] = Binv[2] * r1dim[j] + Binv[3] * r2dim[j]; } // Compute (S^T)^{-1} - C * tmp : 4 x D4_P for (uint64_t i = 0; i < 4; i++) { IVDEP ALIGNED for (uint64_t j = 0; j < D4_P; j++) { Slater_inv[i * D4_P + j] -= C[i * 2] * tmp[j]; Slater_inv[i * D4_P + j] -= C[i * 2 + 1] * tmp[D4_P + j]; } } return QMCKL_SUCCESS; } static inline qmckl_exit_code qmckl_woodbury_2x2_5( const qmckl_context context, const double* __restrict Updates, const uint64_t* __restrict Updates_index, const double breakdown, double* __restrict Slater_inv, double* __restrict determinant) { /* C := S^{-1} * U, dim x 2 B := 1 + V * C, 2 x 2 D := V * S^{-1}, 2 x dim */ if (qmckl_context_check(context) == QMCKL_NULL_CONTEXT) { return qmckl_failwith(context, QMCKL_NULL_CONTEXT, "qmckl_woodbury_2x2_5", NULL); } const uint64_t row1 = (Updates_index[0] - 1); const uint64_t row2 = (Updates_index[1] - 1); // Compute C = (S^T)^{-1}U : 5 x 2 double __attribute__((aligned(8))) C[2 * 5]; for (uint64_t i = 0; i < 5; i++) { C[i * 2] = 0; C[i * 2 + 1] = 0; IVDEP ALIGNED for (uint64_t k = 0; k < D5_P; k++) { C[i * 2] += Slater_inv[i * D5_P + k] * Updates[k]; C[i * 2 + 1] += Slater_inv[i * D5_P + k] * Updates[D5_P + k]; } } // Compute B = 1 + VC : 2 x 2 const double B0 = C[row1 * 2] + 1; const double B1 = C[row1 * 2 + 1]; const double B2 = C[row2 * 2]; const double B3 = C[row2 * 2 + 1] + 1; // Check if determinant of inverted matrix is not zero double det = B0 * B3 - B1 * B2; if (fabs(det) < breakdown) { return QMCKL_FAILURE; } // Update det(S) when passed if (determinant) *determinant *= det; // Compute B^{-1} with explicit formula for 2 x 2 inversion double __attribute__((aligned(8))) Binv[4], idet = 1.0 / det; Binv[0] = idet * B3; Binv[1] = -1.0 * idet * B1; Binv[2] = -1.0 * idet * B2; Binv[3] = idet * B0; // tmp = B^{-1}D : 2 x D5_P double __attribute__((aligned(8))) tmp[2 * D5_P]; double* r1dim = &(Slater_inv[row1 * D5_P]); double* r2dim = &(Slater_inv[row2 * D5_P]); IVDEP ALIGNED for (uint64_t j = 0; j < D5_P; j++) { tmp[j] = Binv[0] * r1dim[j] + Binv[1] * r2dim[j]; tmp[D5_P + j] = Binv[2] * r1dim[j] + Binv[3] * r2dim[j]; } // Compute (S^T)^{-1} - C * tmp : 5 x D5_P for (uint64_t i = 0; i < 5; i++) { IVDEP ALIGNED for (uint64_t j = 0; j < D5_P; j++) { Slater_inv[i * D5_P + j] -= C[i * 2] * tmp[j]; Slater_inv[i * D5_P + j] -= C[i * 2 + 1] * tmp[D5_P + j]; } } return QMCKL_SUCCESS; } static inline qmckl_exit_code qmckl_woodbury_2x2_6( const qmckl_context context, const double* __restrict Updates, const uint64_t* __restrict Updates_index, const double breakdown, double* __restrict Slater_inv, double* __restrict determinant) { /* C := S^{-1} * U, dim x 2 B := 1 + V * C, 2 x 2 D := V * S^{-1}, 2 x dim */ if (qmckl_context_check(context) == QMCKL_NULL_CONTEXT) { return qmckl_failwith(context, QMCKL_NULL_CONTEXT, "qmckl_woodbury_2x2_6", NULL); } const uint64_t row1 = (Updates_index[0] - 1); const uint64_t row2 = (Updates_index[1] - 1); // Compute C = (S^T)^{-1}U : 6 x 2 double __attribute__((aligned(8))) C[2 * 6]; for (uint64_t i = 0; i < 6; i++) { C[i * 2] = 0; C[i * 2 + 1] = 0; IVDEP ALIGNED for (uint64_t k = 0; k < D6_P; k++) { C[i * 2] += Slater_inv[i * D6_P + k] * Updates[k]; C[i * 2 + 1] += Slater_inv[i * D6_P + k] * Updates[D6_P + k]; } } // Compute B = 1 + VC : 2 x 2 const double B0 = C[row1 * 2] + 1; const double B1 = C[row1 * 2 + 1]; const double B2 = C[row2 * 2]; const double B3 = C[row2 * 2 + 1] + 1; // Check if determinant of inverted matrix is not zero double det = B0 * B3 - B1 * B2; if (fabs(det) < breakdown) { return QMCKL_FAILURE; } // Update det(S) when passed if (determinant) *determinant *= det; // Compute B^{-1} with explicit formula for 2 x 2 inversion double __attribute__((aligned(8))) Binv[4], idet = 1.0 / det; Binv[0] = idet * B3; Binv[1] = -1.0 * idet * B1; Binv[2] = -1.0 * idet * B2; Binv[3] = idet * B0; // tmp = B^{-1}D : 2 x D6_P double __attribute__((aligned(8))) tmp[2 * D6_P]; double* r1dim = &(Slater_inv[row1 * D6_P]); double* r2dim = &(Slater_inv[row2 * D6_P]); IVDEP ALIGNED for (uint64_t j = 0; j < D6_P; j++) { tmp[j] = Binv[0] * r1dim[j] + Binv[1] * r2dim[j]; tmp[D6_P + j] = Binv[2] * r1dim[j] + Binv[3] * r2dim[j]; } // Compute (S^T)^{-1} - C * tmp : 6 x D6_P for (uint64_t i = 0; i < 6; i++) { IVDEP ALIGNED for (uint64_t j = 0; j < D6_P; j++) { Slater_inv[i * D6_P + j] -= C[i * 2] * tmp[j]; Slater_inv[i * D6_P + j] -= C[i * 2 + 1] * tmp[D6_P + j]; } } return QMCKL_SUCCESS; } static inline qmckl_exit_code qmckl_woodbury_2x2_7( const qmckl_context context, const double* __restrict Updates, const uint64_t* __restrict Updates_index, const double breakdown, double* __restrict Slater_inv, double* __restrict determinant) { /* C := S^{-1} * U, dim x 2 B := 1 + V * C, 2 x 2 D := V * S^{-1}, 2 x dim */ if (qmckl_context_check(context) == QMCKL_NULL_CONTEXT) { return qmckl_failwith(context, QMCKL_NULL_CONTEXT, "qmckl_woodbury_2x2_7", NULL); } const uint64_t row1 = (Updates_index[0] - 1); const uint64_t row2 = (Updates_index[1] - 1); // Compute C = (S^T)^{-1}U : 7 x 2 double __attribute__((aligned(8))) C[2 * 7]; for (uint64_t i = 0; i < 7; i++) { C[i * 2] = 0; C[i * 2 + 1] = 0; IVDEP ALIGNED for (uint64_t k = 0; k < D7_P; k++) { C[i * 2] += Slater_inv[i * D7_P + k] * Updates[k]; C[i * 2 + 1] += Slater_inv[i * D7_P + k] * Updates[D7_P + k]; } } // Compute B = 1 + VC : 2 x 2 const double B0 = C[row1 * 2] + 1; const double B1 = C[row1 * 2 + 1]; const double B2 = C[row2 * 2]; const double B3 = C[row2 * 2 + 1] + 1; // Check if determinant of inverted matrix is not zero double det = B0 * B3 - B1 * B2; if (fabs(det) < breakdown) { return QMCKL_FAILURE; } // Update det(S) when passed if (determinant) *determinant *= det; // Compute B^{-1} with explicit formula for 2 x 2 inversion double __attribute__((aligned(8))) Binv[4], idet = 1.0 / det; Binv[0] = idet * B3; Binv[1] = -1.0 * idet * B1; Binv[2] = -1.0 * idet * B2; Binv[3] = idet * B0; // tmp = B^{-1}D : 2 x D7_P double __attribute__((aligned(8))) tmp[2 * D7_P]; double* r1dim = &(Slater_inv[row1 * D7_P]); double* r2dim = &(Slater_inv[row2 * D7_P]); IVDEP ALIGNED for (uint64_t j = 0; j < D7_P; j++) { tmp[j] = Binv[0] * r1dim[j] + Binv[1] * r2dim[j]; tmp[D7_P + j] = Binv[2] * r1dim[j] + Binv[3] * r2dim[j]; } // Compute (S^T)^{-1} - C * tmp : 7 x D7_P for (uint64_t i = 0; i < 7; i++) { IVDEP ALIGNED for (uint64_t j = 0; j < D7_P; j++) { Slater_inv[i * D7_P + j] -= C[i * 2] * tmp[j]; Slater_inv[i * D7_P + j] -= C[i * 2 + 1] * tmp[D7_P + j]; } } return QMCKL_SUCCESS; } static inline qmckl_exit_code qmckl_woodbury_2x2_8( const qmckl_context context, const double* __restrict Updates, const uint64_t* __restrict Updates_index, const double breakdown, double* __restrict Slater_inv, double* __restrict determinant) { /* C := S^{-1} * U, dim x 2 B := 1 + V * C, 2 x 2 D := V * S^{-1}, 2 x dim */ if (qmckl_context_check(context) == QMCKL_NULL_CONTEXT) { return qmckl_failwith(context, QMCKL_NULL_CONTEXT, "qmckl_woodbury_2x2_8", NULL); } const uint64_t row1 = (Updates_index[0] - 1); const uint64_t row2 = (Updates_index[1] - 1); // Compute C = (S^T)^{-1}U : 8 x 2 double __attribute__((aligned(8))) C[2 * 8]; for (uint64_t i = 0; i < 8; i++) { C[i * 2] = 0; C[i * 2 + 1] = 0; IVDEP ALIGNED for (uint64_t k = 0; k < D8_P; k++) { C[i * 2] += Slater_inv[i * D8_P + k] * Updates[k]; C[i * 2 + 1] += Slater_inv[i * D8_P + k] * Updates[D8_P + k]; } } // Compute B = 1 + VC : 2 x 2 const double B0 = C[row1 * 2] + 1; const double B1 = C[row1 * 2 + 1]; const double B2 = C[row2 * 2]; const double B3 = C[row2 * 2 + 1] + 1; // Check if determinant of inverted matrix is not zero double det = B0 * B3 - B1 * B2; if (fabs(det) < breakdown) { return QMCKL_FAILURE; } // Update det(S) when passed if (determinant) *determinant *= det; // Compute B^{-1} with explicit formula for 2 x 2 inversion double __attribute__((aligned(8))) Binv[4], idet = 1.0 / det; Binv[0] = idet * B3; Binv[1] = -1.0 * idet * B1; Binv[2] = -1.0 * idet * B2; Binv[3] = idet * B0; // tmp = B^{-1}D : 2 x D8_P double __attribute__((aligned(8))) tmp[2 * D8_P]; double* r1dim = &(Slater_inv[row1 * D8_P]); double* r2dim = &(Slater_inv[row2 * D8_P]); IVDEP ALIGNED for (uint64_t j = 0; j < D8_P; j++) { tmp[j] = Binv[0] * r1dim[j] + Binv[1] * r2dim[j]; tmp[D8_P + j] = Binv[2] * r1dim[j] + Binv[3] * r2dim[j]; } // Compute (S^T)^{-1} - C * tmp : 8 x D8_P for (uint64_t i = 0; i < 8; i++) { IVDEP ALIGNED for (uint64_t j = 0; j < D8_P; j++) { Slater_inv[i * D8_P + j] -= C[i * 2] * tmp[j]; Slater_inv[i * D8_P + j] -= C[i * 2 + 1] * tmp[D8_P + j]; } } return QMCKL_SUCCESS; } static inline qmckl_exit_code qmckl_woodbury_2x2_9( const qmckl_context context, const double* __restrict Updates, const uint64_t* __restrict Updates_index, const double breakdown, double* __restrict Slater_inv, double* __restrict determinant) { /* C := S^{-1} * U, dim x 2 B := 1 + V * C, 2 x 2 D := V * S^{-1}, 2 x dim */ if (qmckl_context_check(context) == QMCKL_NULL_CONTEXT) { return qmckl_failwith(context, QMCKL_NULL_CONTEXT, "qmckl_woodbury_2x2_9", NULL); } const uint64_t row1 = (Updates_index[0] - 1); const uint64_t row2 = (Updates_index[1] - 1); // Compute C = (S^T)^{-1}U : 9 x 2 double __attribute__((aligned(8))) C[2 * 9]; for (uint64_t i = 0; i < 9; i++) { C[i * 2] = 0; C[i * 2 + 1] = 0; IVDEP ALIGNED for (uint64_t k = 0; k < D9_P; k++) { C[i * 2] += Slater_inv[i * D9_P + k] * Updates[k]; C[i * 2 + 1] += Slater_inv[i * D9_P + k] * Updates[D9_P + k]; } } // Compute B = 1 + VC : 2 x 2 const double B0 = C[row1 * 2] + 1; const double B1 = C[row1 * 2 + 1]; const double B2 = C[row2 * 2]; const double B3 = C[row2 * 2 + 1] + 1; // Check if determinant of inverted matrix is not zero double det = B0 * B3 - B1 * B2; if (fabs(det) < breakdown) { return QMCKL_FAILURE; } // Update det(S) when passed if (determinant) *determinant *= det; // Compute B^{-1} with explicit formula for 2 x 2 inversion double __attribute__((aligned(8))) Binv[4], idet = 1.0 / det; Binv[0] = idet * B3; Binv[1] = -1.0 * idet * B1; Binv[2] = -1.0 * idet * B2; Binv[3] = idet * B0; // tmp = B^{-1}D : 2 x D9_P double __attribute__((aligned(8))) tmp[2 * D9_P]; double* r1dim = &(Slater_inv[row1 * D9_P]); double* r2dim = &(Slater_inv[row2 * D9_P]); IVDEP ALIGNED for (uint64_t j = 0; j < D9_P; j++) { tmp[j] = Binv[0] * r1dim[j] + Binv[1] * r2dim[j]; tmp[D9_P + j] = Binv[2] * r1dim[j] + Binv[3] * r2dim[j]; } // Compute (S^T)^{-1} - C * tmp : 9 x D9_P for (uint64_t i = 0; i < 9; i++) { IVDEP ALIGNED for (uint64_t j = 0; j < D9_P; j++) { Slater_inv[i * D9_P + j] -= C[i * 2] * tmp[j]; Slater_inv[i * D9_P + j] -= C[i * 2 + 1] * tmp[D9_P + j]; } } return QMCKL_SUCCESS; } static inline qmckl_exit_code qmckl_woodbury_2x2_10( const qmckl_context context, const double* __restrict Updates, const uint64_t* __restrict Updates_index, const double breakdown, double* __restrict Slater_inv, double* __restrict determinant) { /* C := S^{-1} * U, dim x 2 B := 1 + V * C, 2 x 2 D := V * S^{-1}, 2 x dim */ if (qmckl_context_check(context) == QMCKL_NULL_CONTEXT) { return qmckl_failwith(context, QMCKL_NULL_CONTEXT, "qmckl_woodbury_2x2_10", NULL); } const uint64_t row1 = (Updates_index[0] - 1); const uint64_t row2 = (Updates_index[1] - 1); // Compute C = (S^T)^{-1}U : 10 x 2 double __attribute__((aligned(8))) C[2 * 10]; for (uint64_t i = 0; i < 10; i++) { C[i * 2] = 0; C[i * 2 + 1] = 0; IVDEP ALIGNED for (uint64_t k = 0; k < D10_P; k++) { C[i * 2] += Slater_inv[i * D10_P + k] * Updates[k]; C[i * 2 + 1] += Slater_inv[i * D10_P + k] * Updates[D10_P + k]; } } // Compute B = 1 + VC : 2 x 2 const double B0 = C[row1 * 2] + 1; const double B1 = C[row1 * 2 + 1]; const double B2 = C[row2 * 2]; const double B3 = C[row2 * 2 + 1] + 1; // Check if determinant of inverted matrix is not zero double det = B0 * B3 - B1 * B2; if (fabs(det) < breakdown) { return QMCKL_FAILURE; } // Update det(S) when passed if (determinant) *determinant *= det; // Compute B^{-1} with explicit formula for 2 x 2 inversion double __attribute__((aligned(8))) Binv[4], idet = 1.0 / det; Binv[0] = idet * B3; Binv[1] = -1.0 * idet * B1; Binv[2] = -1.0 * idet * B2; Binv[3] = idet * B0; // tmp = B^{-1}D : 2 x D10_P double __attribute__((aligned(8))) tmp[2 * D10_P]; double* r1dim = &(Slater_inv[row1 * D10_P]); double* r2dim = &(Slater_inv[row2 * D10_P]); IVDEP ALIGNED for (uint64_t j = 0; j < D10_P; j++) { tmp[j] = Binv[0] * r1dim[j] + Binv[1] * r2dim[j]; tmp[D10_P + j] = Binv[2] * r1dim[j] + Binv[3] * r2dim[j]; } // Compute (S^T)^{-1} - C * tmp : 10 x D10_P for (uint64_t i = 0; i < 10; i++) { IVDEP ALIGNED for (uint64_t j = 0; j < D10_P; j++) { Slater_inv[i * D10_P + j] -= C[i * 2] * tmp[j]; Slater_inv[i * D10_P + j] -= C[i * 2 + 1] * tmp[D10_P + j]; } } return QMCKL_SUCCESS; } static inline qmckl_exit_code qmckl_woodbury_2x2_11( const qmckl_context context, const double* __restrict Updates, const uint64_t* __restrict Updates_index, const double breakdown, double* __restrict Slater_inv, double* __restrict determinant) { /* C := S^{-1} * U, dim x 2 B := 1 + V * C, 2 x 2 D := V * S^{-1}, 2 x dim */ if (qmckl_context_check(context) == QMCKL_NULL_CONTEXT) { return qmckl_failwith(context, QMCKL_NULL_CONTEXT, "qmckl_woodbury_2x2_11", NULL); } const uint64_t row1 = (Updates_index[0] - 1); const uint64_t row2 = (Updates_index[1] - 1); // Compute C = (S^T)^{-1}U : 11 x 2 double __attribute__((aligned(8))) C[2 * 11]; for (uint64_t i = 0; i < 11; i++) { C[i * 2] = 0; C[i * 2 + 1] = 0; IVDEP ALIGNED for (uint64_t k = 0; k < D11_P; k++) { C[i * 2] += Slater_inv[i * D11_P + k] * Updates[k]; C[i * 2 + 1] += Slater_inv[i * D11_P + k] * Updates[D11_P + k]; } } // Compute B = 1 + VC : 2 x 2 const double B0 = C[row1 * 2] + 1; const double B1 = C[row1 * 2 + 1]; const double B2 = C[row2 * 2]; const double B3 = C[row2 * 2 + 1] + 1; // Check if determinant of inverted matrix is not zero double det = B0 * B3 - B1 * B2; if (fabs(det) < breakdown) { return QMCKL_FAILURE; } // Update det(S) when passed if (determinant) *determinant *= det; // Compute B^{-1} with explicit formula for 2 x 2 inversion double __attribute__((aligned(8))) Binv[4], idet = 1.0 / det; Binv[0] = idet * B3; Binv[1] = -1.0 * idet * B1; Binv[2] = -1.0 * idet * B2; Binv[3] = idet * B0; // tmp = B^{-1}D : 2 x D11_P double __attribute__((aligned(8))) tmp[2 * D11_P]; double* r1dim = &(Slater_inv[row1 * D11_P]); double* r2dim = &(Slater_inv[row2 * D11_P]); IVDEP ALIGNED for (uint64_t j = 0; j < D11_P; j++) { tmp[j] = Binv[0] * r1dim[j] + Binv[1] * r2dim[j]; tmp[D11_P + j] = Binv[2] * r1dim[j] + Binv[3] * r2dim[j]; } // Compute (S^T)^{-1} - C * tmp : 11 x D11_P for (uint64_t i = 0; i < 11; i++) { IVDEP ALIGNED for (uint64_t j = 0; j < D11_P; j++) { Slater_inv[i * D11_P + j] -= C[i * 2] * tmp[j]; Slater_inv[i * D11_P + j] -= C[i * 2 + 1] * tmp[D11_P + j]; } } return QMCKL_SUCCESS; } static inline qmckl_exit_code qmckl_woodbury_2x2_12( const qmckl_context context, const double* __restrict Updates, const uint64_t* __restrict Updates_index, const double breakdown, double* __restrict Slater_inv, double* __restrict determinant) { /* C := S^{-1} * U, dim x 2 B := 1 + V * C, 2 x 2 D := V * S^{-1}, 2 x dim */ if (qmckl_context_check(context) == QMCKL_NULL_CONTEXT) { return qmckl_failwith(context, QMCKL_NULL_CONTEXT, "qmckl_woodbury_2x2_12", NULL); } const uint64_t row1 = (Updates_index[0] - 1); const uint64_t row2 = (Updates_index[1] - 1); // Compute C = (S^T)^{-1}U : 12 x 2 double __attribute__((aligned(8))) C[2 * 12]; for (uint64_t i = 0; i < 12; i++) { C[i * 2] = 0; C[i * 2 + 1] = 0; IVDEP ALIGNED for (uint64_t k = 0; k < D12_P; k++) { C[i * 2] += Slater_inv[i * D12_P + k] * Updates[k]; C[i * 2 + 1] += Slater_inv[i * D12_P + k] * Updates[D12_P + k]; } } // Compute B = 1 + VC : 2 x 2 const double B0 = C[row1 * 2] + 1; const double B1 = C[row1 * 2 + 1]; const double B2 = C[row2 * 2]; const double B3 = C[row2 * 2 + 1] + 1; // Check if determinant of inverted matrix is not zero double det = B0 * B3 - B1 * B2; if (fabs(det) < breakdown) { return QMCKL_FAILURE; } // Update det(S) when passed if (determinant) *determinant *= det; // Compute B^{-1} with explicit formula for 2 x 2 inversion double __attribute__((aligned(8))) Binv[4], idet = 1.0 / det; Binv[0] = idet * B3; Binv[1] = -1.0 * idet * B1; Binv[2] = -1.0 * idet * B2; Binv[3] = idet * B0; // tmp = B^{-1}D : 2 x D12_P double __attribute__((aligned(8))) tmp[2 * D12_P]; double* r1dim = &(Slater_inv[row1 * D12_P]); double* r2dim = &(Slater_inv[row2 * D12_P]); IVDEP ALIGNED for (uint64_t j = 0; j < D12_P; j++) { tmp[j] = Binv[0] * r1dim[j] + Binv[1] * r2dim[j]; tmp[D12_P + j] = Binv[2] * r1dim[j] + Binv[3] * r2dim[j]; } // Compute (S^T)^{-1} - C * tmp : 12 x D12_P for (uint64_t i = 0; i < 12; i++) { IVDEP ALIGNED for (uint64_t j = 0; j < D12_P; j++) { Slater_inv[i * D12_P + j] -= C[i * 2] * tmp[j]; Slater_inv[i * D12_P + j] -= C[i * 2 + 1] * tmp[D12_P + j]; } } return QMCKL_SUCCESS; } static inline qmckl_exit_code qmckl_woodbury_2x2_13( const qmckl_context context, const double* __restrict Updates, const uint64_t* __restrict Updates_index, const double breakdown, double* __restrict Slater_inv, double* __restrict determinant) { /* C := S^{-1} * U, dim x 2 B := 1 + V * C, 2 x 2 D := V * S^{-1}, 2 x dim */ if (qmckl_context_check(context) == QMCKL_NULL_CONTEXT) { return qmckl_failwith(context, QMCKL_NULL_CONTEXT, "qmckl_woodbury_2x2_13", NULL); } const uint64_t row1 = (Updates_index[0] - 1); const uint64_t row2 = (Updates_index[1] - 1); // Compute C = (S^T)^{-1}U : 13 x 2 double __attribute__((aligned(8))) C[2 * 13]; for (uint64_t i = 0; i < 13; i++) { C[i * 2] = 0; C[i * 2 + 1] = 0; IVDEP ALIGNED for (uint64_t k = 0; k < D13_P; k++) { C[i * 2] += Slater_inv[i * D13_P + k] * Updates[k]; C[i * 2 + 1] += Slater_inv[i * D13_P + k] * Updates[D13_P + k]; } } // Compute B = 1 + VC : 2 x 2 const double B0 = C[row1 * 2] + 1; const double B1 = C[row1 * 2 + 1]; const double B2 = C[row2 * 2]; const double B3 = C[row2 * 2 + 1] + 1; // Check if determinant of inverted matrix is not zero double det = B0 * B3 - B1 * B2; if (fabs(det) < breakdown) { return QMCKL_FAILURE; } // Update det(S) when passed if (determinant) *determinant *= det; // Compute B^{-1} with explicit formula for 2 x 2 inversion double __attribute__((aligned(8))) Binv[4], idet = 1.0 / det; Binv[0] = idet * B3; Binv[1] = -1.0 * idet * B1; Binv[2] = -1.0 * idet * B2; Binv[3] = idet * B0; // tmp = B^{-1}D : 2 x D13_P double __attribute__((aligned(8))) tmp[2 * D13_P]; double* r1dim = &(Slater_inv[row1 * D13_P]); double* r2dim = &(Slater_inv[row2 * D13_P]); IVDEP ALIGNED for (uint64_t j = 0; j < D13_P; j++) { tmp[j] = Binv[0] * r1dim[j] + Binv[1] * r2dim[j]; tmp[D13_P + j] = Binv[2] * r1dim[j] + Binv[3] * r2dim[j]; } // Compute (S^T)^{-1} - C * tmp : 13 x D13_P for (uint64_t i = 0; i < 13; i++) { IVDEP ALIGNED for (uint64_t j = 0; j < D13_P; j++) { Slater_inv[i * D13_P + j] -= C[i * 2] * tmp[j]; Slater_inv[i * D13_P + j] -= C[i * 2 + 1] * tmp[D13_P + j]; } } return QMCKL_SUCCESS; } static inline qmckl_exit_code qmckl_woodbury_2x2_14( const qmckl_context context, const double* __restrict Updates, const uint64_t* __restrict Updates_index, const double breakdown, double* __restrict Slater_inv, double* __restrict determinant) { /* C := S^{-1} * U, dim x 2 B := 1 + V * C, 2 x 2 D := V * S^{-1}, 2 x dim */ if (qmckl_context_check(context) == QMCKL_NULL_CONTEXT) { return qmckl_failwith(context, QMCKL_NULL_CONTEXT, "qmckl_woodbury_2x2_14", NULL); } const uint64_t row1 = (Updates_index[0] - 1); const uint64_t row2 = (Updates_index[1] - 1); // Compute C = (S^T)^{-1}U : 14 x 2 double __attribute__((aligned(8))) C[2 * 14]; for (uint64_t i = 0; i < 14; i++) { C[i * 2] = 0; C[i * 2 + 1] = 0; IVDEP ALIGNED for (uint64_t k = 0; k < D14_P; k++) { C[i * 2] += Slater_inv[i * D14_P + k] * Updates[k]; C[i * 2 + 1] += Slater_inv[i * D14_P + k] * Updates[D14_P + k]; } } // Compute B = 1 + VC : 2 x 2 const double B0 = C[row1 * 2] + 1; const double B1 = C[row1 * 2 + 1]; const double B2 = C[row2 * 2]; const double B3 = C[row2 * 2 + 1] + 1; // Check if determinant of inverted matrix is not zero double det = B0 * B3 - B1 * B2; if (fabs(det) < breakdown) { return QMCKL_FAILURE; } // Update det(S) when passed if (determinant) *determinant *= det; // Compute B^{-1} with explicit formula for 2 x 2 inversion double __attribute__((aligned(8))) Binv[4], idet = 1.0 / det; Binv[0] = idet * B3; Binv[1] = -1.0 * idet * B1; Binv[2] = -1.0 * idet * B2; Binv[3] = idet * B0; // tmp = B^{-1}D : 2 x D14_P double __attribute__((aligned(8))) tmp[2 * D14_P]; double* r1dim = &(Slater_inv[row1 * D14_P]); double* r2dim = &(Slater_inv[row2 * D14_P]); IVDEP ALIGNED for (uint64_t j = 0; j < D14_P; j++) { tmp[j] = Binv[0] * r1dim[j] + Binv[1] * r2dim[j]; tmp[D14_P + j] = Binv[2] * r1dim[j] + Binv[3] * r2dim[j]; } // Compute (S^T)^{-1} - C * tmp : 14 x D14_P for (uint64_t i = 0; i < 14; i++) { IVDEP ALIGNED for (uint64_t j = 0; j < D14_P; j++) { Slater_inv[i * D14_P + j] -= C[i * 2] * tmp[j]; Slater_inv[i * D14_P + j] -= C[i * 2 + 1] * tmp[D14_P + j]; } } return QMCKL_SUCCESS; } static inline qmckl_exit_code qmckl_woodbury_2x2_15( const qmckl_context context, const double* __restrict Updates, const uint64_t* __restrict Updates_index, const double breakdown, double* __restrict Slater_inv, double* __restrict determinant) { /* C := S^{-1} * U, dim x 2 B := 1 + V * C, 2 x 2 D := V * S^{-1}, 2 x dim */ if (qmckl_context_check(context) == QMCKL_NULL_CONTEXT) { return qmckl_failwith(context, QMCKL_NULL_CONTEXT, "qmckl_woodbury_2x2_15", NULL); } const uint64_t row1 = (Updates_index[0] - 1); const uint64_t row2 = (Updates_index[1] - 1); // Compute C = (S^T)^{-1}U : 15 x 2 double __attribute__((aligned(8))) C[2 * 15]; for (uint64_t i = 0; i < 15; i++) { C[i * 2] = 0; C[i * 2 + 1] = 0; IVDEP ALIGNED for (uint64_t k = 0; k < D15_P; k++) { C[i * 2] += Slater_inv[i * D15_P + k] * Updates[k]; C[i * 2 + 1] += Slater_inv[i * D15_P + k] * Updates[D15_P + k]; } } // Compute B = 1 + VC : 2 x 2 const double B0 = C[row1 * 2] + 1; const double B1 = C[row1 * 2 + 1]; const double B2 = C[row2 * 2]; const double B3 = C[row2 * 2 + 1] + 1; // Check if determinant of inverted matrix is not zero double det = B0 * B3 - B1 * B2; if (fabs(det) < breakdown) { return QMCKL_FAILURE; } // Update det(S) when passed if (determinant) *determinant *= det; // Compute B^{-1} with explicit formula for 2 x 2 inversion double __attribute__((aligned(8))) Binv[4], idet = 1.0 / det; Binv[0] = idet * B3; Binv[1] = -1.0 * idet * B1; Binv[2] = -1.0 * idet * B2; Binv[3] = idet * B0; // tmp = B^{-1}D : 2 x D15_P double __attribute__((aligned(8))) tmp[2 * D15_P]; double* r1dim = &(Slater_inv[row1 * D15_P]); double* r2dim = &(Slater_inv[row2 * D15_P]); IVDEP ALIGNED for (uint64_t j = 0; j < D15_P; j++) { tmp[j] = Binv[0] * r1dim[j] + Binv[1] * r2dim[j]; tmp[D15_P + j] = Binv[2] * r1dim[j] + Binv[3] * r2dim[j]; } // Compute (S^T)^{-1} - C * tmp : 15 x D15_P for (uint64_t i = 0; i < 15; i++) { IVDEP ALIGNED for (uint64_t j = 0; j < D15_P; j++) { Slater_inv[i * D15_P + j] -= C[i * 2] * tmp[j]; Slater_inv[i * D15_P + j] -= C[i * 2 + 1] * tmp[D15_P + j]; } } return QMCKL_SUCCESS; } static inline qmckl_exit_code qmckl_woodbury_2x2_16( const qmckl_context context, const double* __restrict Updates, const uint64_t* __restrict Updates_index, const double breakdown, double* __restrict Slater_inv, double* __restrict determinant) { /* C := S^{-1} * U, dim x 2 B := 1 + V * C, 2 x 2 D := V * S^{-1}, 2 x dim */ if (qmckl_context_check(context) == QMCKL_NULL_CONTEXT) { return qmckl_failwith(context, QMCKL_NULL_CONTEXT, "qmckl_woodbury_2x2_16", NULL); } const uint64_t row1 = (Updates_index[0] - 1); const uint64_t row2 = (Updates_index[1] - 1); // Compute C = (S^T)^{-1}U : 16 x 2 double __attribute__((aligned(8))) C[2 * 16]; for (uint64_t i = 0; i < 16; i++) { C[i * 2] = 0; C[i * 2 + 1] = 0; IVDEP ALIGNED for (uint64_t k = 0; k < D16_P; k++) { C[i * 2] += Slater_inv[i * D16_P + k] * Updates[k]; C[i * 2 + 1] += Slater_inv[i * D16_P + k] * Updates[D16_P + k]; } } // Compute B = 1 + VC : 2 x 2 const double B0 = C[row1 * 2] + 1; const double B1 = C[row1 * 2 + 1]; const double B2 = C[row2 * 2]; const double B3 = C[row2 * 2 + 1] + 1; // Check if determinant of inverted matrix is not zero double det = B0 * B3 - B1 * B2; if (fabs(det) < breakdown) { return QMCKL_FAILURE; } // Update det(S) when passed if (determinant) *determinant *= det; // Compute B^{-1} with explicit formula for 2 x 2 inversion double __attribute__((aligned(8))) Binv[4], idet = 1.0 / det; Binv[0] = idet * B3; Binv[1] = -1.0 * idet * B1; Binv[2] = -1.0 * idet * B2; Binv[3] = idet * B0; // tmp = B^{-1}D : 2 x D16_P double __attribute__((aligned(8))) tmp[2 * D16_P]; double* r1dim = &(Slater_inv[row1 * D16_P]); double* r2dim = &(Slater_inv[row2 * D16_P]); IVDEP ALIGNED for (uint64_t j = 0; j < D16_P; j++) { tmp[j] = Binv[0] * r1dim[j] + Binv[1] * r2dim[j]; tmp[D16_P + j] = Binv[2] * r1dim[j] + Binv[3] * r2dim[j]; } // Compute (S^T)^{-1} - C * tmp : 16 x D16_P for (uint64_t i = 0; i < 16; i++) { IVDEP ALIGNED for (uint64_t j = 0; j < D16_P; j++) { Slater_inv[i * D16_P + j] -= C[i * 2] * tmp[j]; Slater_inv[i * D16_P + j] -= C[i * 2 + 1] * tmp[D16_P + j]; } } return QMCKL_SUCCESS; } static inline qmckl_exit_code qmckl_woodbury_2x2_17( const qmckl_context context, const double* __restrict Updates, const uint64_t* __restrict Updates_index, const double breakdown, double* __restrict Slater_inv, double* __restrict determinant) { /* C := S^{-1} * U, dim x 2 B := 1 + V * C, 2 x 2 D := V * S^{-1}, 2 x dim */ if (qmckl_context_check(context) == QMCKL_NULL_CONTEXT) { return qmckl_failwith(context, QMCKL_NULL_CONTEXT, "qmckl_woodbury_2x2_17", NULL); } const uint64_t row1 = (Updates_index[0] - 1); const uint64_t row2 = (Updates_index[1] - 1); // Compute C = (S^T)^{-1}U : 17 x 2 double __attribute__((aligned(8))) C[2 * 17]; for (uint64_t i = 0; i < 17; i++) { C[i * 2] = 0; C[i * 2 + 1] = 0; IVDEP ALIGNED for (uint64_t k = 0; k < D17_P; k++) { C[i * 2] += Slater_inv[i * D17_P + k] * Updates[k]; C[i * 2 + 1] += Slater_inv[i * D17_P + k] * Updates[D17_P + k]; } } // Compute B = 1 + VC : 2 x 2 const double B0 = C[row1 * 2] + 1; const double B1 = C[row1 * 2 + 1]; const double B2 = C[row2 * 2]; const double B3 = C[row2 * 2 + 1] + 1; // Check if determinant of inverted matrix is not zero double det = B0 * B3 - B1 * B2; if (fabs(det) < breakdown) { return QMCKL_FAILURE; } // Update det(S) when passed if (determinant) *determinant *= det; // Compute B^{-1} with explicit formula for 2 x 2 inversion double __attribute__((aligned(8))) Binv[4], idet = 1.0 / det; Binv[0] = idet * B3; Binv[1] = -1.0 * idet * B1; Binv[2] = -1.0 * idet * B2; Binv[3] = idet * B0; // tmp = B^{-1}D : 2 x D17_P double __attribute__((aligned(8))) tmp[2 * D17_P]; double* r1dim = &(Slater_inv[row1 * D17_P]); double* r2dim = &(Slater_inv[row2 * D17_P]); IVDEP ALIGNED for (uint64_t j = 0; j < D17_P; j++) { tmp[j] = Binv[0] * r1dim[j] + Binv[1] * r2dim[j]; tmp[D17_P + j] = Binv[2] * r1dim[j] + Binv[3] * r2dim[j]; } // Compute (S^T)^{-1} - C * tmp : 17 x D17_P for (uint64_t i = 0; i < 17; i++) { IVDEP ALIGNED for (uint64_t j = 0; j < D17_P; j++) { Slater_inv[i * D17_P + j] -= C[i * 2] * tmp[j]; Slater_inv[i * D17_P + j] -= C[i * 2 + 1] * tmp[D17_P + j]; } } return QMCKL_SUCCESS; } static inline qmckl_exit_code qmckl_woodbury_2x2_18( const qmckl_context context, const double* __restrict Updates, const uint64_t* __restrict Updates_index, const double breakdown, double* __restrict Slater_inv, double* __restrict determinant) { /* C := S^{-1} * U, dim x 2 B := 1 + V * C, 2 x 2 D := V * S^{-1}, 2 x dim */ if (qmckl_context_check(context) == QMCKL_NULL_CONTEXT) { return qmckl_failwith(context, QMCKL_NULL_CONTEXT, "qmckl_woodbury_2x2_18", NULL); } const uint64_t row1 = (Updates_index[0] - 1); const uint64_t row2 = (Updates_index[1] - 1); // Compute C = (S^T)^{-1}U : 18 x 2 double __attribute__((aligned(8))) C[2 * 18]; for (uint64_t i = 0; i < 18; i++) { C[i * 2] = 0; C[i * 2 + 1] = 0; IVDEP ALIGNED for (uint64_t k = 0; k < D18_P; k++) { C[i * 2] += Slater_inv[i * D18_P + k] * Updates[k]; C[i * 2 + 1] += Slater_inv[i * D18_P + k] * Updates[D18_P + k]; } } // Compute B = 1 + VC : 2 x 2 const double B0 = C[row1 * 2] + 1; const double B1 = C[row1 * 2 + 1]; const double B2 = C[row2 * 2]; const double B3 = C[row2 * 2 + 1] + 1; // Check if determinant of inverted matrix is not zero double det = B0 * B3 - B1 * B2; if (fabs(det) < breakdown) { return QMCKL_FAILURE; } // Update det(S) when passed if (determinant) *determinant *= det; // Compute B^{-1} with explicit formula for 2 x 2 inversion double __attribute__((aligned(8))) Binv[4], idet = 1.0 / det; Binv[0] = idet * B3; Binv[1] = -1.0 * idet * B1; Binv[2] = -1.0 * idet * B2; Binv[3] = idet * B0; // tmp = B^{-1}D : 2 x D18_P double __attribute__((aligned(8))) tmp[2 * D18_P]; double* r1dim = &(Slater_inv[row1 * D18_P]); double* r2dim = &(Slater_inv[row2 * D18_P]); IVDEP ALIGNED for (uint64_t j = 0; j < D18_P; j++) { tmp[j] = Binv[0] * r1dim[j] + Binv[1] * r2dim[j]; tmp[D18_P + j] = Binv[2] * r1dim[j] + Binv[3] * r2dim[j]; } // Compute (S^T)^{-1} - C * tmp : 18 x D18_P for (uint64_t i = 0; i < 18; i++) { IVDEP ALIGNED for (uint64_t j = 0; j < D18_P; j++) { Slater_inv[i * D18_P + j] -= C[i * 2] * tmp[j]; Slater_inv[i * D18_P + j] -= C[i * 2 + 1] * tmp[D18_P + j]; } } return QMCKL_SUCCESS; } static inline qmckl_exit_code qmckl_woodbury_2x2_19( const qmckl_context context, const double* __restrict Updates, const uint64_t* __restrict Updates_index, const double breakdown, double* __restrict Slater_inv, double* __restrict determinant) { /* C := S^{-1} * U, dim x 2 B := 1 + V * C, 2 x 2 D := V * S^{-1}, 2 x dim */ if (qmckl_context_check(context) == QMCKL_NULL_CONTEXT) { return qmckl_failwith(context, QMCKL_NULL_CONTEXT, "qmckl_woodbury_2x2_19", NULL); } const uint64_t row1 = (Updates_index[0] - 1); const uint64_t row2 = (Updates_index[1] - 1); // Compute C = (S^T)^{-1}U : 19 x 2 double __attribute__((aligned(8))) C[2 * 19]; for (uint64_t i = 0; i < 19; i++) { C[i * 2] = 0; C[i * 2 + 1] = 0; IVDEP ALIGNED for (uint64_t k = 0; k < D19_P; k++) { C[i * 2] += Slater_inv[i * D19_P + k] * Updates[k]; C[i * 2 + 1] += Slater_inv[i * D19_P + k] * Updates[D19_P + k]; } } // Compute B = 1 + VC : 2 x 2 const double B0 = C[row1 * 2] + 1; const double B1 = C[row1 * 2 + 1]; const double B2 = C[row2 * 2]; const double B3 = C[row2 * 2 + 1] + 1; // Check if determinant of inverted matrix is not zero double det = B0 * B3 - B1 * B2; if (fabs(det) < breakdown) { return QMCKL_FAILURE; } // Update det(S) when passed if (determinant) *determinant *= det; // Compute B^{-1} with explicit formula for 2 x 2 inversion double __attribute__((aligned(8))) Binv[4], idet = 1.0 / det; Binv[0] = idet * B3; Binv[1] = -1.0 * idet * B1; Binv[2] = -1.0 * idet * B2; Binv[3] = idet * B0; // tmp = B^{-1}D : 2 x D19_P double __attribute__((aligned(8))) tmp[2 * D19_P]; double* r1dim = &(Slater_inv[row1 * D19_P]); double* r2dim = &(Slater_inv[row2 * D19_P]); IVDEP ALIGNED for (uint64_t j = 0; j < D19_P; j++) { tmp[j] = Binv[0] * r1dim[j] + Binv[1] * r2dim[j]; tmp[D19_P + j] = Binv[2] * r1dim[j] + Binv[3] * r2dim[j]; } // Compute (S^T)^{-1} - C * tmp : 19 x D19_P for (uint64_t i = 0; i < 19; i++) { IVDEP ALIGNED for (uint64_t j = 0; j < D19_P; j++) { Slater_inv[i * D19_P + j] -= C[i * 2] * tmp[j]; Slater_inv[i * D19_P + j] -= C[i * 2 + 1] * tmp[D19_P + j]; } } return QMCKL_SUCCESS; } static inline qmckl_exit_code qmckl_woodbury_2x2_20( const qmckl_context context, const double* __restrict Updates, const uint64_t* __restrict Updates_index, const double breakdown, double* __restrict Slater_inv, double* __restrict determinant) { /* C := S^{-1} * U, dim x 2 B := 1 + V * C, 2 x 2 D := V * S^{-1}, 2 x dim */ if (qmckl_context_check(context) == QMCKL_NULL_CONTEXT) { return qmckl_failwith(context, QMCKL_NULL_CONTEXT, "qmckl_woodbury_2x2_20", NULL); } const uint64_t row1 = (Updates_index[0] - 1); const uint64_t row2 = (Updates_index[1] - 1); // Compute C = (S^T)^{-1}U : 20 x 2 double __attribute__((aligned(8))) C[2 * 20]; for (uint64_t i = 0; i < 20; i++) { C[i * 2] = 0; C[i * 2 + 1] = 0; IVDEP ALIGNED for (uint64_t k = 0; k < D20_P; k++) { C[i * 2] += Slater_inv[i * D20_P + k] * Updates[k]; C[i * 2 + 1] += Slater_inv[i * D20_P + k] * Updates[D20_P + k]; } } // Compute B = 1 + VC : 2 x 2 const double B0 = C[row1 * 2] + 1; const double B1 = C[row1 * 2 + 1]; const double B2 = C[row2 * 2]; const double B3 = C[row2 * 2 + 1] + 1; // Check if determinant of inverted matrix is not zero double det = B0 * B3 - B1 * B2; if (fabs(det) < breakdown) { return QMCKL_FAILURE; } // Update det(S) when passed if (determinant) *determinant *= det; // Compute B^{-1} with explicit formula for 2 x 2 inversion double __attribute__((aligned(8))) Binv[4], idet = 1.0 / det; Binv[0] = idet * B3; Binv[1] = -1.0 * idet * B1; Binv[2] = -1.0 * idet * B2; Binv[3] = idet * B0; // tmp = B^{-1}D : 2 x D20_P double __attribute__((aligned(8))) tmp[2 * D20_P]; double* r1dim = &(Slater_inv[row1 * D20_P]); double* r2dim = &(Slater_inv[row2 * D20_P]); IVDEP ALIGNED for (uint64_t j = 0; j < D20_P; j++) { tmp[j] = Binv[0] * r1dim[j] + Binv[1] * r2dim[j]; tmp[D20_P + j] = Binv[2] * r1dim[j] + Binv[3] * r2dim[j]; } // Compute (S^T)^{-1} - C * tmp : 20 x D20_P for (uint64_t i = 0; i < 20; i++) { IVDEP ALIGNED for (uint64_t j = 0; j < D20_P; j++) { Slater_inv[i * D20_P + j] -= C[i * 2] * tmp[j]; Slater_inv[i * D20_P + j] -= C[i * 2 + 1] * tmp[D20_P + j]; } } return QMCKL_SUCCESS; } static inline qmckl_exit_code qmckl_woodbury_2x2_21( const qmckl_context context, const double* __restrict Updates, const uint64_t* __restrict Updates_index, const double breakdown, double* __restrict Slater_inv, double* __restrict determinant) { /* C := S^{-1} * U, dim x 2 B := 1 + V * C, 2 x 2 D := V * S^{-1}, 2 x dim */ if (qmckl_context_check(context) == QMCKL_NULL_CONTEXT) { return qmckl_failwith(context, QMCKL_NULL_CONTEXT, "qmckl_woodbury_2x2_21", NULL); } const uint64_t row1 = (Updates_index[0] - 1); const uint64_t row2 = (Updates_index[1] - 1); // Compute C = (S^T)^{-1}U : 21 x 2 double __attribute__((aligned(8))) C[2 * 21]; for (uint64_t i = 0; i < 21; i++) { C[i * 2] = 0; C[i * 2 + 1] = 0; IVDEP ALIGNED for (uint64_t k = 0; k < D21_P; k++) { C[i * 2] += Slater_inv[i * D21_P + k] * Updates[k]; C[i * 2 + 1] += Slater_inv[i * D21_P + k] * Updates[D21_P + k]; } } // Compute B = 1 + VC : 2 x 2 const double B0 = C[row1 * 2] + 1; const double B1 = C[row1 * 2 + 1]; const double B2 = C[row2 * 2]; const double B3 = C[row2 * 2 + 1] + 1; // Check if determinant of inverted matrix is not zero double det = B0 * B3 - B1 * B2; if (fabs(det) < breakdown) { return QMCKL_FAILURE; } // Update det(S) when passed if (determinant) *determinant *= det; // Compute B^{-1} with explicit formula for 2 x 2 inversion double __attribute__((aligned(8))) Binv[4], idet = 1.0 / det; Binv[0] = idet * B3; Binv[1] = -1.0 * idet * B1; Binv[2] = -1.0 * idet * B2; Binv[3] = idet * B0; // tmp = B^{-1}D : 2 x D21_P double __attribute__((aligned(8))) tmp[2 * D21_P]; double* r1dim = &(Slater_inv[row1 * D21_P]); double* r2dim = &(Slater_inv[row2 * D21_P]); IVDEP ALIGNED for (uint64_t j = 0; j < D21_P; j++) { tmp[j] = Binv[0] * r1dim[j] + Binv[1] * r2dim[j]; tmp[D21_P + j] = Binv[2] * r1dim[j] + Binv[3] * r2dim[j]; } // Compute (S^T)^{-1} - C * tmp : 21 x D21_P for (uint64_t i = 0; i < 21; i++) { IVDEP ALIGNED for (uint64_t j = 0; j < D21_P; j++) { Slater_inv[i * D21_P + j] -= C[i * 2] * tmp[j]; Slater_inv[i * D21_P + j] -= C[i * 2 + 1] * tmp[D21_P + j]; } } return QMCKL_SUCCESS; } qmckl_exit_code qmckl_woodbury_2x2(const qmckl_context context, const uint64_t LDS, const uint64_t Dim, const double* Updates, const uint64_t* Updates_index, const double breakdown, double* Slater_inv, double* determinant) { if (qmckl_context_check(context) == QMCKL_NULL_CONTEXT) { return qmckl_failwith(context, QMCKL_NULL_CONTEXT, "qmckl_woodbury_2x2", NULL); } if (LDS == (1+(Dim-1)/SIMD_LENGTH)*SIMD_LENGTH) { // Most cases switch (Dim) { case 2: return qmckl_woodbury_2x2_2(context, Updates, Updates_index, breakdown, Slater_inv, determinant); case 3: return qmckl_woodbury_2x2_3(context, Updates, Updates_index, breakdown, Slater_inv, determinant); case 4: return qmckl_woodbury_2x2_4(context, Updates, Updates_index, breakdown, Slater_inv, determinant); case 5: return qmckl_woodbury_2x2_5(context, Updates, Updates_index, breakdown, Slater_inv, determinant); case 6: return qmckl_woodbury_2x2_6(context, Updates, Updates_index, breakdown, Slater_inv, determinant); case 7: return qmckl_woodbury_2x2_7(context, Updates, Updates_index, breakdown, Slater_inv, determinant); case 8: return qmckl_woodbury_2x2_8(context, Updates, Updates_index, breakdown, Slater_inv, determinant); case 9: return qmckl_woodbury_2x2_9(context, Updates, Updates_index, breakdown, Slater_inv, determinant); case 10: return qmckl_woodbury_2x2_10(context, Updates, Updates_index, breakdown, Slater_inv, determinant); case 11: return qmckl_woodbury_2x2_11(context, Updates, Updates_index, breakdown, Slater_inv, determinant); case 12: return qmckl_woodbury_2x2_12(context, Updates, Updates_index, breakdown, Slater_inv, determinant); case 13: return qmckl_woodbury_2x2_13(context, Updates, Updates_index, breakdown, Slater_inv, determinant); case 14: return qmckl_woodbury_2x2_14(context, Updates, Updates_index, breakdown, Slater_inv, determinant); case 15: return qmckl_woodbury_2x2_15(context, Updates, Updates_index, breakdown, Slater_inv, determinant); case 16: return qmckl_woodbury_2x2_16(context, Updates, Updates_index, breakdown, Slater_inv, determinant); case 17: return qmckl_woodbury_2x2_17(context, Updates, Updates_index, breakdown, Slater_inv, determinant); case 18: return qmckl_woodbury_2x2_18(context, Updates, Updates_index, breakdown, Slater_inv, determinant); case 19: return qmckl_woodbury_2x2_19(context, Updates, Updates_index, breakdown, Slater_inv, determinant); case 20: return qmckl_woodbury_2x2_20(context, Updates, Updates_index, breakdown, Slater_inv, determinant); case 21: return qmckl_woodbury_2x2_21(context, Updates, Updates_index, breakdown, Slater_inv, determinant); } } else { // When SIMD_LENGTH > 1, called with LDS == Dim AND Dim != (1+(Dim-1)/SIMD_LENGTH)*SIMD_LENGTH) return qmckl_woodbury_2x2_hpc(context, LDS, Dim, Updates, Updates_index, breakdown, Slater_inv, determinant); } return QMCKL_FAILURE; }
3.1.4 Performance
This function is most efficient when used in cases where there are only 2 rank-1 updates and it is sure they will not result in a singular matrix.
4 Woodbury 3x3
4.1 qmckl_woodbury_3x3
The 3x3 version of the Woodbury 2x2 kernel. It is used to apply three rank-1 updates at once. The formula used in this kernel is the same as for Woodbury 2x2, except for the sizes of the following matrices:
\(C:= S^{-1}U\), a Dim \(\times 3\) matrix \(B := 1 + VC\), the \(3 \times 3\) matrix that is going to be inverted \(D := VS^{-1}\), a \(3 \times Dim\) matrix
If the determinant of the Slater-matrix is passed, it will be updated to the determinant resulting from applying the updates to the original matrix. #pragma ivdep #pragma vector aligned
qmcklcontext | context | in | Global state |
uint64t | LDS | in | Leading dimension of Slaterinv |
uint64t | Dim | in | Dimension of Slaterinv |
double | Updates[3*Dim] | in | Array containing the updates |
uint64t | Updatesindex[3] | in | Array containing the rank-1 updates |
double | breakdown | in | Break-down parameter on which to fail or not |
double | Slaterinv[LDS*Dim] | inout | Array containing the inverse of a Slater-matrix |
double* | determinant | inout | Determinant of Slater-matrix |
4.1.1 Requirements
context
is notqmckl_null_context
LDS >= 2
Dim >= 2
Updates
is allocated with \(3 \times Dim\) elementsUpdates_index
is allocated with \(3\) elementsbreakdown
is a small number such that \(0 < breakdown << 1\)Slater_inv
is allocated with \(Dim \times Dim\) elements
4.1.2 C header
qmckl_exit_code qmckl_woodbury_3x3 ( const context qmckl_context, const LDS uint64_t, const Dim uint64_t, const Updates* double, const Updates_index* uint64_t, const breakdown double, Slater_inv* double, determinant* double* );
4.1.3 C source
qmckl_exit_code qmckl_woodbury_3x3_hpc(const qmckl_context context, const uint64_t LDS, const uint64_t Dim, const double* __restrict Updates, const uint64_t* __restrict Updates_index, const double breakdown, double* __restrict Slater_inv, double* __restrict determinant) { /* C := S^{-1} * U, dim x 3 B := 1 + V * C, 3 x 3 D := V * S^{-1}, 3 x dim */ if (qmckl_context_check(context) == QMCKL_NULL_CONTEXT) { return qmckl_failwith(context, QMCKL_NULL_CONTEXT, "qmckl_woodbury_3x3_hpc", NULL); } const uint64_t row1 = (Updates_index[0] - 1); const uint64_t row2 = (Updates_index[1] - 1); const uint64_t row3 = (Updates_index[2] - 1); // Compute C = (S^T)^{-1}U : Dim x 3 double __attribute__((aligned(8))) C[3 * Dim]; for (uint64_t i = 0; i < Dim; i++) { C[i * 3] = 0; C[i * 3 + 1] = 0; C[i * 3 + 2] = 0; IVDEP ALIGNED for (uint64_t k = 0; k < LDS; k++) { C[i * 3] += Slater_inv[i * LDS + k] * Updates[k]; C[i * 3 + 1] += Slater_inv[i * LDS + k] * Updates[LDS + k]; C[i * 3 + 2] += Slater_inv[i * LDS + k] * Updates[2 * LDS + k]; } } // Compute B = 1 + VC : 3 x 3 const double B0 = C[row1 * 3] + 1; const double B1 = C[row1 * 3 + 1]; const double B2 = C[row1 * 3 + 2]; const double B3 = C[row2 * 3]; const double B4 = C[row2 * 3 + 1] + 1; const double B5 = C[row2 * 3 + 2]; const double B6 = C[row3 * 3]; const double B7 = C[row3 * 3 + 1]; const double B8 = C[row3 * 3 + 2] + 1; // Check if determinant of B is not too close to zero double det; det = B0 * (B4 * B8 - B5 * B7) - B1 * (B3 * B8 - B5 * B6) + B2 * (B3 * B7 - B4 * B6); if (fabs(det) < breakdown) { return QMCKL_FAILURE; } // Update det(Slater) if passed if (determinant) *determinant *= det; // Compute B^{-1} with explicit formula for 3 x 3 inversion double __attribute__((aligned(8))) Binv[9], idet = 1.0 / det; Binv[0] = (B4 * B8 - B7 * B5) * idet; Binv[1] = -(B1 * B8 - B7 * B2) * idet; Binv[2] = (B1 * B5 - B4 * B2) * idet; Binv[3] = -(B3 * B8 - B6 * B5) * idet; Binv[4] = (B0 * B8 - B6 * B2) * idet; Binv[5] = -(B0 * B5 - B3 * B2) * idet; Binv[6] = (B3 * B7 - B6 * B4) * idet; Binv[7] = -(B0 * B7 - B6 * B1) * idet; Binv[8] = (B0 * B4 - B3 * B1) * idet; // tmp = B^{-1}D : 3 x LDS double __attribute__((aligned(8))) tmp[3 * LDS]; double* r1dim = &(Slater_inv[row1 * LDS]); double* r2dim = &(Slater_inv[row2 * LDS]); double* r3dim = &(Slater_inv[row3 * LDS]); IVDEP ALIGNED for (uint64_t j = 0; j < LDS; j++) { tmp[j] = Binv[0] * r1dim[j] + Binv[1] * r2dim[j] + Binv[2] * r3dim[j]; tmp[LDS + j] = Binv[3] * r1dim[j] + Binv[4] * r2dim[j] + Binv[5] * r3dim[j]; tmp[2 * LDS + j] = Binv[6] * r1dim[j] + Binv[7] * r2dim[j] + Binv[8] * r3dim[j]; } // Compute (S^T)^{-1} - C * tmp : Dim x LDS for (uint64_t i = 0; i < Dim; i++) { IVDEP ALIGNED for (uint64_t j = 0; j < LDS; j++) { Slater_inv[i * LDS + j] -= C[i * 3] * tmp[j]; Slater_inv[i * LDS + j] -= C[i * 3 + 1] * tmp[LDS + j]; Slater_inv[i * LDS + j] -= C[i * 3 + 2] * tmp[2 * LDS + j]; } } return QMCKL_SUCCESS; }
qmckl_exit_code qmckl_woodbury_3x3_{Dim}( const qmckl_context context, const double* __restrict Updates, const uint64_t* __restrict Updates_index, const double breakdown, double* __restrict Slater_inv, double* __restrict determinant) { /* C := S^{-1} * U, dim x 3 B := 1 + V * C, 3 x 3 D := V * S^{-1}, 3 x dim */ if (qmckl_context_check(context) == QMCKL_NULL_CONTEXT) { return qmckl_failwith(context, QMCKL_NULL_CONTEXT, "qmckl_woodbury_3x3_{Dim}", NULL); } const uint64_t row1 = (Updates_index[0] - 1); const uint64_t row2 = (Updates_index[1] - 1); const uint64_t row3 = (Updates_index[2] - 1); // Compute C = (S^T)^{-1}U : {Dim} x 3 double __attribute__((aligned(8))) C[3 * {Dim}]; for (uint64_t i = 0; i < {Dim}; i++) { C[i * 3] = 0; C[i * 3 + 1] = 0; C[i * 3 + 2] = 0; IVDEP ALIGNED for (uint64_t k = 0; k < D{Dim}_P; k++) { C[i * 3] += Slater_inv[i * D{Dim}_P + k] * Updates[k]; C[i * 3 + 1] += Slater_inv[i * D{Dim}_P + k] * Updates[D{Dim}_P + k]; C[i * 3 + 2] += Slater_inv[i * D{Dim}_P + k] * Updates[2 * D{Dim}_P + k]; } } // Compute B = 1 + VC : 3 x 3 const double B0 = C[row1 * 3] + 1; const double B1 = C[row1 * 3 + 1]; const double B2 = C[row1 * 3 + 2]; const double B3 = C[row2 * 3]; const double B4 = C[row2 * 3 + 1] + 1; const double B5 = C[row2 * 3 + 2]; const double B6 = C[row3 * 3]; const double B7 = C[row3 * 3 + 1]; const double B8 = C[row3 * 3 + 2] + 1; // Check if determinant of B is not too close to zero double det; det = B0 * (B4 * B8 - B5 * B7) - B1 * (B3 * B8 - B5 * B6) + B2 * (B3 * B7 - B4 * B6); if (fabs(det) < breakdown) { return QMCKL_FAILURE; } // Update det(Slater) if passed if (determinant) *determinant *= det; // Compute B^{-1} with explicit formula for 3 x 3 inversion double __attribute__((aligned(8))) Binv[9], idet = 1.0 / det; Binv[0] = (B4 * B8 - B7 * B5) * idet; Binv[1] = -(B1 * B8 - B7 * B2) * idet; Binv[2] = (B1 * B5 - B4 * B2) * idet; Binv[3] = -(B3 * B8 - B6 * B5) * idet; Binv[4] = (B0 * B8 - B6 * B2) * idet; Binv[5] = -(B0 * B5 - B3 * B2) * idet; Binv[6] = (B3 * B7 - B6 * B4) * idet; Binv[7] = -(B0 * B7 - B6 * B1) * idet; Binv[8] = (B0 * B4 - B3 * B1) * idet; // tmp = B^{-1}D : 3 x D{Dim}_P double __attribute__((aligned(8))) tmp[3 * D{Dim}_P]; double* r1dim = &(Slater_inv[row1 * D{Dim}_P]); double* r2dim = &(Slater_inv[row2 * D{Dim}_P]); double* r3dim = &(Slater_inv[row3 * D{Dim}_P]); IVDEP ALIGNED for (uint64_t j = 0; j < D{Dim}_P; j++) { tmp[j] = Binv[0] * r1dim[j] + Binv[1] * r2dim[j] + Binv[2] * r3dim[j]; tmp[D{Dim}_P + j] = Binv[3] * r1dim[j] + Binv[4] * r2dim[j] + Binv[5] * r3dim[j]; tmp[2 * D{Dim}_P + j] = Binv[6] * r1dim[j] + Binv[7] * r2dim[j] + Binv[8] * r3dim[j]; } // Compute (S^T)^{-1} - C * tmp : {Dim} x D{Dim}_P for (uint64_t i = 0; i < {Dim}; i++) { IVDEP ALIGNED for (uint64_t j = 0; j < D{Dim}_P; j++) { Slater_inv[i * D{Dim}_P + j] -= C[i * 3] * tmp[j]; Slater_inv[i * D{Dim}_P + j] -= C[i * 3 + 1] * tmp[D{Dim}_P + j]; Slater_inv[i * D{Dim}_P + j] -= C[i * 3 + 2] * tmp[2 * D{Dim}_P + j]; } } return QMCKL_SUCCESS; }
qmckl_exit_code qmckl_woodbury_3x3_2( const qmckl_context context, const double* __restrict Updates, const uint64_t* __restrict Updates_index, const double breakdown, double* __restrict Slater_inv, double* __restrict determinant) { /* C := S^{-1} * U, dim x 3 B := 1 + V * C, 3 x 3 D := V * S^{-1}, 3 x dim */ if (qmckl_context_check(context) == QMCKL_NULL_CONTEXT) { return qmckl_failwith(context, QMCKL_NULL_CONTEXT, "qmckl_woodbury_3x3_2", NULL); } const uint64_t row1 = (Updates_index[0] - 1); const uint64_t row2 = (Updates_index[1] - 1); const uint64_t row3 = (Updates_index[2] - 1); // Compute C = (S^T)^{-1}U : 2 x 3 double __attribute__((aligned(8))) C[3 * 2]; for (uint64_t i = 0; i < 2; i++) { C[i * 3] = 0; C[i * 3 + 1] = 0; C[i * 3 + 2] = 0; IVDEP ALIGNED for (uint64_t k = 0; k < D2_P; k++) { C[i * 3] += Slater_inv[i * D2_P + k] * Updates[k]; C[i * 3 + 1] += Slater_inv[i * D2_P + k] * Updates[D2_P + k]; C[i * 3 + 2] += Slater_inv[i * D2_P + k] * Updates[2 * D2_P + k]; } } // Compute B = 1 + VC : 3 x 3 const double B0 = C[row1 * 3] + 1; const double B1 = C[row1 * 3 + 1]; const double B2 = C[row1 * 3 + 2]; const double B3 = C[row2 * 3]; const double B4 = C[row2 * 3 + 1] + 1; const double B5 = C[row2 * 3 + 2]; const double B6 = C[row3 * 3]; const double B7 = C[row3 * 3 + 1]; const double B8 = C[row3 * 3 + 2] + 1; // Check if determinant of B is not too close to zero double det; det = B0 * (B4 * B8 - B5 * B7) - B1 * (B3 * B8 - B5 * B6) + B2 * (B3 * B7 - B4 * B6); if (fabs(det) < breakdown) { return QMCKL_FAILURE; } // Update det(Slater) if passed if (determinant) *determinant *= det; // Compute B^{-1} with explicit formula for 3 x 3 inversion double __attribute__((aligned(8))) Binv[9], idet = 1.0 / det; Binv[0] = (B4 * B8 - B7 * B5) * idet; Binv[1] = -(B1 * B8 - B7 * B2) * idet; Binv[2] = (B1 * B5 - B4 * B2) * idet; Binv[3] = -(B3 * B8 - B6 * B5) * idet; Binv[4] = (B0 * B8 - B6 * B2) * idet; Binv[5] = -(B0 * B5 - B3 * B2) * idet; Binv[6] = (B3 * B7 - B6 * B4) * idet; Binv[7] = -(B0 * B7 - B6 * B1) * idet; Binv[8] = (B0 * B4 - B3 * B1) * idet; // tmp = B^{-1}D : 3 x D2_P double __attribute__((aligned(8))) tmp[3 * D2_P]; double* r1dim = &(Slater_inv[row1 * D2_P]); double* r2dim = &(Slater_inv[row2 * D2_P]); double* r3dim = &(Slater_inv[row3 * D2_P]); IVDEP ALIGNED for (uint64_t j = 0; j < D2_P; j++) { tmp[j] = Binv[0] * r1dim[j] + Binv[1] * r2dim[j] + Binv[2] * r3dim[j]; tmp[D2_P + j] = Binv[3] * r1dim[j] + Binv[4] * r2dim[j] + Binv[5] * r3dim[j]; tmp[2 * D2_P + j] = Binv[6] * r1dim[j] + Binv[7] * r2dim[j] + Binv[8] * r3dim[j]; } // Compute (S^T)^{-1} - C * tmp : 2 x D2_P for (uint64_t i = 0; i < 2; i++) { IVDEP ALIGNED for (uint64_t j = 0; j < D2_P; j++) { Slater_inv[i * D2_P + j] -= C[i * 3] * tmp[j]; Slater_inv[i * D2_P + j] -= C[i * 3 + 1] * tmp[D2_P + j]; Slater_inv[i * D2_P + j] -= C[i * 3 + 2] * tmp[2 * D2_P + j]; } } return QMCKL_SUCCESS; } qmckl_exit_code qmckl_woodbury_3x3_3( const qmckl_context context, const double* __restrict Updates, const uint64_t* __restrict Updates_index, const double breakdown, double* __restrict Slater_inv, double* __restrict determinant) { /* C := S^{-1} * U, dim x 3 B := 1 + V * C, 3 x 3 D := V * S^{-1}, 3 x dim */ if (qmckl_context_check(context) == QMCKL_NULL_CONTEXT) { return qmckl_failwith(context, QMCKL_NULL_CONTEXT, "qmckl_woodbury_3x3_3", NULL); } const uint64_t row1 = (Updates_index[0] - 1); const uint64_t row2 = (Updates_index[1] - 1); const uint64_t row3 = (Updates_index[2] - 1); // Compute C = (S^T)^{-1}U : 3 x 3 double __attribute__((aligned(8))) C[3 * 3]; for (uint64_t i = 0; i < 3; i++) { C[i * 3] = 0; C[i * 3 + 1] = 0; C[i * 3 + 2] = 0; IVDEP ALIGNED for (uint64_t k = 0; k < D3_P; k++) { C[i * 3] += Slater_inv[i * D3_P + k] * Updates[k]; C[i * 3 + 1] += Slater_inv[i * D3_P + k] * Updates[D3_P + k]; C[i * 3 + 2] += Slater_inv[i * D3_P + k] * Updates[2 * D3_P + k]; } } // Compute B = 1 + VC : 3 x 3 const double B0 = C[row1 * 3] + 1; const double B1 = C[row1 * 3 + 1]; const double B2 = C[row1 * 3 + 2]; const double B3 = C[row2 * 3]; const double B4 = C[row2 * 3 + 1] + 1; const double B5 = C[row2 * 3 + 2]; const double B6 = C[row3 * 3]; const double B7 = C[row3 * 3 + 1]; const double B8 = C[row3 * 3 + 2] + 1; // Check if determinant of B is not too close to zero double det; det = B0 * (B4 * B8 - B5 * B7) - B1 * (B3 * B8 - B5 * B6) + B2 * (B3 * B7 - B4 * B6); if (fabs(det) < breakdown) { return QMCKL_FAILURE; } // Update det(Slater) if passed if (determinant) *determinant *= det; // Compute B^{-1} with explicit formula for 3 x 3 inversion double __attribute__((aligned(8))) Binv[9], idet = 1.0 / det; Binv[0] = (B4 * B8 - B7 * B5) * idet; Binv[1] = -(B1 * B8 - B7 * B2) * idet; Binv[2] = (B1 * B5 - B4 * B2) * idet; Binv[3] = -(B3 * B8 - B6 * B5) * idet; Binv[4] = (B0 * B8 - B6 * B2) * idet; Binv[5] = -(B0 * B5 - B3 * B2) * idet; Binv[6] = (B3 * B7 - B6 * B4) * idet; Binv[7] = -(B0 * B7 - B6 * B1) * idet; Binv[8] = (B0 * B4 - B3 * B1) * idet; // tmp = B^{-1}D : 3 x D3_P double __attribute__((aligned(8))) tmp[3 * D3_P]; double* r1dim = &(Slater_inv[row1 * D3_P]); double* r2dim = &(Slater_inv[row2 * D3_P]); double* r3dim = &(Slater_inv[row3 * D3_P]); IVDEP ALIGNED for (uint64_t j = 0; j < D3_P; j++) { tmp[j] = Binv[0] * r1dim[j] + Binv[1] * r2dim[j] + Binv[2] * r3dim[j]; tmp[D3_P + j] = Binv[3] * r1dim[j] + Binv[4] * r2dim[j] + Binv[5] * r3dim[j]; tmp[2 * D3_P + j] = Binv[6] * r1dim[j] + Binv[7] * r2dim[j] + Binv[8] * r3dim[j]; } // Compute (S^T)^{-1} - C * tmp : 3 x D3_P for (uint64_t i = 0; i < 3; i++) { IVDEP ALIGNED for (uint64_t j = 0; j < D3_P; j++) { Slater_inv[i * D3_P + j] -= C[i * 3] * tmp[j]; Slater_inv[i * D3_P + j] -= C[i * 3 + 1] * tmp[D3_P + j]; Slater_inv[i * D3_P + j] -= C[i * 3 + 2] * tmp[2 * D3_P + j]; } } return QMCKL_SUCCESS; } qmckl_exit_code qmckl_woodbury_3x3_4( const qmckl_context context, const double* __restrict Updates, const uint64_t* __restrict Updates_index, const double breakdown, double* __restrict Slater_inv, double* __restrict determinant) { /* C := S^{-1} * U, dim x 3 B := 1 + V * C, 3 x 3 D := V * S^{-1}, 3 x dim */ if (qmckl_context_check(context) == QMCKL_NULL_CONTEXT) { return qmckl_failwith(context, QMCKL_NULL_CONTEXT, "qmckl_woodbury_3x3_4", NULL); } const uint64_t row1 = (Updates_index[0] - 1); const uint64_t row2 = (Updates_index[1] - 1); const uint64_t row3 = (Updates_index[2] - 1); // Compute C = (S^T)^{-1}U : 4 x 3 double __attribute__((aligned(8))) C[3 * 4]; for (uint64_t i = 0; i < 4; i++) { C[i * 3] = 0; C[i * 3 + 1] = 0; C[i * 3 + 2] = 0; IVDEP ALIGNED for (uint64_t k = 0; k < D4_P; k++) { C[i * 3] += Slater_inv[i * D4_P + k] * Updates[k]; C[i * 3 + 1] += Slater_inv[i * D4_P + k] * Updates[D4_P + k]; C[i * 3 + 2] += Slater_inv[i * D4_P + k] * Updates[2 * D4_P + k]; } } // Compute B = 1 + VC : 3 x 3 const double B0 = C[row1 * 3] + 1; const double B1 = C[row1 * 3 + 1]; const double B2 = C[row1 * 3 + 2]; const double B3 = C[row2 * 3]; const double B4 = C[row2 * 3 + 1] + 1; const double B5 = C[row2 * 3 + 2]; const double B6 = C[row3 * 3]; const double B7 = C[row3 * 3 + 1]; const double B8 = C[row3 * 3 + 2] + 1; // Check if determinant of B is not too close to zero double det; det = B0 * (B4 * B8 - B5 * B7) - B1 * (B3 * B8 - B5 * B6) + B2 * (B3 * B7 - B4 * B6); if (fabs(det) < breakdown) { return QMCKL_FAILURE; } // Update det(Slater) if passed if (determinant) *determinant *= det; // Compute B^{-1} with explicit formula for 3 x 3 inversion double __attribute__((aligned(8))) Binv[9], idet = 1.0 / det; Binv[0] = (B4 * B8 - B7 * B5) * idet; Binv[1] = -(B1 * B8 - B7 * B2) * idet; Binv[2] = (B1 * B5 - B4 * B2) * idet; Binv[3] = -(B3 * B8 - B6 * B5) * idet; Binv[4] = (B0 * B8 - B6 * B2) * idet; Binv[5] = -(B0 * B5 - B3 * B2) * idet; Binv[6] = (B3 * B7 - B6 * B4) * idet; Binv[7] = -(B0 * B7 - B6 * B1) * idet; Binv[8] = (B0 * B4 - B3 * B1) * idet; // tmp = B^{-1}D : 3 x D4_P double __attribute__((aligned(8))) tmp[3 * D4_P]; double* r1dim = &(Slater_inv[row1 * D4_P]); double* r2dim = &(Slater_inv[row2 * D4_P]); double* r3dim = &(Slater_inv[row3 * D4_P]); IVDEP ALIGNED for (uint64_t j = 0; j < D4_P; j++) { tmp[j] = Binv[0] * r1dim[j] + Binv[1] * r2dim[j] + Binv[2] * r3dim[j]; tmp[D4_P + j] = Binv[3] * r1dim[j] + Binv[4] * r2dim[j] + Binv[5] * r3dim[j]; tmp[2 * D4_P + j] = Binv[6] * r1dim[j] + Binv[7] * r2dim[j] + Binv[8] * r3dim[j]; } // Compute (S^T)^{-1} - C * tmp : 4 x D4_P for (uint64_t i = 0; i < 4; i++) { IVDEP ALIGNED for (uint64_t j = 0; j < D4_P; j++) { Slater_inv[i * D4_P + j] -= C[i * 3] * tmp[j]; Slater_inv[i * D4_P + j] -= C[i * 3 + 1] * tmp[D4_P + j]; Slater_inv[i * D4_P + j] -= C[i * 3 + 2] * tmp[2 * D4_P + j]; } } return QMCKL_SUCCESS; } qmckl_exit_code qmckl_woodbury_3x3_5( const qmckl_context context, const double* __restrict Updates, const uint64_t* __restrict Updates_index, const double breakdown, double* __restrict Slater_inv, double* __restrict determinant) { /* C := S^{-1} * U, dim x 3 B := 1 + V * C, 3 x 3 D := V * S^{-1}, 3 x dim */ if (qmckl_context_check(context) == QMCKL_NULL_CONTEXT) { return qmckl_failwith(context, QMCKL_NULL_CONTEXT, "qmckl_woodbury_3x3_5", NULL); } const uint64_t row1 = (Updates_index[0] - 1); const uint64_t row2 = (Updates_index[1] - 1); const uint64_t row3 = (Updates_index[2] - 1); // Compute C = (S^T)^{-1}U : 5 x 3 double __attribute__((aligned(8))) C[3 * 5]; for (uint64_t i = 0; i < 5; i++) { C[i * 3] = 0; C[i * 3 + 1] = 0; C[i * 3 + 2] = 0; IVDEP ALIGNED for (uint64_t k = 0; k < D5_P; k++) { C[i * 3] += Slater_inv[i * D5_P + k] * Updates[k]; C[i * 3 + 1] += Slater_inv[i * D5_P + k] * Updates[D5_P + k]; C[i * 3 + 2] += Slater_inv[i * D5_P + k] * Updates[2 * D5_P + k]; } } // Compute B = 1 + VC : 3 x 3 const double B0 = C[row1 * 3] + 1; const double B1 = C[row1 * 3 + 1]; const double B2 = C[row1 * 3 + 2]; const double B3 = C[row2 * 3]; const double B4 = C[row2 * 3 + 1] + 1; const double B5 = C[row2 * 3 + 2]; const double B6 = C[row3 * 3]; const double B7 = C[row3 * 3 + 1]; const double B8 = C[row3 * 3 + 2] + 1; // Check if determinant of B is not too close to zero double det; det = B0 * (B4 * B8 - B5 * B7) - B1 * (B3 * B8 - B5 * B6) + B2 * (B3 * B7 - B4 * B6); if (fabs(det) < breakdown) { return QMCKL_FAILURE; } // Update det(Slater) if passed if (determinant) *determinant *= det; // Compute B^{-1} with explicit formula for 3 x 3 inversion double __attribute__((aligned(8))) Binv[9], idet = 1.0 / det; Binv[0] = (B4 * B8 - B7 * B5) * idet; Binv[1] = -(B1 * B8 - B7 * B2) * idet; Binv[2] = (B1 * B5 - B4 * B2) * idet; Binv[3] = -(B3 * B8 - B6 * B5) * idet; Binv[4] = (B0 * B8 - B6 * B2) * idet; Binv[5] = -(B0 * B5 - B3 * B2) * idet; Binv[6] = (B3 * B7 - B6 * B4) * idet; Binv[7] = -(B0 * B7 - B6 * B1) * idet; Binv[8] = (B0 * B4 - B3 * B1) * idet; // tmp = B^{-1}D : 3 x D5_P double __attribute__((aligned(8))) tmp[3 * D5_P]; double* r1dim = &(Slater_inv[row1 * D5_P]); double* r2dim = &(Slater_inv[row2 * D5_P]); double* r3dim = &(Slater_inv[row3 * D5_P]); IVDEP ALIGNED for (uint64_t j = 0; j < D5_P; j++) { tmp[j] = Binv[0] * r1dim[j] + Binv[1] * r2dim[j] + Binv[2] * r3dim[j]; tmp[D5_P + j] = Binv[3] * r1dim[j] + Binv[4] * r2dim[j] + Binv[5] * r3dim[j]; tmp[2 * D5_P + j] = Binv[6] * r1dim[j] + Binv[7] * r2dim[j] + Binv[8] * r3dim[j]; } // Compute (S^T)^{-1} - C * tmp : 5 x D5_P for (uint64_t i = 0; i < 5; i++) { IVDEP ALIGNED for (uint64_t j = 0; j < D5_P; j++) { Slater_inv[i * D5_P + j] -= C[i * 3] * tmp[j]; Slater_inv[i * D5_P + j] -= C[i * 3 + 1] * tmp[D5_P + j]; Slater_inv[i * D5_P + j] -= C[i * 3 + 2] * tmp[2 * D5_P + j]; } } return QMCKL_SUCCESS; } qmckl_exit_code qmckl_woodbury_3x3_6( const qmckl_context context, const double* __restrict Updates, const uint64_t* __restrict Updates_index, const double breakdown, double* __restrict Slater_inv, double* __restrict determinant) { /* C := S^{-1} * U, dim x 3 B := 1 + V * C, 3 x 3 D := V * S^{-1}, 3 x dim */ if (qmckl_context_check(context) == QMCKL_NULL_CONTEXT) { return qmckl_failwith(context, QMCKL_NULL_CONTEXT, "qmckl_woodbury_3x3_6", NULL); } const uint64_t row1 = (Updates_index[0] - 1); const uint64_t row2 = (Updates_index[1] - 1); const uint64_t row3 = (Updates_index[2] - 1); // Compute C = (S^T)^{-1}U : 6 x 3 double __attribute__((aligned(8))) C[3 * 6]; for (uint64_t i = 0; i < 6; i++) { C[i * 3] = 0; C[i * 3 + 1] = 0; C[i * 3 + 2] = 0; IVDEP ALIGNED for (uint64_t k = 0; k < D6_P; k++) { C[i * 3] += Slater_inv[i * D6_P + k] * Updates[k]; C[i * 3 + 1] += Slater_inv[i * D6_P + k] * Updates[D6_P + k]; C[i * 3 + 2] += Slater_inv[i * D6_P + k] * Updates[2 * D6_P + k]; } } // Compute B = 1 + VC : 3 x 3 const double B0 = C[row1 * 3] + 1; const double B1 = C[row1 * 3 + 1]; const double B2 = C[row1 * 3 + 2]; const double B3 = C[row2 * 3]; const double B4 = C[row2 * 3 + 1] + 1; const double B5 = C[row2 * 3 + 2]; const double B6 = C[row3 * 3]; const double B7 = C[row3 * 3 + 1]; const double B8 = C[row3 * 3 + 2] + 1; // Check if determinant of B is not too close to zero double det; det = B0 * (B4 * B8 - B5 * B7) - B1 * (B3 * B8 - B5 * B6) + B2 * (B3 * B7 - B4 * B6); if (fabs(det) < breakdown) { return QMCKL_FAILURE; } // Update det(Slater) if passed if (determinant) *determinant *= det; // Compute B^{-1} with explicit formula for 3 x 3 inversion double __attribute__((aligned(8))) Binv[9], idet = 1.0 / det; Binv[0] = (B4 * B8 - B7 * B5) * idet; Binv[1] = -(B1 * B8 - B7 * B2) * idet; Binv[2] = (B1 * B5 - B4 * B2) * idet; Binv[3] = -(B3 * B8 - B6 * B5) * idet; Binv[4] = (B0 * B8 - B6 * B2) * idet; Binv[5] = -(B0 * B5 - B3 * B2) * idet; Binv[6] = (B3 * B7 - B6 * B4) * idet; Binv[7] = -(B0 * B7 - B6 * B1) * idet; Binv[8] = (B0 * B4 - B3 * B1) * idet; // tmp = B^{-1}D : 3 x D6_P double __attribute__((aligned(8))) tmp[3 * D6_P]; double* r1dim = &(Slater_inv[row1 * D6_P]); double* r2dim = &(Slater_inv[row2 * D6_P]); double* r3dim = &(Slater_inv[row3 * D6_P]); IVDEP ALIGNED for (uint64_t j = 0; j < D6_P; j++) { tmp[j] = Binv[0] * r1dim[j] + Binv[1] * r2dim[j] + Binv[2] * r3dim[j]; tmp[D6_P + j] = Binv[3] * r1dim[j] + Binv[4] * r2dim[j] + Binv[5] * r3dim[j]; tmp[2 * D6_P + j] = Binv[6] * r1dim[j] + Binv[7] * r2dim[j] + Binv[8] * r3dim[j]; } // Compute (S^T)^{-1} - C * tmp : 6 x D6_P for (uint64_t i = 0; i < 6; i++) { IVDEP ALIGNED for (uint64_t j = 0; j < D6_P; j++) { Slater_inv[i * D6_P + j] -= C[i * 3] * tmp[j]; Slater_inv[i * D6_P + j] -= C[i * 3 + 1] * tmp[D6_P + j]; Slater_inv[i * D6_P + j] -= C[i * 3 + 2] * tmp[2 * D6_P + j]; } } return QMCKL_SUCCESS; } qmckl_exit_code qmckl_woodbury_3x3_7( const qmckl_context context, const double* __restrict Updates, const uint64_t* __restrict Updates_index, const double breakdown, double* __restrict Slater_inv, double* __restrict determinant) { /* C := S^{-1} * U, dim x 3 B := 1 + V * C, 3 x 3 D := V * S^{-1}, 3 x dim */ if (qmckl_context_check(context) == QMCKL_NULL_CONTEXT) { return qmckl_failwith(context, QMCKL_NULL_CONTEXT, "qmckl_woodbury_3x3_7", NULL); } const uint64_t row1 = (Updates_index[0] - 1); const uint64_t row2 = (Updates_index[1] - 1); const uint64_t row3 = (Updates_index[2] - 1); // Compute C = (S^T)^{-1}U : 7 x 3 double __attribute__((aligned(8))) C[3 * 7]; for (uint64_t i = 0; i < 7; i++) { C[i * 3] = 0; C[i * 3 + 1] = 0; C[i * 3 + 2] = 0; IVDEP ALIGNED for (uint64_t k = 0; k < D7_P; k++) { C[i * 3] += Slater_inv[i * D7_P + k] * Updates[k]; C[i * 3 + 1] += Slater_inv[i * D7_P + k] * Updates[D7_P + k]; C[i * 3 + 2] += Slater_inv[i * D7_P + k] * Updates[2 * D7_P + k]; } } // Compute B = 1 + VC : 3 x 3 const double B0 = C[row1 * 3] + 1; const double B1 = C[row1 * 3 + 1]; const double B2 = C[row1 * 3 + 2]; const double B3 = C[row2 * 3]; const double B4 = C[row2 * 3 + 1] + 1; const double B5 = C[row2 * 3 + 2]; const double B6 = C[row3 * 3]; const double B7 = C[row3 * 3 + 1]; const double B8 = C[row3 * 3 + 2] + 1; // Check if determinant of B is not too close to zero double det; det = B0 * (B4 * B8 - B5 * B7) - B1 * (B3 * B8 - B5 * B6) + B2 * (B3 * B7 - B4 * B6); if (fabs(det) < breakdown) { return QMCKL_FAILURE; } // Update det(Slater) if passed if (determinant) *determinant *= det; // Compute B^{-1} with explicit formula for 3 x 3 inversion double __attribute__((aligned(8))) Binv[9], idet = 1.0 / det; Binv[0] = (B4 * B8 - B7 * B5) * idet; Binv[1] = -(B1 * B8 - B7 * B2) * idet; Binv[2] = (B1 * B5 - B4 * B2) * idet; Binv[3] = -(B3 * B8 - B6 * B5) * idet; Binv[4] = (B0 * B8 - B6 * B2) * idet; Binv[5] = -(B0 * B5 - B3 * B2) * idet; Binv[6] = (B3 * B7 - B6 * B4) * idet; Binv[7] = -(B0 * B7 - B6 * B1) * idet; Binv[8] = (B0 * B4 - B3 * B1) * idet; // tmp = B^{-1}D : 3 x D7_P double __attribute__((aligned(8))) tmp[3 * D7_P]; double* r1dim = &(Slater_inv[row1 * D7_P]); double* r2dim = &(Slater_inv[row2 * D7_P]); double* r3dim = &(Slater_inv[row3 * D7_P]); IVDEP ALIGNED for (uint64_t j = 0; j < D7_P; j++) { tmp[j] = Binv[0] * r1dim[j] + Binv[1] * r2dim[j] + Binv[2] * r3dim[j]; tmp[D7_P + j] = Binv[3] * r1dim[j] + Binv[4] * r2dim[j] + Binv[5] * r3dim[j]; tmp[2 * D7_P + j] = Binv[6] * r1dim[j] + Binv[7] * r2dim[j] + Binv[8] * r3dim[j]; } // Compute (S^T)^{-1} - C * tmp : 7 x D7_P for (uint64_t i = 0; i < 7; i++) { IVDEP ALIGNED for (uint64_t j = 0; j < D7_P; j++) { Slater_inv[i * D7_P + j] -= C[i * 3] * tmp[j]; Slater_inv[i * D7_P + j] -= C[i * 3 + 1] * tmp[D7_P + j]; Slater_inv[i * D7_P + j] -= C[i * 3 + 2] * tmp[2 * D7_P + j]; } } return QMCKL_SUCCESS; } qmckl_exit_code qmckl_woodbury_3x3_8( const qmckl_context context, const double* __restrict Updates, const uint64_t* __restrict Updates_index, const double breakdown, double* __restrict Slater_inv, double* __restrict determinant) { /* C := S^{-1} * U, dim x 3 B := 1 + V * C, 3 x 3 D := V * S^{-1}, 3 x dim */ if (qmckl_context_check(context) == QMCKL_NULL_CONTEXT) { return qmckl_failwith(context, QMCKL_NULL_CONTEXT, "qmckl_woodbury_3x3_8", NULL); } const uint64_t row1 = (Updates_index[0] - 1); const uint64_t row2 = (Updates_index[1] - 1); const uint64_t row3 = (Updates_index[2] - 1); // Compute C = (S^T)^{-1}U : 8 x 3 double __attribute__((aligned(8))) C[3 * 8]; for (uint64_t i = 0; i < 8; i++) { C[i * 3] = 0; C[i * 3 + 1] = 0; C[i * 3 + 2] = 0; IVDEP ALIGNED for (uint64_t k = 0; k < D8_P; k++) { C[i * 3] += Slater_inv[i * D8_P + k] * Updates[k]; C[i * 3 + 1] += Slater_inv[i * D8_P + k] * Updates[D8_P + k]; C[i * 3 + 2] += Slater_inv[i * D8_P + k] * Updates[2 * D8_P + k]; } } // Compute B = 1 + VC : 3 x 3 const double B0 = C[row1 * 3] + 1; const double B1 = C[row1 * 3 + 1]; const double B2 = C[row1 * 3 + 2]; const double B3 = C[row2 * 3]; const double B4 = C[row2 * 3 + 1] + 1; const double B5 = C[row2 * 3 + 2]; const double B6 = C[row3 * 3]; const double B7 = C[row3 * 3 + 1]; const double B8 = C[row3 * 3 + 2] + 1; // Check if determinant of B is not too close to zero double det; det = B0 * (B4 * B8 - B5 * B7) - B1 * (B3 * B8 - B5 * B6) + B2 * (B3 * B7 - B4 * B6); if (fabs(det) < breakdown) { return QMCKL_FAILURE; } // Update det(Slater) if passed if (determinant) *determinant *= det; // Compute B^{-1} with explicit formula for 3 x 3 inversion double __attribute__((aligned(8))) Binv[9], idet = 1.0 / det; Binv[0] = (B4 * B8 - B7 * B5) * idet; Binv[1] = -(B1 * B8 - B7 * B2) * idet; Binv[2] = (B1 * B5 - B4 * B2) * idet; Binv[3] = -(B3 * B8 - B6 * B5) * idet; Binv[4] = (B0 * B8 - B6 * B2) * idet; Binv[5] = -(B0 * B5 - B3 * B2) * idet; Binv[6] = (B3 * B7 - B6 * B4) * idet; Binv[7] = -(B0 * B7 - B6 * B1) * idet; Binv[8] = (B0 * B4 - B3 * B1) * idet; // tmp = B^{-1}D : 3 x D8_P double __attribute__((aligned(8))) tmp[3 * D8_P]; double* r1dim = &(Slater_inv[row1 * D8_P]); double* r2dim = &(Slater_inv[row2 * D8_P]); double* r3dim = &(Slater_inv[row3 * D8_P]); IVDEP ALIGNED for (uint64_t j = 0; j < D8_P; j++) { tmp[j] = Binv[0] * r1dim[j] + Binv[1] * r2dim[j] + Binv[2] * r3dim[j]; tmp[D8_P + j] = Binv[3] * r1dim[j] + Binv[4] * r2dim[j] + Binv[5] * r3dim[j]; tmp[2 * D8_P + j] = Binv[6] * r1dim[j] + Binv[7] * r2dim[j] + Binv[8] * r3dim[j]; } // Compute (S^T)^{-1} - C * tmp : 8 x D8_P for (uint64_t i = 0; i < 8; i++) { IVDEP ALIGNED for (uint64_t j = 0; j < D8_P; j++) { Slater_inv[i * D8_P + j] -= C[i * 3] * tmp[j]; Slater_inv[i * D8_P + j] -= C[i * 3 + 1] * tmp[D8_P + j]; Slater_inv[i * D8_P + j] -= C[i * 3 + 2] * tmp[2 * D8_P + j]; } } return QMCKL_SUCCESS; } qmckl_exit_code qmckl_woodbury_3x3_9( const qmckl_context context, const double* __restrict Updates, const uint64_t* __restrict Updates_index, const double breakdown, double* __restrict Slater_inv, double* __restrict determinant) { /* C := S^{-1} * U, dim x 3 B := 1 + V * C, 3 x 3 D := V * S^{-1}, 3 x dim */ if (qmckl_context_check(context) == QMCKL_NULL_CONTEXT) { return qmckl_failwith(context, QMCKL_NULL_CONTEXT, "qmckl_woodbury_3x3_9", NULL); } const uint64_t row1 = (Updates_index[0] - 1); const uint64_t row2 = (Updates_index[1] - 1); const uint64_t row3 = (Updates_index[2] - 1); // Compute C = (S^T)^{-1}U : 9 x 3 double __attribute__((aligned(8))) C[3 * 9]; for (uint64_t i = 0; i < 9; i++) { C[i * 3] = 0; C[i * 3 + 1] = 0; C[i * 3 + 2] = 0; IVDEP ALIGNED for (uint64_t k = 0; k < D9_P; k++) { C[i * 3] += Slater_inv[i * D9_P + k] * Updates[k]; C[i * 3 + 1] += Slater_inv[i * D9_P + k] * Updates[D9_P + k]; C[i * 3 + 2] += Slater_inv[i * D9_P + k] * Updates[2 * D9_P + k]; } } // Compute B = 1 + VC : 3 x 3 const double B0 = C[row1 * 3] + 1; const double B1 = C[row1 * 3 + 1]; const double B2 = C[row1 * 3 + 2]; const double B3 = C[row2 * 3]; const double B4 = C[row2 * 3 + 1] + 1; const double B5 = C[row2 * 3 + 2]; const double B6 = C[row3 * 3]; const double B7 = C[row3 * 3 + 1]; const double B8 = C[row3 * 3 + 2] + 1; // Check if determinant of B is not too close to zero double det; det = B0 * (B4 * B8 - B5 * B7) - B1 * (B3 * B8 - B5 * B6) + B2 * (B3 * B7 - B4 * B6); if (fabs(det) < breakdown) { return QMCKL_FAILURE; } // Update det(Slater) if passed if (determinant) *determinant *= det; // Compute B^{-1} with explicit formula for 3 x 3 inversion double __attribute__((aligned(8))) Binv[9], idet = 1.0 / det; Binv[0] = (B4 * B8 - B7 * B5) * idet; Binv[1] = -(B1 * B8 - B7 * B2) * idet; Binv[2] = (B1 * B5 - B4 * B2) * idet; Binv[3] = -(B3 * B8 - B6 * B5) * idet; Binv[4] = (B0 * B8 - B6 * B2) * idet; Binv[5] = -(B0 * B5 - B3 * B2) * idet; Binv[6] = (B3 * B7 - B6 * B4) * idet; Binv[7] = -(B0 * B7 - B6 * B1) * idet; Binv[8] = (B0 * B4 - B3 * B1) * idet; // tmp = B^{-1}D : 3 x D9_P double __attribute__((aligned(8))) tmp[3 * D9_P]; double* r1dim = &(Slater_inv[row1 * D9_P]); double* r2dim = &(Slater_inv[row2 * D9_P]); double* r3dim = &(Slater_inv[row3 * D9_P]); IVDEP ALIGNED for (uint64_t j = 0; j < D9_P; j++) { tmp[j] = Binv[0] * r1dim[j] + Binv[1] * r2dim[j] + Binv[2] * r3dim[j]; tmp[D9_P + j] = Binv[3] * r1dim[j] + Binv[4] * r2dim[j] + Binv[5] * r3dim[j]; tmp[2 * D9_P + j] = Binv[6] * r1dim[j] + Binv[7] * r2dim[j] + Binv[8] * r3dim[j]; } // Compute (S^T)^{-1} - C * tmp : 9 x D9_P for (uint64_t i = 0; i < 9; i++) { IVDEP ALIGNED for (uint64_t j = 0; j < D9_P; j++) { Slater_inv[i * D9_P + j] -= C[i * 3] * tmp[j]; Slater_inv[i * D9_P + j] -= C[i * 3 + 1] * tmp[D9_P + j]; Slater_inv[i * D9_P + j] -= C[i * 3 + 2] * tmp[2 * D9_P + j]; } } return QMCKL_SUCCESS; } qmckl_exit_code qmckl_woodbury_3x3_10( const qmckl_context context, const double* __restrict Updates, const uint64_t* __restrict Updates_index, const double breakdown, double* __restrict Slater_inv, double* __restrict determinant) { /* C := S^{-1} * U, dim x 3 B := 1 + V * C, 3 x 3 D := V * S^{-1}, 3 x dim */ if (qmckl_context_check(context) == QMCKL_NULL_CONTEXT) { return qmckl_failwith(context, QMCKL_NULL_CONTEXT, "qmckl_woodbury_3x3_10", NULL); } const uint64_t row1 = (Updates_index[0] - 1); const uint64_t row2 = (Updates_index[1] - 1); const uint64_t row3 = (Updates_index[2] - 1); // Compute C = (S^T)^{-1}U : 10 x 3 double __attribute__((aligned(8))) C[3 * 10]; for (uint64_t i = 0; i < 10; i++) { C[i * 3] = 0; C[i * 3 + 1] = 0; C[i * 3 + 2] = 0; IVDEP ALIGNED for (uint64_t k = 0; k < D10_P; k++) { C[i * 3] += Slater_inv[i * D10_P + k] * Updates[k]; C[i * 3 + 1] += Slater_inv[i * D10_P + k] * Updates[D10_P + k]; C[i * 3 + 2] += Slater_inv[i * D10_P + k] * Updates[2 * D10_P + k]; } } // Compute B = 1 + VC : 3 x 3 const double B0 = C[row1 * 3] + 1; const double B1 = C[row1 * 3 + 1]; const double B2 = C[row1 * 3 + 2]; const double B3 = C[row2 * 3]; const double B4 = C[row2 * 3 + 1] + 1; const double B5 = C[row2 * 3 + 2]; const double B6 = C[row3 * 3]; const double B7 = C[row3 * 3 + 1]; const double B8 = C[row3 * 3 + 2] + 1; // Check if determinant of B is not too close to zero double det; det = B0 * (B4 * B8 - B5 * B7) - B1 * (B3 * B8 - B5 * B6) + B2 * (B3 * B7 - B4 * B6); if (fabs(det) < breakdown) { return QMCKL_FAILURE; } // Update det(Slater) if passed if (determinant) *determinant *= det; // Compute B^{-1} with explicit formula for 3 x 3 inversion double __attribute__((aligned(8))) Binv[9], idet = 1.0 / det; Binv[0] = (B4 * B8 - B7 * B5) * idet; Binv[1] = -(B1 * B8 - B7 * B2) * idet; Binv[2] = (B1 * B5 - B4 * B2) * idet; Binv[3] = -(B3 * B8 - B6 * B5) * idet; Binv[4] = (B0 * B8 - B6 * B2) * idet; Binv[5] = -(B0 * B5 - B3 * B2) * idet; Binv[6] = (B3 * B7 - B6 * B4) * idet; Binv[7] = -(B0 * B7 - B6 * B1) * idet; Binv[8] = (B0 * B4 - B3 * B1) * idet; // tmp = B^{-1}D : 3 x D10_P double __attribute__((aligned(8))) tmp[3 * D10_P]; double* r1dim = &(Slater_inv[row1 * D10_P]); double* r2dim = &(Slater_inv[row2 * D10_P]); double* r3dim = &(Slater_inv[row3 * D10_P]); IVDEP ALIGNED for (uint64_t j = 0; j < D10_P; j++) { tmp[j] = Binv[0] * r1dim[j] + Binv[1] * r2dim[j] + Binv[2] * r3dim[j]; tmp[D10_P + j] = Binv[3] * r1dim[j] + Binv[4] * r2dim[j] + Binv[5] * r3dim[j]; tmp[2 * D10_P + j] = Binv[6] * r1dim[j] + Binv[7] * r2dim[j] + Binv[8] * r3dim[j]; } // Compute (S^T)^{-1} - C * tmp : 10 x D10_P for (uint64_t i = 0; i < 10; i++) { IVDEP ALIGNED for (uint64_t j = 0; j < D10_P; j++) { Slater_inv[i * D10_P + j] -= C[i * 3] * tmp[j]; Slater_inv[i * D10_P + j] -= C[i * 3 + 1] * tmp[D10_P + j]; Slater_inv[i * D10_P + j] -= C[i * 3 + 2] * tmp[2 * D10_P + j]; } } return QMCKL_SUCCESS; } qmckl_exit_code qmckl_woodbury_3x3_11( const qmckl_context context, const double* __restrict Updates, const uint64_t* __restrict Updates_index, const double breakdown, double* __restrict Slater_inv, double* __restrict determinant) { /* C := S^{-1} * U, dim x 3 B := 1 + V * C, 3 x 3 D := V * S^{-1}, 3 x dim */ if (qmckl_context_check(context) == QMCKL_NULL_CONTEXT) { return qmckl_failwith(context, QMCKL_NULL_CONTEXT, "qmckl_woodbury_3x3_11", NULL); } const uint64_t row1 = (Updates_index[0] - 1); const uint64_t row2 = (Updates_index[1] - 1); const uint64_t row3 = (Updates_index[2] - 1); // Compute C = (S^T)^{-1}U : 11 x 3 double __attribute__((aligned(8))) C[3 * 11]; for (uint64_t i = 0; i < 11; i++) { C[i * 3] = 0; C[i * 3 + 1] = 0; C[i * 3 + 2] = 0; IVDEP ALIGNED for (uint64_t k = 0; k < D11_P; k++) { C[i * 3] += Slater_inv[i * D11_P + k] * Updates[k]; C[i * 3 + 1] += Slater_inv[i * D11_P + k] * Updates[D11_P + k]; C[i * 3 + 2] += Slater_inv[i * D11_P + k] * Updates[2 * D11_P + k]; } } // Compute B = 1 + VC : 3 x 3 const double B0 = C[row1 * 3] + 1; const double B1 = C[row1 * 3 + 1]; const double B2 = C[row1 * 3 + 2]; const double B3 = C[row2 * 3]; const double B4 = C[row2 * 3 + 1] + 1; const double B5 = C[row2 * 3 + 2]; const double B6 = C[row3 * 3]; const double B7 = C[row3 * 3 + 1]; const double B8 = C[row3 * 3 + 2] + 1; // Check if determinant of B is not too close to zero double det; det = B0 * (B4 * B8 - B5 * B7) - B1 * (B3 * B8 - B5 * B6) + B2 * (B3 * B7 - B4 * B6); if (fabs(det) < breakdown) { return QMCKL_FAILURE; } // Update det(Slater) if passed if (determinant) *determinant *= det; // Compute B^{-1} with explicit formula for 3 x 3 inversion double __attribute__((aligned(8))) Binv[9], idet = 1.0 / det; Binv[0] = (B4 * B8 - B7 * B5) * idet; Binv[1] = -(B1 * B8 - B7 * B2) * idet; Binv[2] = (B1 * B5 - B4 * B2) * idet; Binv[3] = -(B3 * B8 - B6 * B5) * idet; Binv[4] = (B0 * B8 - B6 * B2) * idet; Binv[5] = -(B0 * B5 - B3 * B2) * idet; Binv[6] = (B3 * B7 - B6 * B4) * idet; Binv[7] = -(B0 * B7 - B6 * B1) * idet; Binv[8] = (B0 * B4 - B3 * B1) * idet; // tmp = B^{-1}D : 3 x D11_P double __attribute__((aligned(8))) tmp[3 * D11_P]; double* r1dim = &(Slater_inv[row1 * D11_P]); double* r2dim = &(Slater_inv[row2 * D11_P]); double* r3dim = &(Slater_inv[row3 * D11_P]); IVDEP ALIGNED for (uint64_t j = 0; j < D11_P; j++) { tmp[j] = Binv[0] * r1dim[j] + Binv[1] * r2dim[j] + Binv[2] * r3dim[j]; tmp[D11_P + j] = Binv[3] * r1dim[j] + Binv[4] * r2dim[j] + Binv[5] * r3dim[j]; tmp[2 * D11_P + j] = Binv[6] * r1dim[j] + Binv[7] * r2dim[j] + Binv[8] * r3dim[j]; } // Compute (S^T)^{-1} - C * tmp : 11 x D11_P for (uint64_t i = 0; i < 11; i++) { IVDEP ALIGNED for (uint64_t j = 0; j < D11_P; j++) { Slater_inv[i * D11_P + j] -= C[i * 3] * tmp[j]; Slater_inv[i * D11_P + j] -= C[i * 3 + 1] * tmp[D11_P + j]; Slater_inv[i * D11_P + j] -= C[i * 3 + 2] * tmp[2 * D11_P + j]; } } return QMCKL_SUCCESS; } qmckl_exit_code qmckl_woodbury_3x3_12( const qmckl_context context, const double* __restrict Updates, const uint64_t* __restrict Updates_index, const double breakdown, double* __restrict Slater_inv, double* __restrict determinant) { /* C := S^{-1} * U, dim x 3 B := 1 + V * C, 3 x 3 D := V * S^{-1}, 3 x dim */ if (qmckl_context_check(context) == QMCKL_NULL_CONTEXT) { return qmckl_failwith(context, QMCKL_NULL_CONTEXT, "qmckl_woodbury_3x3_12", NULL); } const uint64_t row1 = (Updates_index[0] - 1); const uint64_t row2 = (Updates_index[1] - 1); const uint64_t row3 = (Updates_index[2] - 1); // Compute C = (S^T)^{-1}U : 12 x 3 double __attribute__((aligned(8))) C[3 * 12]; for (uint64_t i = 0; i < 12; i++) { C[i * 3] = 0; C[i * 3 + 1] = 0; C[i * 3 + 2] = 0; IVDEP ALIGNED for (uint64_t k = 0; k < D12_P; k++) { C[i * 3] += Slater_inv[i * D12_P + k] * Updates[k]; C[i * 3 + 1] += Slater_inv[i * D12_P + k] * Updates[D12_P + k]; C[i * 3 + 2] += Slater_inv[i * D12_P + k] * Updates[2 * D12_P + k]; } } // Compute B = 1 + VC : 3 x 3 const double B0 = C[row1 * 3] + 1; const double B1 = C[row1 * 3 + 1]; const double B2 = C[row1 * 3 + 2]; const double B3 = C[row2 * 3]; const double B4 = C[row2 * 3 + 1] + 1; const double B5 = C[row2 * 3 + 2]; const double B6 = C[row3 * 3]; const double B7 = C[row3 * 3 + 1]; const double B8 = C[row3 * 3 + 2] + 1; // Check if determinant of B is not too close to zero double det; det = B0 * (B4 * B8 - B5 * B7) - B1 * (B3 * B8 - B5 * B6) + B2 * (B3 * B7 - B4 * B6); if (fabs(det) < breakdown) { return QMCKL_FAILURE; } // Update det(Slater) if passed if (determinant) *determinant *= det; // Compute B^{-1} with explicit formula for 3 x 3 inversion double __attribute__((aligned(8))) Binv[9], idet = 1.0 / det; Binv[0] = (B4 * B8 - B7 * B5) * idet; Binv[1] = -(B1 * B8 - B7 * B2) * idet; Binv[2] = (B1 * B5 - B4 * B2) * idet; Binv[3] = -(B3 * B8 - B6 * B5) * idet; Binv[4] = (B0 * B8 - B6 * B2) * idet; Binv[5] = -(B0 * B5 - B3 * B2) * idet; Binv[6] = (B3 * B7 - B6 * B4) * idet; Binv[7] = -(B0 * B7 - B6 * B1) * idet; Binv[8] = (B0 * B4 - B3 * B1) * idet; // tmp = B^{-1}D : 3 x D12_P double __attribute__((aligned(8))) tmp[3 * D12_P]; double* r1dim = &(Slater_inv[row1 * D12_P]); double* r2dim = &(Slater_inv[row2 * D12_P]); double* r3dim = &(Slater_inv[row3 * D12_P]); IVDEP ALIGNED for (uint64_t j = 0; j < D12_P; j++) { tmp[j] = Binv[0] * r1dim[j] + Binv[1] * r2dim[j] + Binv[2] * r3dim[j]; tmp[D12_P + j] = Binv[3] * r1dim[j] + Binv[4] * r2dim[j] + Binv[5] * r3dim[j]; tmp[2 * D12_P + j] = Binv[6] * r1dim[j] + Binv[7] * r2dim[j] + Binv[8] * r3dim[j]; } // Compute (S^T)^{-1} - C * tmp : 12 x D12_P for (uint64_t i = 0; i < 12; i++) { IVDEP ALIGNED for (uint64_t j = 0; j < D12_P; j++) { Slater_inv[i * D12_P + j] -= C[i * 3] * tmp[j]; Slater_inv[i * D12_P + j] -= C[i * 3 + 1] * tmp[D12_P + j]; Slater_inv[i * D12_P + j] -= C[i * 3 + 2] * tmp[2 * D12_P + j]; } } return QMCKL_SUCCESS; } qmckl_exit_code qmckl_woodbury_3x3_13( const qmckl_context context, const double* __restrict Updates, const uint64_t* __restrict Updates_index, const double breakdown, double* __restrict Slater_inv, double* __restrict determinant) { /* C := S^{-1} * U, dim x 3 B := 1 + V * C, 3 x 3 D := V * S^{-1}, 3 x dim */ if (qmckl_context_check(context) == QMCKL_NULL_CONTEXT) { return qmckl_failwith(context, QMCKL_NULL_CONTEXT, "qmckl_woodbury_3x3_13", NULL); } const uint64_t row1 = (Updates_index[0] - 1); const uint64_t row2 = (Updates_index[1] - 1); const uint64_t row3 = (Updates_index[2] - 1); // Compute C = (S^T)^{-1}U : 13 x 3 double __attribute__((aligned(8))) C[3 * 13]; for (uint64_t i = 0; i < 13; i++) { C[i * 3] = 0; C[i * 3 + 1] = 0; C[i * 3 + 2] = 0; IVDEP ALIGNED for (uint64_t k = 0; k < D13_P; k++) { C[i * 3] += Slater_inv[i * D13_P + k] * Updates[k]; C[i * 3 + 1] += Slater_inv[i * D13_P + k] * Updates[D13_P + k]; C[i * 3 + 2] += Slater_inv[i * D13_P + k] * Updates[2 * D13_P + k]; } } // Compute B = 1 + VC : 3 x 3 const double B0 = C[row1 * 3] + 1; const double B1 = C[row1 * 3 + 1]; const double B2 = C[row1 * 3 + 2]; const double B3 = C[row2 * 3]; const double B4 = C[row2 * 3 + 1] + 1; const double B5 = C[row2 * 3 + 2]; const double B6 = C[row3 * 3]; const double B7 = C[row3 * 3 + 1]; const double B8 = C[row3 * 3 + 2] + 1; // Check if determinant of B is not too close to zero double det; det = B0 * (B4 * B8 - B5 * B7) - B1 * (B3 * B8 - B5 * B6) + B2 * (B3 * B7 - B4 * B6); if (fabs(det) < breakdown) { return QMCKL_FAILURE; } // Update det(Slater) if passed if (determinant) *determinant *= det; // Compute B^{-1} with explicit formula for 3 x 3 inversion double __attribute__((aligned(8))) Binv[9], idet = 1.0 / det; Binv[0] = (B4 * B8 - B7 * B5) * idet; Binv[1] = -(B1 * B8 - B7 * B2) * idet; Binv[2] = (B1 * B5 - B4 * B2) * idet; Binv[3] = -(B3 * B8 - B6 * B5) * idet; Binv[4] = (B0 * B8 - B6 * B2) * idet; Binv[5] = -(B0 * B5 - B3 * B2) * idet; Binv[6] = (B3 * B7 - B6 * B4) * idet; Binv[7] = -(B0 * B7 - B6 * B1) * idet; Binv[8] = (B0 * B4 - B3 * B1) * idet; // tmp = B^{-1}D : 3 x D13_P double __attribute__((aligned(8))) tmp[3 * D13_P]; double* r1dim = &(Slater_inv[row1 * D13_P]); double* r2dim = &(Slater_inv[row2 * D13_P]); double* r3dim = &(Slater_inv[row3 * D13_P]); IVDEP ALIGNED for (uint64_t j = 0; j < D13_P; j++) { tmp[j] = Binv[0] * r1dim[j] + Binv[1] * r2dim[j] + Binv[2] * r3dim[j]; tmp[D13_P + j] = Binv[3] * r1dim[j] + Binv[4] * r2dim[j] + Binv[5] * r3dim[j]; tmp[2 * D13_P + j] = Binv[6] * r1dim[j] + Binv[7] * r2dim[j] + Binv[8] * r3dim[j]; } // Compute (S^T)^{-1} - C * tmp : 13 x D13_P for (uint64_t i = 0; i < 13; i++) { IVDEP ALIGNED for (uint64_t j = 0; j < D13_P; j++) { Slater_inv[i * D13_P + j] -= C[i * 3] * tmp[j]; Slater_inv[i * D13_P + j] -= C[i * 3 + 1] * tmp[D13_P + j]; Slater_inv[i * D13_P + j] -= C[i * 3 + 2] * tmp[2 * D13_P + j]; } } return QMCKL_SUCCESS; } qmckl_exit_code qmckl_woodbury_3x3_14( const qmckl_context context, const double* __restrict Updates, const uint64_t* __restrict Updates_index, const double breakdown, double* __restrict Slater_inv, double* __restrict determinant) { /* C := S^{-1} * U, dim x 3 B := 1 + V * C, 3 x 3 D := V * S^{-1}, 3 x dim */ if (qmckl_context_check(context) == QMCKL_NULL_CONTEXT) { return qmckl_failwith(context, QMCKL_NULL_CONTEXT, "qmckl_woodbury_3x3_14", NULL); } const uint64_t row1 = (Updates_index[0] - 1); const uint64_t row2 = (Updates_index[1] - 1); const uint64_t row3 = (Updates_index[2] - 1); // Compute C = (S^T)^{-1}U : 14 x 3 double __attribute__((aligned(8))) C[3 * 14]; for (uint64_t i = 0; i < 14; i++) { C[i * 3] = 0; C[i * 3 + 1] = 0; C[i * 3 + 2] = 0; IVDEP ALIGNED for (uint64_t k = 0; k < D14_P; k++) { C[i * 3] += Slater_inv[i * D14_P + k] * Updates[k]; C[i * 3 + 1] += Slater_inv[i * D14_P + k] * Updates[D14_P + k]; C[i * 3 + 2] += Slater_inv[i * D14_P + k] * Updates[2 * D14_P + k]; } } // Compute B = 1 + VC : 3 x 3 const double B0 = C[row1 * 3] + 1; const double B1 = C[row1 * 3 + 1]; const double B2 = C[row1 * 3 + 2]; const double B3 = C[row2 * 3]; const double B4 = C[row2 * 3 + 1] + 1; const double B5 = C[row2 * 3 + 2]; const double B6 = C[row3 * 3]; const double B7 = C[row3 * 3 + 1]; const double B8 = C[row3 * 3 + 2] + 1; // Check if determinant of B is not too close to zero double det; det = B0 * (B4 * B8 - B5 * B7) - B1 * (B3 * B8 - B5 * B6) + B2 * (B3 * B7 - B4 * B6); if (fabs(det) < breakdown) { return QMCKL_FAILURE; } // Update det(Slater) if passed if (determinant) *determinant *= det; // Compute B^{-1} with explicit formula for 3 x 3 inversion double __attribute__((aligned(8))) Binv[9], idet = 1.0 / det; Binv[0] = (B4 * B8 - B7 * B5) * idet; Binv[1] = -(B1 * B8 - B7 * B2) * idet; Binv[2] = (B1 * B5 - B4 * B2) * idet; Binv[3] = -(B3 * B8 - B6 * B5) * idet; Binv[4] = (B0 * B8 - B6 * B2) * idet; Binv[5] = -(B0 * B5 - B3 * B2) * idet; Binv[6] = (B3 * B7 - B6 * B4) * idet; Binv[7] = -(B0 * B7 - B6 * B1) * idet; Binv[8] = (B0 * B4 - B3 * B1) * idet; // tmp = B^{-1}D : 3 x D14_P double __attribute__((aligned(8))) tmp[3 * D14_P]; double* r1dim = &(Slater_inv[row1 * D14_P]); double* r2dim = &(Slater_inv[row2 * D14_P]); double* r3dim = &(Slater_inv[row3 * D14_P]); IVDEP ALIGNED for (uint64_t j = 0; j < D14_P; j++) { tmp[j] = Binv[0] * r1dim[j] + Binv[1] * r2dim[j] + Binv[2] * r3dim[j]; tmp[D14_P + j] = Binv[3] * r1dim[j] + Binv[4] * r2dim[j] + Binv[5] * r3dim[j]; tmp[2 * D14_P + j] = Binv[6] * r1dim[j] + Binv[7] * r2dim[j] + Binv[8] * r3dim[j]; } // Compute (S^T)^{-1} - C * tmp : 14 x D14_P for (uint64_t i = 0; i < 14; i++) { IVDEP ALIGNED for (uint64_t j = 0; j < D14_P; j++) { Slater_inv[i * D14_P + j] -= C[i * 3] * tmp[j]; Slater_inv[i * D14_P + j] -= C[i * 3 + 1] * tmp[D14_P + j]; Slater_inv[i * D14_P + j] -= C[i * 3 + 2] * tmp[2 * D14_P + j]; } } return QMCKL_SUCCESS; } qmckl_exit_code qmckl_woodbury_3x3_15( const qmckl_context context, const double* __restrict Updates, const uint64_t* __restrict Updates_index, const double breakdown, double* __restrict Slater_inv, double* __restrict determinant) { /* C := S^{-1} * U, dim x 3 B := 1 + V * C, 3 x 3 D := V * S^{-1}, 3 x dim */ if (qmckl_context_check(context) == QMCKL_NULL_CONTEXT) { return qmckl_failwith(context, QMCKL_NULL_CONTEXT, "qmckl_woodbury_3x3_15", NULL); } const uint64_t row1 = (Updates_index[0] - 1); const uint64_t row2 = (Updates_index[1] - 1); const uint64_t row3 = (Updates_index[2] - 1); // Compute C = (S^T)^{-1}U : 15 x 3 double __attribute__((aligned(8))) C[3 * 15]; for (uint64_t i = 0; i < 15; i++) { C[i * 3] = 0; C[i * 3 + 1] = 0; C[i * 3 + 2] = 0; IVDEP ALIGNED for (uint64_t k = 0; k < D15_P; k++) { C[i * 3] += Slater_inv[i * D15_P + k] * Updates[k]; C[i * 3 + 1] += Slater_inv[i * D15_P + k] * Updates[D15_P + k]; C[i * 3 + 2] += Slater_inv[i * D15_P + k] * Updates[2 * D15_P + k]; } } // Compute B = 1 + VC : 3 x 3 const double B0 = C[row1 * 3] + 1; const double B1 = C[row1 * 3 + 1]; const double B2 = C[row1 * 3 + 2]; const double B3 = C[row2 * 3]; const double B4 = C[row2 * 3 + 1] + 1; const double B5 = C[row2 * 3 + 2]; const double B6 = C[row3 * 3]; const double B7 = C[row3 * 3 + 1]; const double B8 = C[row3 * 3 + 2] + 1; // Check if determinant of B is not too close to zero double det; det = B0 * (B4 * B8 - B5 * B7) - B1 * (B3 * B8 - B5 * B6) + B2 * (B3 * B7 - B4 * B6); if (fabs(det) < breakdown) { return QMCKL_FAILURE; } // Update det(Slater) if passed if (determinant) *determinant *= det; // Compute B^{-1} with explicit formula for 3 x 3 inversion double __attribute__((aligned(8))) Binv[9], idet = 1.0 / det; Binv[0] = (B4 * B8 - B7 * B5) * idet; Binv[1] = -(B1 * B8 - B7 * B2) * idet; Binv[2] = (B1 * B5 - B4 * B2) * idet; Binv[3] = -(B3 * B8 - B6 * B5) * idet; Binv[4] = (B0 * B8 - B6 * B2) * idet; Binv[5] = -(B0 * B5 - B3 * B2) * idet; Binv[6] = (B3 * B7 - B6 * B4) * idet; Binv[7] = -(B0 * B7 - B6 * B1) * idet; Binv[8] = (B0 * B4 - B3 * B1) * idet; // tmp = B^{-1}D : 3 x D15_P double __attribute__((aligned(8))) tmp[3 * D15_P]; double* r1dim = &(Slater_inv[row1 * D15_P]); double* r2dim = &(Slater_inv[row2 * D15_P]); double* r3dim = &(Slater_inv[row3 * D15_P]); IVDEP ALIGNED for (uint64_t j = 0; j < D15_P; j++) { tmp[j] = Binv[0] * r1dim[j] + Binv[1] * r2dim[j] + Binv[2] * r3dim[j]; tmp[D15_P + j] = Binv[3] * r1dim[j] + Binv[4] * r2dim[j] + Binv[5] * r3dim[j]; tmp[2 * D15_P + j] = Binv[6] * r1dim[j] + Binv[7] * r2dim[j] + Binv[8] * r3dim[j]; } // Compute (S^T)^{-1} - C * tmp : 15 x D15_P for (uint64_t i = 0; i < 15; i++) { IVDEP ALIGNED for (uint64_t j = 0; j < D15_P; j++) { Slater_inv[i * D15_P + j] -= C[i * 3] * tmp[j]; Slater_inv[i * D15_P + j] -= C[i * 3 + 1] * tmp[D15_P + j]; Slater_inv[i * D15_P + j] -= C[i * 3 + 2] * tmp[2 * D15_P + j]; } } return QMCKL_SUCCESS; } qmckl_exit_code qmckl_woodbury_3x3_16( const qmckl_context context, const double* __restrict Updates, const uint64_t* __restrict Updates_index, const double breakdown, double* __restrict Slater_inv, double* __restrict determinant) { /* C := S^{-1} * U, dim x 3 B := 1 + V * C, 3 x 3 D := V * S^{-1}, 3 x dim */ if (qmckl_context_check(context) == QMCKL_NULL_CONTEXT) { return qmckl_failwith(context, QMCKL_NULL_CONTEXT, "qmckl_woodbury_3x3_16", NULL); } const uint64_t row1 = (Updates_index[0] - 1); const uint64_t row2 = (Updates_index[1] - 1); const uint64_t row3 = (Updates_index[2] - 1); // Compute C = (S^T)^{-1}U : 16 x 3 double __attribute__((aligned(8))) C[3 * 16]; for (uint64_t i = 0; i < 16; i++) { C[i * 3] = 0; C[i * 3 + 1] = 0; C[i * 3 + 2] = 0; IVDEP ALIGNED for (uint64_t k = 0; k < D16_P; k++) { C[i * 3] += Slater_inv[i * D16_P + k] * Updates[k]; C[i * 3 + 1] += Slater_inv[i * D16_P + k] * Updates[D16_P + k]; C[i * 3 + 2] += Slater_inv[i * D16_P + k] * Updates[2 * D16_P + k]; } } // Compute B = 1 + VC : 3 x 3 const double B0 = C[row1 * 3] + 1; const double B1 = C[row1 * 3 + 1]; const double B2 = C[row1 * 3 + 2]; const double B3 = C[row2 * 3]; const double B4 = C[row2 * 3 + 1] + 1; const double B5 = C[row2 * 3 + 2]; const double B6 = C[row3 * 3]; const double B7 = C[row3 * 3 + 1]; const double B8 = C[row3 * 3 + 2] + 1; // Check if determinant of B is not too close to zero double det; det = B0 * (B4 * B8 - B5 * B7) - B1 * (B3 * B8 - B5 * B6) + B2 * (B3 * B7 - B4 * B6); if (fabs(det) < breakdown) { return QMCKL_FAILURE; } // Update det(Slater) if passed if (determinant) *determinant *= det; // Compute B^{-1} with explicit formula for 3 x 3 inversion double __attribute__((aligned(8))) Binv[9], idet = 1.0 / det; Binv[0] = (B4 * B8 - B7 * B5) * idet; Binv[1] = -(B1 * B8 - B7 * B2) * idet; Binv[2] = (B1 * B5 - B4 * B2) * idet; Binv[3] = -(B3 * B8 - B6 * B5) * idet; Binv[4] = (B0 * B8 - B6 * B2) * idet; Binv[5] = -(B0 * B5 - B3 * B2) * idet; Binv[6] = (B3 * B7 - B6 * B4) * idet; Binv[7] = -(B0 * B7 - B6 * B1) * idet; Binv[8] = (B0 * B4 - B3 * B1) * idet; // tmp = B^{-1}D : 3 x D16_P double __attribute__((aligned(8))) tmp[3 * D16_P]; double* r1dim = &(Slater_inv[row1 * D16_P]); double* r2dim = &(Slater_inv[row2 * D16_P]); double* r3dim = &(Slater_inv[row3 * D16_P]); IVDEP ALIGNED for (uint64_t j = 0; j < D16_P; j++) { tmp[j] = Binv[0] * r1dim[j] + Binv[1] * r2dim[j] + Binv[2] * r3dim[j]; tmp[D16_P + j] = Binv[3] * r1dim[j] + Binv[4] * r2dim[j] + Binv[5] * r3dim[j]; tmp[2 * D16_P + j] = Binv[6] * r1dim[j] + Binv[7] * r2dim[j] + Binv[8] * r3dim[j]; } // Compute (S^T)^{-1} - C * tmp : 16 x D16_P for (uint64_t i = 0; i < 16; i++) { IVDEP ALIGNED for (uint64_t j = 0; j < D16_P; j++) { Slater_inv[i * D16_P + j] -= C[i * 3] * tmp[j]; Slater_inv[i * D16_P + j] -= C[i * 3 + 1] * tmp[D16_P + j]; Slater_inv[i * D16_P + j] -= C[i * 3 + 2] * tmp[2 * D16_P + j]; } } return QMCKL_SUCCESS; } qmckl_exit_code qmckl_woodbury_3x3_17( const qmckl_context context, const double* __restrict Updates, const uint64_t* __restrict Updates_index, const double breakdown, double* __restrict Slater_inv, double* __restrict determinant) { /* C := S^{-1} * U, dim x 3 B := 1 + V * C, 3 x 3 D := V * S^{-1}, 3 x dim */ if (qmckl_context_check(context) == QMCKL_NULL_CONTEXT) { return qmckl_failwith(context, QMCKL_NULL_CONTEXT, "qmckl_woodbury_3x3_17", NULL); } const uint64_t row1 = (Updates_index[0] - 1); const uint64_t row2 = (Updates_index[1] - 1); const uint64_t row3 = (Updates_index[2] - 1); // Compute C = (S^T)^{-1}U : 17 x 3 double __attribute__((aligned(8))) C[3 * 17]; for (uint64_t i = 0; i < 17; i++) { C[i * 3] = 0; C[i * 3 + 1] = 0; C[i * 3 + 2] = 0; IVDEP ALIGNED for (uint64_t k = 0; k < D17_P; k++) { C[i * 3] += Slater_inv[i * D17_P + k] * Updates[k]; C[i * 3 + 1] += Slater_inv[i * D17_P + k] * Updates[D17_P + k]; C[i * 3 + 2] += Slater_inv[i * D17_P + k] * Updates[2 * D17_P + k]; } } // Compute B = 1 + VC : 3 x 3 const double B0 = C[row1 * 3] + 1; const double B1 = C[row1 * 3 + 1]; const double B2 = C[row1 * 3 + 2]; const double B3 = C[row2 * 3]; const double B4 = C[row2 * 3 + 1] + 1; const double B5 = C[row2 * 3 + 2]; const double B6 = C[row3 * 3]; const double B7 = C[row3 * 3 + 1]; const double B8 = C[row3 * 3 + 2] + 1; // Check if determinant of B is not too close to zero double det; det = B0 * (B4 * B8 - B5 * B7) - B1 * (B3 * B8 - B5 * B6) + B2 * (B3 * B7 - B4 * B6); if (fabs(det) < breakdown) { return QMCKL_FAILURE; } // Update det(Slater) if passed if (determinant) *determinant *= det; // Compute B^{-1} with explicit formula for 3 x 3 inversion double __attribute__((aligned(8))) Binv[9], idet = 1.0 / det; Binv[0] = (B4 * B8 - B7 * B5) * idet; Binv[1] = -(B1 * B8 - B7 * B2) * idet; Binv[2] = (B1 * B5 - B4 * B2) * idet; Binv[3] = -(B3 * B8 - B6 * B5) * idet; Binv[4] = (B0 * B8 - B6 * B2) * idet; Binv[5] = -(B0 * B5 - B3 * B2) * idet; Binv[6] = (B3 * B7 - B6 * B4) * idet; Binv[7] = -(B0 * B7 - B6 * B1) * idet; Binv[8] = (B0 * B4 - B3 * B1) * idet; // tmp = B^{-1}D : 3 x D17_P double __attribute__((aligned(8))) tmp[3 * D17_P]; double* r1dim = &(Slater_inv[row1 * D17_P]); double* r2dim = &(Slater_inv[row2 * D17_P]); double* r3dim = &(Slater_inv[row3 * D17_P]); IVDEP ALIGNED for (uint64_t j = 0; j < D17_P; j++) { tmp[j] = Binv[0] * r1dim[j] + Binv[1] * r2dim[j] + Binv[2] * r3dim[j]; tmp[D17_P + j] = Binv[3] * r1dim[j] + Binv[4] * r2dim[j] + Binv[5] * r3dim[j]; tmp[2 * D17_P + j] = Binv[6] * r1dim[j] + Binv[7] * r2dim[j] + Binv[8] * r3dim[j]; } // Compute (S^T)^{-1} - C * tmp : 17 x D17_P for (uint64_t i = 0; i < 17; i++) { IVDEP ALIGNED for (uint64_t j = 0; j < D17_P; j++) { Slater_inv[i * D17_P + j] -= C[i * 3] * tmp[j]; Slater_inv[i * D17_P + j] -= C[i * 3 + 1] * tmp[D17_P + j]; Slater_inv[i * D17_P + j] -= C[i * 3 + 2] * tmp[2 * D17_P + j]; } } return QMCKL_SUCCESS; } qmckl_exit_code qmckl_woodbury_3x3_18( const qmckl_context context, const double* __restrict Updates, const uint64_t* __restrict Updates_index, const double breakdown, double* __restrict Slater_inv, double* __restrict determinant) { /* C := S^{-1} * U, dim x 3 B := 1 + V * C, 3 x 3 D := V * S^{-1}, 3 x dim */ if (qmckl_context_check(context) == QMCKL_NULL_CONTEXT) { return qmckl_failwith(context, QMCKL_NULL_CONTEXT, "qmckl_woodbury_3x3_18", NULL); } const uint64_t row1 = (Updates_index[0] - 1); const uint64_t row2 = (Updates_index[1] - 1); const uint64_t row3 = (Updates_index[2] - 1); // Compute C = (S^T)^{-1}U : 18 x 3 double __attribute__((aligned(8))) C[3 * 18]; for (uint64_t i = 0; i < 18; i++) { C[i * 3] = 0; C[i * 3 + 1] = 0; C[i * 3 + 2] = 0; IVDEP ALIGNED for (uint64_t k = 0; k < D18_P; k++) { C[i * 3] += Slater_inv[i * D18_P + k] * Updates[k]; C[i * 3 + 1] += Slater_inv[i * D18_P + k] * Updates[D18_P + k]; C[i * 3 + 2] += Slater_inv[i * D18_P + k] * Updates[2 * D18_P + k]; } } // Compute B = 1 + VC : 3 x 3 const double B0 = C[row1 * 3] + 1; const double B1 = C[row1 * 3 + 1]; const double B2 = C[row1 * 3 + 2]; const double B3 = C[row2 * 3]; const double B4 = C[row2 * 3 + 1] + 1; const double B5 = C[row2 * 3 + 2]; const double B6 = C[row3 * 3]; const double B7 = C[row3 * 3 + 1]; const double B8 = C[row3 * 3 + 2] + 1; // Check if determinant of B is not too close to zero double det; det = B0 * (B4 * B8 - B5 * B7) - B1 * (B3 * B8 - B5 * B6) + B2 * (B3 * B7 - B4 * B6); if (fabs(det) < breakdown) { return QMCKL_FAILURE; } // Update det(Slater) if passed if (determinant) *determinant *= det; // Compute B^{-1} with explicit formula for 3 x 3 inversion double __attribute__((aligned(8))) Binv[9], idet = 1.0 / det; Binv[0] = (B4 * B8 - B7 * B5) * idet; Binv[1] = -(B1 * B8 - B7 * B2) * idet; Binv[2] = (B1 * B5 - B4 * B2) * idet; Binv[3] = -(B3 * B8 - B6 * B5) * idet; Binv[4] = (B0 * B8 - B6 * B2) * idet; Binv[5] = -(B0 * B5 - B3 * B2) * idet; Binv[6] = (B3 * B7 - B6 * B4) * idet; Binv[7] = -(B0 * B7 - B6 * B1) * idet; Binv[8] = (B0 * B4 - B3 * B1) * idet; // tmp = B^{-1}D : 3 x D18_P double __attribute__((aligned(8))) tmp[3 * D18_P]; double* r1dim = &(Slater_inv[row1 * D18_P]); double* r2dim = &(Slater_inv[row2 * D18_P]); double* r3dim = &(Slater_inv[row3 * D18_P]); IVDEP ALIGNED for (uint64_t j = 0; j < D18_P; j++) { tmp[j] = Binv[0] * r1dim[j] + Binv[1] * r2dim[j] + Binv[2] * r3dim[j]; tmp[D18_P + j] = Binv[3] * r1dim[j] + Binv[4] * r2dim[j] + Binv[5] * r3dim[j]; tmp[2 * D18_P + j] = Binv[6] * r1dim[j] + Binv[7] * r2dim[j] + Binv[8] * r3dim[j]; } // Compute (S^T)^{-1} - C * tmp : 18 x D18_P for (uint64_t i = 0; i < 18; i++) { IVDEP ALIGNED for (uint64_t j = 0; j < D18_P; j++) { Slater_inv[i * D18_P + j] -= C[i * 3] * tmp[j]; Slater_inv[i * D18_P + j] -= C[i * 3 + 1] * tmp[D18_P + j]; Slater_inv[i * D18_P + j] -= C[i * 3 + 2] * tmp[2 * D18_P + j]; } } return QMCKL_SUCCESS; } qmckl_exit_code qmckl_woodbury_3x3_19( const qmckl_context context, const double* __restrict Updates, const uint64_t* __restrict Updates_index, const double breakdown, double* __restrict Slater_inv, double* __restrict determinant) { /* C := S^{-1} * U, dim x 3 B := 1 + V * C, 3 x 3 D := V * S^{-1}, 3 x dim */ if (qmckl_context_check(context) == QMCKL_NULL_CONTEXT) { return qmckl_failwith(context, QMCKL_NULL_CONTEXT, "qmckl_woodbury_3x3_19", NULL); } const uint64_t row1 = (Updates_index[0] - 1); const uint64_t row2 = (Updates_index[1] - 1); const uint64_t row3 = (Updates_index[2] - 1); // Compute C = (S^T)^{-1}U : 19 x 3 double __attribute__((aligned(8))) C[3 * 19]; for (uint64_t i = 0; i < 19; i++) { C[i * 3] = 0; C[i * 3 + 1] = 0; C[i * 3 + 2] = 0; IVDEP ALIGNED for (uint64_t k = 0; k < D19_P; k++) { C[i * 3] += Slater_inv[i * D19_P + k] * Updates[k]; C[i * 3 + 1] += Slater_inv[i * D19_P + k] * Updates[D19_P + k]; C[i * 3 + 2] += Slater_inv[i * D19_P + k] * Updates[2 * D19_P + k]; } } // Compute B = 1 + VC : 3 x 3 const double B0 = C[row1 * 3] + 1; const double B1 = C[row1 * 3 + 1]; const double B2 = C[row1 * 3 + 2]; const double B3 = C[row2 * 3]; const double B4 = C[row2 * 3 + 1] + 1; const double B5 = C[row2 * 3 + 2]; const double B6 = C[row3 * 3]; const double B7 = C[row3 * 3 + 1]; const double B8 = C[row3 * 3 + 2] + 1; // Check if determinant of B is not too close to zero double det; det = B0 * (B4 * B8 - B5 * B7) - B1 * (B3 * B8 - B5 * B6) + B2 * (B3 * B7 - B4 * B6); if (fabs(det) < breakdown) { return QMCKL_FAILURE; } // Update det(Slater) if passed if (determinant) *determinant *= det; // Compute B^{-1} with explicit formula for 3 x 3 inversion double __attribute__((aligned(8))) Binv[9], idet = 1.0 / det; Binv[0] = (B4 * B8 - B7 * B5) * idet; Binv[1] = -(B1 * B8 - B7 * B2) * idet; Binv[2] = (B1 * B5 - B4 * B2) * idet; Binv[3] = -(B3 * B8 - B6 * B5) * idet; Binv[4] = (B0 * B8 - B6 * B2) * idet; Binv[5] = -(B0 * B5 - B3 * B2) * idet; Binv[6] = (B3 * B7 - B6 * B4) * idet; Binv[7] = -(B0 * B7 - B6 * B1) * idet; Binv[8] = (B0 * B4 - B3 * B1) * idet; // tmp = B^{-1}D : 3 x D19_P double __attribute__((aligned(8))) tmp[3 * D19_P]; double* r1dim = &(Slater_inv[row1 * D19_P]); double* r2dim = &(Slater_inv[row2 * D19_P]); double* r3dim = &(Slater_inv[row3 * D19_P]); IVDEP ALIGNED for (uint64_t j = 0; j < D19_P; j++) { tmp[j] = Binv[0] * r1dim[j] + Binv[1] * r2dim[j] + Binv[2] * r3dim[j]; tmp[D19_P + j] = Binv[3] * r1dim[j] + Binv[4] * r2dim[j] + Binv[5] * r3dim[j]; tmp[2 * D19_P + j] = Binv[6] * r1dim[j] + Binv[7] * r2dim[j] + Binv[8] * r3dim[j]; } // Compute (S^T)^{-1} - C * tmp : 19 x D19_P for (uint64_t i = 0; i < 19; i++) { IVDEP ALIGNED for (uint64_t j = 0; j < D19_P; j++) { Slater_inv[i * D19_P + j] -= C[i * 3] * tmp[j]; Slater_inv[i * D19_P + j] -= C[i * 3 + 1] * tmp[D19_P + j]; Slater_inv[i * D19_P + j] -= C[i * 3 + 2] * tmp[2 * D19_P + j]; } } return QMCKL_SUCCESS; } qmckl_exit_code qmckl_woodbury_3x3_20( const qmckl_context context, const double* __restrict Updates, const uint64_t* __restrict Updates_index, const double breakdown, double* __restrict Slater_inv, double* __restrict determinant) { /* C := S^{-1} * U, dim x 3 B := 1 + V * C, 3 x 3 D := V * S^{-1}, 3 x dim */ if (qmckl_context_check(context) == QMCKL_NULL_CONTEXT) { return qmckl_failwith(context, QMCKL_NULL_CONTEXT, "qmckl_woodbury_3x3_20", NULL); } const uint64_t row1 = (Updates_index[0] - 1); const uint64_t row2 = (Updates_index[1] - 1); const uint64_t row3 = (Updates_index[2] - 1); // Compute C = (S^T)^{-1}U : 20 x 3 double __attribute__((aligned(8))) C[3 * 20]; for (uint64_t i = 0; i < 20; i++) { C[i * 3] = 0; C[i * 3 + 1] = 0; C[i * 3 + 2] = 0; IVDEP ALIGNED for (uint64_t k = 0; k < D20_P; k++) { C[i * 3] += Slater_inv[i * D20_P + k] * Updates[k]; C[i * 3 + 1] += Slater_inv[i * D20_P + k] * Updates[D20_P + k]; C[i * 3 + 2] += Slater_inv[i * D20_P + k] * Updates[2 * D20_P + k]; } } // Compute B = 1 + VC : 3 x 3 const double B0 = C[row1 * 3] + 1; const double B1 = C[row1 * 3 + 1]; const double B2 = C[row1 * 3 + 2]; const double B3 = C[row2 * 3]; const double B4 = C[row2 * 3 + 1] + 1; const double B5 = C[row2 * 3 + 2]; const double B6 = C[row3 * 3]; const double B7 = C[row3 * 3 + 1]; const double B8 = C[row3 * 3 + 2] + 1; // Check if determinant of B is not too close to zero double det; det = B0 * (B4 * B8 - B5 * B7) - B1 * (B3 * B8 - B5 * B6) + B2 * (B3 * B7 - B4 * B6); if (fabs(det) < breakdown) { return QMCKL_FAILURE; } // Update det(Slater) if passed if (determinant) *determinant *= det; // Compute B^{-1} with explicit formula for 3 x 3 inversion double __attribute__((aligned(8))) Binv[9], idet = 1.0 / det; Binv[0] = (B4 * B8 - B7 * B5) * idet; Binv[1] = -(B1 * B8 - B7 * B2) * idet; Binv[2] = (B1 * B5 - B4 * B2) * idet; Binv[3] = -(B3 * B8 - B6 * B5) * idet; Binv[4] = (B0 * B8 - B6 * B2) * idet; Binv[5] = -(B0 * B5 - B3 * B2) * idet; Binv[6] = (B3 * B7 - B6 * B4) * idet; Binv[7] = -(B0 * B7 - B6 * B1) * idet; Binv[8] = (B0 * B4 - B3 * B1) * idet; // tmp = B^{-1}D : 3 x D20_P double __attribute__((aligned(8))) tmp[3 * D20_P]; double* r1dim = &(Slater_inv[row1 * D20_P]); double* r2dim = &(Slater_inv[row2 * D20_P]); double* r3dim = &(Slater_inv[row3 * D20_P]); IVDEP ALIGNED for (uint64_t j = 0; j < D20_P; j++) { tmp[j] = Binv[0] * r1dim[j] + Binv[1] * r2dim[j] + Binv[2] * r3dim[j]; tmp[D20_P + j] = Binv[3] * r1dim[j] + Binv[4] * r2dim[j] + Binv[5] * r3dim[j]; tmp[2 * D20_P + j] = Binv[6] * r1dim[j] + Binv[7] * r2dim[j] + Binv[8] * r3dim[j]; } // Compute (S^T)^{-1} - C * tmp : 20 x D20_P for (uint64_t i = 0; i < 20; i++) { IVDEP ALIGNED for (uint64_t j = 0; j < D20_P; j++) { Slater_inv[i * D20_P + j] -= C[i * 3] * tmp[j]; Slater_inv[i * D20_P + j] -= C[i * 3 + 1] * tmp[D20_P + j]; Slater_inv[i * D20_P + j] -= C[i * 3 + 2] * tmp[2 * D20_P + j]; } } return QMCKL_SUCCESS; } qmckl_exit_code qmckl_woodbury_3x3_21( const qmckl_context context, const double* __restrict Updates, const uint64_t* __restrict Updates_index, const double breakdown, double* __restrict Slater_inv, double* __restrict determinant) { /* C := S^{-1} * U, dim x 3 B := 1 + V * C, 3 x 3 D := V * S^{-1}, 3 x dim */ if (qmckl_context_check(context) == QMCKL_NULL_CONTEXT) { return qmckl_failwith(context, QMCKL_NULL_CONTEXT, "qmckl_woodbury_3x3_21", NULL); } const uint64_t row1 = (Updates_index[0] - 1); const uint64_t row2 = (Updates_index[1] - 1); const uint64_t row3 = (Updates_index[2] - 1); // Compute C = (S^T)^{-1}U : 21 x 3 double __attribute__((aligned(8))) C[3 * 21]; for (uint64_t i = 0; i < 21; i++) { C[i * 3] = 0; C[i * 3 + 1] = 0; C[i * 3 + 2] = 0; IVDEP ALIGNED for (uint64_t k = 0; k < D21_P; k++) { C[i * 3] += Slater_inv[i * D21_P + k] * Updates[k]; C[i * 3 + 1] += Slater_inv[i * D21_P + k] * Updates[D21_P + k]; C[i * 3 + 2] += Slater_inv[i * D21_P + k] * Updates[2 * D21_P + k]; } } // Compute B = 1 + VC : 3 x 3 const double B0 = C[row1 * 3] + 1; const double B1 = C[row1 * 3 + 1]; const double B2 = C[row1 * 3 + 2]; const double B3 = C[row2 * 3]; const double B4 = C[row2 * 3 + 1] + 1; const double B5 = C[row2 * 3 + 2]; const double B6 = C[row3 * 3]; const double B7 = C[row3 * 3 + 1]; const double B8 = C[row3 * 3 + 2] + 1; // Check if determinant of B is not too close to zero double det; det = B0 * (B4 * B8 - B5 * B7) - B1 * (B3 * B8 - B5 * B6) + B2 * (B3 * B7 - B4 * B6); if (fabs(det) < breakdown) { return QMCKL_FAILURE; } // Update det(Slater) if passed if (determinant) *determinant *= det; // Compute B^{-1} with explicit formula for 3 x 3 inversion double __attribute__((aligned(8))) Binv[9], idet = 1.0 / det; Binv[0] = (B4 * B8 - B7 * B5) * idet; Binv[1] = -(B1 * B8 - B7 * B2) * idet; Binv[2] = (B1 * B5 - B4 * B2) * idet; Binv[3] = -(B3 * B8 - B6 * B5) * idet; Binv[4] = (B0 * B8 - B6 * B2) * idet; Binv[5] = -(B0 * B5 - B3 * B2) * idet; Binv[6] = (B3 * B7 - B6 * B4) * idet; Binv[7] = -(B0 * B7 - B6 * B1) * idet; Binv[8] = (B0 * B4 - B3 * B1) * idet; // tmp = B^{-1}D : 3 x D21_P double __attribute__((aligned(8))) tmp[3 * D21_P]; double* r1dim = &(Slater_inv[row1 * D21_P]); double* r2dim = &(Slater_inv[row2 * D21_P]); double* r3dim = &(Slater_inv[row3 * D21_P]); IVDEP ALIGNED for (uint64_t j = 0; j < D21_P; j++) { tmp[j] = Binv[0] * r1dim[j] + Binv[1] * r2dim[j] + Binv[2] * r3dim[j]; tmp[D21_P + j] = Binv[3] * r1dim[j] + Binv[4] * r2dim[j] + Binv[5] * r3dim[j]; tmp[2 * D21_P + j] = Binv[6] * r1dim[j] + Binv[7] * r2dim[j] + Binv[8] * r3dim[j]; } // Compute (S^T)^{-1} - C * tmp : 21 x D21_P for (uint64_t i = 0; i < 21; i++) { IVDEP ALIGNED for (uint64_t j = 0; j < D21_P; j++) { Slater_inv[i * D21_P + j] -= C[i * 3] * tmp[j]; Slater_inv[i * D21_P + j] -= C[i * 3 + 1] * tmp[D21_P + j]; Slater_inv[i * D21_P + j] -= C[i * 3 + 2] * tmp[2 * D21_P + j]; } } return QMCKL_SUCCESS; } qmckl_exit_code qmckl_woodbury_3x3(const qmckl_context context, const uint64_t LDS, const uint64_t Dim, const double* Updates, const uint64_t* Updates_index, const double breakdown, double* Slater_inv, double* determinant) { if (qmckl_context_check(context) == QMCKL_NULL_CONTEXT) { return qmckl_failwith(context, QMCKL_NULL_CONTEXT, "qmckl_woodbury_3x3", NULL); } if (LDS == (1+(Dim-1)/SIMD_LENGTH)*SIMD_LENGTH) { // Most cases switch (Dim) { case 2: return qmckl_woodbury_3x3_2(context, Updates, Updates_index, breakdown, Slater_inv, determinant); case 3: return qmckl_woodbury_3x3_3(context, Updates, Updates_index, breakdown, Slater_inv, determinant); case 4: return qmckl_woodbury_3x3_4(context, Updates, Updates_index, breakdown, Slater_inv, determinant); case 5: return qmckl_woodbury_3x3_5(context, Updates, Updates_index, breakdown, Slater_inv, determinant); case 6: return qmckl_woodbury_3x3_6(context, Updates, Updates_index, breakdown, Slater_inv, determinant); case 7: return qmckl_woodbury_3x3_7(context, Updates, Updates_index, breakdown, Slater_inv, determinant); case 8: return qmckl_woodbury_3x3_8(context, Updates, Updates_index, breakdown, Slater_inv, determinant); case 9: return qmckl_woodbury_3x3_9(context, Updates, Updates_index, breakdown, Slater_inv, determinant); case 10: return qmckl_woodbury_3x3_10(context, Updates, Updates_index, breakdown, Slater_inv, determinant); case 11: return qmckl_woodbury_3x3_11(context, Updates, Updates_index, breakdown, Slater_inv, determinant); case 12: return qmckl_woodbury_3x3_12(context, Updates, Updates_index, breakdown, Slater_inv, determinant); case 13: return qmckl_woodbury_3x3_13(context, Updates, Updates_index, breakdown, Slater_inv, determinant); case 14: return qmckl_woodbury_3x3_14(context, Updates, Updates_index, breakdown, Slater_inv, determinant); case 15: return qmckl_woodbury_3x3_15(context, Updates, Updates_index, breakdown, Slater_inv, determinant); case 16: return qmckl_woodbury_3x3_16(context, Updates, Updates_index, breakdown, Slater_inv, determinant); case 17: return qmckl_woodbury_3x3_17(context, Updates, Updates_index, breakdown, Slater_inv, determinant); case 18: return qmckl_woodbury_3x3_18(context, Updates, Updates_index, breakdown, Slater_inv, determinant); case 19: return qmckl_woodbury_3x3_19(context, Updates, Updates_index, breakdown, Slater_inv, determinant); case 20: return qmckl_woodbury_3x3_20(context, Updates, Updates_index, breakdown, Slater_inv, determinant); case 21: return qmckl_woodbury_3x3_21(context, Updates, Updates_index, breakdown, Slater_inv, determinant); } } else { // When SIMD_LENGTH > 1, called with LDS == Dim AND Dim != (1+(Dim-1)/SIMD_LENGTH)*SIMD_LENGTH) return qmckl_woodbury_3x3_hpc(context, LDS, Dim, Updates, Updates_index, breakdown, Slater_inv, determinant); } return QMCKL_FAILURE; }
4.1.4 Performance…
This function is most efficient when used in cases where there are only 3 rank-1 updates and it is sure they will not result in a singular matrix.
5 Sherman-Morrison with update splitting
5.1 qmckl_sherman_morrison_splitting
This is a variation on the 'Naive' Sherman-Morrison kernel. Whenever the denominator \(1+v_j^T S^{-1} u_j\) in the Sherman-Morrison formula is deemed to be too close to zero, the update \(u_j\) is split in half: \(u_j \rightarrow \frac{1}{2} u_j\). One half is applied immediately –necessarily increasing the value of the denominator because of the split– while the other halve is put in a queue that will be applied when all the remaining updates have been treated.
The kernel is executed recursively until the queue is eiter empty and all updates are applied successfully, or the size of the queue equals the number of initial updates. In the last case the Slater-matrix that would have resulted from applying the updates is singular and therefore the kernel exits with an exit code.
If the determinant of the Slater-matrix is passed, it will be updated to the determinant resulting from applying the updates to the original matrix.
qmcklcontext | context | in | Global state |
uint64t | LDS | in | Leading dimension of Slaterinv |
uint64t | Dim | in | Dimension of Slaterinv |
uint64t | Nupdates | in | Number of rank-1 updates to be applied to Slaterinv |
double | Updates[Nupdates*Dim] | in | Array containing the updates |
uint64t | Updatesindex[Nupdates] | in | Array containing the rank-1 updates |
double | breakdown | in | Break-down parameter on which to fail or not |
double | Slaterinv[LDS*Dim] | inout | Array containing the inverse of a Slater-matrix |
double* | determinant | inout | Determinant of the Slater-matrix |
If the determinant of the Slater-matrix is passed, it will be updated to the determinant resulting from applying the updates to the original matrix.
5.1.1 Requirements
context
is notQMCKL_NULL_CONTEXT
LDS >= 2
Dim >= 2
N_updates >= 1
Updates
is allocated with \(N_updates \times Dim\) elementsUpdates_index
is allocated with \(N_updates\) elementsbreakdown
is a small number such that \(0 < breakdown << 1\)Slater_inv
is allocated with \(Dim \times Dim\) elements
5.1.2 C header
qmckl_exit_code qmckl_sherman_morrison_splitting ( const context qmckl_context, const LDS uint64_t, const Dim uint64_t, const N_updates uint64_t, const Updates* double, const Updates_index* uint64_t, const breakdown double, Slater_inv* double, determinant* double* );
5.1.3 C source
qmckl_exit_code qmckl_sherman_morrison_splitting(const qmckl_context context, const uint64_t LDS, const uint64_t Dim, const uint64_t N_updates, const double* Updates, const uint64_t* Updates_index, const double breakdown, double* Slater_inv, double* determinant) { if (qmckl_context_check(context) == QMCKL_NULL_CONTEXT) { return qmckl_failwith(context, QMCKL_NULL_CONTEXT, "qmckl_sherman_morrison_splitting", NULL); } double __attribute__((aligned(8))) later_updates[LDS * N_updates]; uint64_t later_index[N_updates]; uint64_t later = 0; qmckl_exit_code rc = qmckl_slagel_splitting( LDS, Dim, N_updates, Updates, Updates_index, breakdown, Slater_inv, later_updates, later_index, &later, determinant); if (rc != QMCKL_SUCCESS) return QMCKL_FAILURE; if (later > 0) { qmckl_exit_code rc = qmckl_sherman_morrison_splitting( context, LDS, Dim, later, later_updates, later_index, breakdown, Slater_inv, determinant); if (rc != QMCKL_SUCCESS) return QMCKL_FAILURE; } return QMCKL_SUCCESS; }
5.1.4 Performance…
This kernel performs best when there are 2 or more rank-1 update cycles and fail-rate is high.
6 Woodbury 3x3 and 2x2 with Sherman-Morrison and update splitting
6.1 qmckl_sherman_morrison_smw32s
The Woodbury 3x3 and 2x2 kernel with Sherman-Morrison and update splitting combines the low-level Woodbury 3x3 kernel, the Woobury 2x2 kernel and Sherman-Morrison with update splitting. It works the almost the same as Woodbury 3x3 with Sherman-Morrison and update splitting, except that when there is a remainder of two rank-1 updates, it is first tried with Woodbury 2x2 instead of sending them all to Sherman-Morrison with update splitting. For example, in the case of 5 updates the updates are applied in 1 block of 3 updates end 1 block of 2 updates.
If the determinant of the Slater-matrix is passed, it will be updated to the determinant resulting from applying the updates to the original matrix.
qmcklcontext | context | in | Global state |
uint64t | LDS | in | Leading dimension of Slaterinv |
uint64t | Dim | in | Dimension of Slaterinv |
uint64t | Nupdates | in | Number of rank-1 updates to be applied to Slaterinv |
double | Updates[Nupdates*Dim] | in | Array containing the updates |
uint64t | Updatesindex[Nupdates] | in | Array containing the rank-1 updates |
double | breakdown | in | Break-down parameter on which to fail or not |
double | Slaterinv[LDS*Dim] | inout | Array containing the inverse of a Slater-matrix |
double* | determinant | inout | Determinant of the Slater-matrix |
6.1.1 Requirements
context
is notQMCKL_NULL_CONTEXT
LDS >= 2
Dim >= 2
N_updates >= 1
Updates
is allocated with \(N_updates \times Dim\) elementsUpdates_index
is allocated with \(N_updates\) elementsbreakdown
is a small number such that \(0 < breakdown << 1\)Slater_inv
is allocated with \(Dim \times Dim\) elements
6.1.2 C header
qmckl_exit_code qmckl_sherman_morrison_smw32s ( const context qmckl_context, const LDS uint64_t, const Dim uint64_t, const N_updates uint64_t, const Updates* double, const Updates_index* uint64_t, const breakdown double, Slater_inv* double, determinant* double* );
6.1.3 C source
qmckl_exit_code qmckl_sherman_morrison_smw32s(const qmckl_context context, const uint64_t LDS, const uint64_t Dim, const uint64_t N_updates, const double* Updates, const uint64_t* Updates_index, const double breakdown, double* Slater_inv, double* determinant) { if (qmckl_context_check(context) == QMCKL_NULL_CONTEXT) { return qmckl_failwith(context, QMCKL_NULL_CONTEXT, "qmckl_sherman_morrison_smw32s", NULL); } double __attribute__((aligned(8))) later_updates[LDS * N_updates]; uint64_t later_index[N_updates]; uint64_t later = 0; // Special case for 4 rank-1 updates: 2+2 if (N_updates == 4) { qmckl_exit_code rc = qmckl_woodbury_2x2(context, LDS, Dim, Updates, Updates_index, breakdown, Slater_inv, determinant); if (rc != QMCKL_SUCCESS) { // Send the entire block to slagel_splitting uint64_t l = 0; rc = qmckl_slagel_splitting(LDS, Dim, 2, Updates, Updates_index, breakdown, Slater_inv, later_updates + (LDS * later), later_index + later, &l, determinant); later += l; } rc = qmckl_woodbury_2x2(context, LDS, Dim, &Updates[2 * LDS], &Updates_index[2], breakdown, Slater_inv, determinant); if (rc != QMCKL_SUCCESS) { // Send the entire block to slagel_splitting uint64_t l = 0; rc = qmckl_slagel_splitting( LDS, Dim, 2, &Updates[2 * LDS], &Updates_index[2], breakdown, Slater_inv, later_updates + (LDS * later), later_index + later, &l, determinant); later += l; } if (later > 0) { rc = qmckl_sherman_morrison_splitting( context, LDS, Dim, later, later_updates, later_index, breakdown, Slater_inv, determinant); } return QMCKL_SUCCESS; } // And for the other cases != 4 // Apply first 3*n_of_3blocks updates in n_of_3blocks blocks of 3 updates // with Woodbury 3x3 kernel uint64_t n_of_3blocks = N_updates / 3; uint64_t remainder = N_updates % 3; uint64_t length_3block = 3 * LDS; if (n_of_3blocks > 0) { for (uint64_t i = 0; i < n_of_3blocks; i++) { const double* Updates_3block = &Updates[i * length_3block]; const uint64_t* Updates_index_3block = &Updates_index[i * 3]; qmckl_exit_code rc = qmckl_woodbury_3x3( context, LDS, Dim, Updates_3block, Updates_index_3block, breakdown, Slater_inv, determinant); if (rc != QMCKL_SUCCESS) { // Send the entire block to slagel_splitting uint64_t l = 0; rc = qmckl_slagel_splitting( LDS, Dim, 3, Updates_3block, Updates_index_3block, breakdown, Slater_inv, later_updates + (LDS * later), later_index + later, &l, determinant); later += l; } } } // Apply last remaining block of 2 updates with Woodbury 2x2 kernel if (remainder == 2) { const double* Updates_2block = &Updates[n_of_3blocks * length_3block]; const uint64_t* Updates_index_2block = &Updates_index[3 * n_of_3blocks]; qmckl_exit_code rc = qmckl_woodbury_2x2( context, LDS, Dim, Updates_2block, Updates_index_2block, breakdown, Slater_inv, determinant); if (rc != QMCKL_SUCCESS) { // Send the entire block to slagel_splitting uint64_t l = 0; rc = qmckl_slagel_splitting( LDS, Dim, 2, Updates_2block, Updates_index_2block, breakdown, Slater_inv, later_updates + (LDS * later), later_index + later, &l, determinant); later += l; } } // Apply last remaining update with slagel_splitting if (remainder == 1) { const double* Updates_1block = &Updates[n_of_3blocks * length_3block]; const uint64_t* Updates_index_1block = &Updates_index[3 * n_of_3blocks]; uint64_t l = 0; qmckl_exit_code rc = qmckl_slagel_splitting( LDS, Dim, 1, Updates_1block, Updates_index_1block, breakdown, Slater_inv, later_updates + (LDS * later), later_index + later, &l, determinant); if (rc != QMCKL_SUCCESS) return QMCKL_FAILURE; later += l; } if (later > 0) { qmckl_exit_code rc = qmckl_sherman_morrison_splitting( context, LDS, Dim, later, later_updates, later_index, breakdown, Slater_inv, determinant); if (rc != QMCKL_SUCCESS) return QMCKL_FAILURE; } return QMCKL_SUCCESS; }
6.1.4 Performance…
This kernel performs best for update cycles with 2 or more rank-1 updates and the fail-rate is low.
7 Helper Functions
Private helper-functions that are used by the Sherman-Morrison-Woodbury kernels. These functions can only be used internally by the kernels in this module.
7.1 qmckl_slagel_splitting
qmckl_slagel_splitting
is the non-recursive, inner part of the 'Sherman-Morrison with update splitting'-kernel.
It is used internally to apply a collection of \(N\) rank-1 updates to the inverse Slater-matrix \(S^{-1}\) and
splitting an update in two equal pieces if necessary. In case of a split, it applies the first half of the update,
while putting the second half in a waiting queue to be applied at the end.
Therefore, when \(1+v_j^TS^{-1}u_j \geq \epsilon\), the update is applied as usual. Otherwise, \(u_j\) will be redefined as \(\frac{1}{2}u_j\). One half is applied immediately, the other half will be applied at the end of the algorithm, using vectors \(u_{j'}=\frac{1}{2}u_j\) and \(v_{j'}^T=v_{j}^T\), which are stored in the array \texttt{later_updates}.
If the determinant of the Slater-matrix is passed, it will be updated to the determinant resulting from applying the updates to the original matrix.
uint64t | LDS | in | Leading dimension of Slaterinv |
uint64t | Dim | in | Dimension of Slaterinv |
uint64t | Nupdates | in | Number of rank-1 updates to be applied to Slaterinv |
double | Updates[Nupdates*Dim] | in | Array containing the rank-1 updates |
uint64t | Updatesindex[Nupdates] | in | Array containing positions of the rank-1 updates |
double | breakdown | in | Break-down parameter on which to fail or not |
double | Slaterinv[LDS*Dim] | inout | Array containing the inverse Slater-matrix |
double | laterupdates[Dim * Nupdates] | inout | Array containing the split updates for later |
uint64t | laterindex[Nupdates] | inout | Array containing the positions of the split updates for later |
uint64t | later | inout | Number of split updates for later |
double* | determinant | inout | Determinant of the Slater-matrix |
7.1.1 Requirements
LDS >= 2
Dim >= 2
N_updates >= 1
Updates
is allocated with \(N_updates \times Dim\) elementsUpdates_index
is allocated with \(N_updates\) elementsbreakdown
is a small number such that \(0 < breakdown << 1\)Slater_inv
is allocated with \(Dim \times Dim\) elementslater_updates
is allocated with \(later \times Dim\) elementslater_index
is allocated with \(N_updates\) elementslater >= 0
7.1.2 C header
double qmckl_slagel_splitting ( const LDS uint64_t, const Dim uint64_t, const N_updates uint64_t, const Updates* double, const Updates_index* uint64_t, const breakdown double, Slater_inv* double, later_updates* double, later_index* uint64_t, later* uint64_t, determinant* double* );
7.1.3 C source
qmckl_exit_code qmckl_slagel_splitting_hpc( uint64_t LDS, uint64_t Dim, uint64_t N_updates, const double* __restrict Updates, const uint64_t* __restrict Updates_index, const double breakdown, double* __restrict Slater_inv, double* __restrict later_updates, uint64_t* __restrict later_index, uint64_t* __restrict later, double* __restrict determinant) { double __attribute__((aligned(8))) C[LDS]; double __attribute__((aligned(8))) D[LDS]; uint64_t l = 0; // For each update while (l < N_updates) { // C = S^{-1} x U_l for (uint64_t i = 0; i < Dim; i++) { C[i] = 0.0f; IVDEP ALIGNED for (uint64_t j = 0; j < LDS; j++) { C[i] += Slater_inv[i * LDS + j] * Updates[l * LDS + j]; } } // Denominator const int cui = Updates_index[l] - 1; double den = 1.0f + C[cui]; if (fabs(den) < breakdown) { // U_l = U_l / 2: split the update in 2 equal halves and save the // second halve in later_updates IVDEP ALIGNED for (uint64_t i = 0; i < LDS; i++) { later_updates[*later * LDS + i] = Updates[l * LDS + i] * 0.5f; C[i] *= 0.5f; } later_index[*later] = Updates_index[l]; (*later)++; den = 1.0f + C[cui]; } // From here onwards we continue with applying the first halve of the // update to Slater_inv double iden = 1.0f / den; if (determinant) *determinant *= den; // D = v^T x S^{-1} : 1 x LDS IVDEP ALIGNED for (uint64_t j = 0; j < LDS; j++) { D[j] = Slater_inv[cui * LDS + j]; } // S^{-1} = S^{-1} - C x D / den for (uint64_t i = 0; i < Dim; i++) { IVDEP ALIGNED for (uint64_t j = 0; j < LDS; j++) { const double update = C[i] * D[j] * iden; Slater_inv[i * LDS + j] -= update; } } l += 1; } return QMCKL_SUCCESS; }
static inline qmckl_exit_code qmckl_slagel_splitting_{Dim}( uint64_t N_updates, const double* __restrict Updates, const uint64_t* __restrict Updates_index, const double breakdown, double* __restrict Slater_inv, double* __restrict later_updates, uint64_t* __restrict later_index, uint64_t* __restrict later, double* __restrict determinant) { double __attribute__((aligned(8))) C[D{Dim}_P]; double __attribute__((aligned(8))) D[D{Dim}_P]; uint64_t l = 0; // For each update while (l < N_updates) { // C = S^{-1} x U_l for (uint64_t i = 0; i < {Dim}; i++) { C[i] = 0.0f; IVDEP ALIGNED for (uint64_t j = 0; j < D{Dim}_P; j++) { C[i] += Slater_inv[i * D{Dim}_P + j] * Updates[l * D{Dim}_P + j]; } } // Denominator const int cui = Updates_index[l] - 1; double den = 1.0f + C[cui]; if (fabs(den) < breakdown) { // U_l = U_l / 2: split the update in 2 equal halves and save the // second halve in later_updates IVDEP ALIGNED for (uint64_t i = 0; i < D{Dim}_P; i++) { later_updates[*later * D{Dim}_P + i] = Updates[l * D{Dim}_P + i] * 0.5f; C[i] *= 0.5f; } later_index[*later] = Updates_index[l]; (*later)++; den = 1.0f + C[cui]; } // From here onwards we continue with applying the first halve of the // update to Slater_inv double iden = 1.0f / den; if (determinant) *determinant *= den; // D = v^T x S^{-1} : 1 x D{Dim}_P IVDEP ALIGNED for (uint64_t j = 0; j < D{Dim}_P; j++) { D[j] = Slater_inv[cui * D{Dim}_P + j]; } // S^{-1} = S^{-1} - C x D / den for (uint64_t i = 0; i < {Dim}; i++) { IVDEP ALIGNED for (uint64_t j = 0; j < D{Dim}_P; j++) { const double update = C[i] * D[j] * iden; Slater_inv[i * D{Dim}_P + j] -= update; } } l += 1; } return QMCKL_SUCCESS; }
static inline qmckl_exit_code qmckl_slagel_splitting_2( uint64_t N_updates, const double* __restrict Updates, const uint64_t* __restrict Updates_index, const double breakdown, double* __restrict Slater_inv, double* __restrict later_updates, uint64_t* __restrict later_index, uint64_t* __restrict later, double* __restrict determinant) { double __attribute__((aligned(8))) C[D2_P]; double __attribute__((aligned(8))) D[D2_P]; uint64_t l = 0; // For each update while (l < N_updates) { // C = S^{-1} x U_l for (uint64_t i = 0; i < 2; i++) { C[i] = 0.0f; IVDEP ALIGNED for (uint64_t j = 0; j < D2_P; j++) { C[i] += Slater_inv[i * D2_P + j] * Updates[l * D2_P + j]; } } // Denominator const int cui = Updates_index[l] - 1; double den = 1.0f + C[cui]; if (fabs(den) < breakdown) { // U_l = U_l / 2: split the update in 2 equal halves and save the // second halve in later_updates IVDEP ALIGNED for (uint64_t i = 0; i < D2_P; i++) { later_updates[*later * D2_P + i] = Updates[l * D2_P + i] * 0.5f; C[i] *= 0.5f; } later_index[*later] = Updates_index[l]; (*later)++; den = 1.0f + C[cui]; } // From here onwards we continue with applying the first halve of the // update to Slater_inv double iden = 1.0f / den; if (determinant) *determinant *= den; // D = v^T x S^{-1} : 1 x D2_P IVDEP ALIGNED for (uint64_t j = 0; j < D2_P; j++) { D[j] = Slater_inv[cui * D2_P + j]; } // S^{-1} = S^{-1} - C x D / den for (uint64_t i = 0; i < 2; i++) { IVDEP ALIGNED for (uint64_t j = 0; j < D2_P; j++) { const double update = C[i] * D[j] * iden; Slater_inv[i * D2_P + j] -= update; } } l += 1; } return QMCKL_SUCCESS; } static inline qmckl_exit_code qmckl_slagel_splitting_3( uint64_t N_updates, const double* __restrict Updates, const uint64_t* __restrict Updates_index, const double breakdown, double* __restrict Slater_inv, double* __restrict later_updates, uint64_t* __restrict later_index, uint64_t* __restrict later, double* __restrict determinant) { double __attribute__((aligned(8))) C[D3_P]; double __attribute__((aligned(8))) D[D3_P]; uint64_t l = 0; // For each update while (l < N_updates) { // C = S^{-1} x U_l for (uint64_t i = 0; i < 3; i++) { C[i] = 0.0f; IVDEP ALIGNED for (uint64_t j = 0; j < D3_P; j++) { C[i] += Slater_inv[i * D3_P + j] * Updates[l * D3_P + j]; } } // Denominator const int cui = Updates_index[l] - 1; double den = 1.0f + C[cui]; if (fabs(den) < breakdown) { // U_l = U_l / 2: split the update in 2 equal halves and save the // second halve in later_updates IVDEP ALIGNED for (uint64_t i = 0; i < D3_P; i++) { later_updates[*later * D3_P + i] = Updates[l * D3_P + i] * 0.5f; C[i] *= 0.5f; } later_index[*later] = Updates_index[l]; (*later)++; den = 1.0f + C[cui]; } // From here onwards we continue with applying the first halve of the // update to Slater_inv double iden = 1.0f / den; if (determinant) *determinant *= den; // D = v^T x S^{-1} : 1 x D3_P IVDEP ALIGNED for (uint64_t j = 0; j < D3_P; j++) { D[j] = Slater_inv[cui * D3_P + j]; } // S^{-1} = S^{-1} - C x D / den for (uint64_t i = 0; i < 3; i++) { IVDEP ALIGNED for (uint64_t j = 0; j < D3_P; j++) { const double update = C[i] * D[j] * iden; Slater_inv[i * D3_P + j] -= update; } } l += 1; } return QMCKL_SUCCESS; } static inline qmckl_exit_code qmckl_slagel_splitting_4( uint64_t N_updates, const double* __restrict Updates, const uint64_t* __restrict Updates_index, const double breakdown, double* __restrict Slater_inv, double* __restrict later_updates, uint64_t* __restrict later_index, uint64_t* __restrict later, double* __restrict determinant) { double __attribute__((aligned(8))) C[D4_P]; double __attribute__((aligned(8))) D[D4_P]; uint64_t l = 0; // For each update while (l < N_updates) { // C = S^{-1} x U_l for (uint64_t i = 0; i < 4; i++) { C[i] = 0.0f; IVDEP ALIGNED for (uint64_t j = 0; j < D4_P; j++) { C[i] += Slater_inv[i * D4_P + j] * Updates[l * D4_P + j]; } } // Denominator const int cui = Updates_index[l] - 1; double den = 1.0f + C[cui]; if (fabs(den) < breakdown) { // U_l = U_l / 2: split the update in 2 equal halves and save the // second halve in later_updates IVDEP ALIGNED for (uint64_t i = 0; i < D4_P; i++) { later_updates[*later * D4_P + i] = Updates[l * D4_P + i] * 0.5f; C[i] *= 0.5f; } later_index[*later] = Updates_index[l]; (*later)++; den = 1.0f + C[cui]; } // From here onwards we continue with applying the first halve of the // update to Slater_inv double iden = 1.0f / den; if (determinant) *determinant *= den; // D = v^T x S^{-1} : 1 x D4_P IVDEP ALIGNED for (uint64_t j = 0; j < D4_P; j++) { D[j] = Slater_inv[cui * D4_P + j]; } // S^{-1} = S^{-1} - C x D / den for (uint64_t i = 0; i < 4; i++) { IVDEP ALIGNED for (uint64_t j = 0; j < D4_P; j++) { const double update = C[i] * D[j] * iden; Slater_inv[i * D4_P + j] -= update; } } l += 1; } return QMCKL_SUCCESS; } static inline qmckl_exit_code qmckl_slagel_splitting_5( uint64_t N_updates, const double* __restrict Updates, const uint64_t* __restrict Updates_index, const double breakdown, double* __restrict Slater_inv, double* __restrict later_updates, uint64_t* __restrict later_index, uint64_t* __restrict later, double* __restrict determinant) { double __attribute__((aligned(8))) C[D5_P]; double __attribute__((aligned(8))) D[D5_P]; uint64_t l = 0; // For each update while (l < N_updates) { // C = S^{-1} x U_l for (uint64_t i = 0; i < 5; i++) { C[i] = 0.0f; IVDEP ALIGNED for (uint64_t j = 0; j < D5_P; j++) { C[i] += Slater_inv[i * D5_P + j] * Updates[l * D5_P + j]; } } // Denominator const int cui = Updates_index[l] - 1; double den = 1.0f + C[cui]; if (fabs(den) < breakdown) { // U_l = U_l / 2: split the update in 2 equal halves and save the // second halve in later_updates IVDEP ALIGNED for (uint64_t i = 0; i < D5_P; i++) { later_updates[*later * D5_P + i] = Updates[l * D5_P + i] * 0.5f; C[i] *= 0.5f; } later_index[*later] = Updates_index[l]; (*later)++; den = 1.0f + C[cui]; } // From here onwards we continue with applying the first halve of the // update to Slater_inv double iden = 1.0f / den; if (determinant) *determinant *= den; // D = v^T x S^{-1} : 1 x D5_P IVDEP ALIGNED for (uint64_t j = 0; j < D5_P; j++) { D[j] = Slater_inv[cui * D5_P + j]; } // S^{-1} = S^{-1} - C x D / den for (uint64_t i = 0; i < 5; i++) { IVDEP ALIGNED for (uint64_t j = 0; j < D5_P; j++) { const double update = C[i] * D[j] * iden; Slater_inv[i * D5_P + j] -= update; } } l += 1; } return QMCKL_SUCCESS; } static inline qmckl_exit_code qmckl_slagel_splitting_6( uint64_t N_updates, const double* __restrict Updates, const uint64_t* __restrict Updates_index, const double breakdown, double* __restrict Slater_inv, double* __restrict later_updates, uint64_t* __restrict later_index, uint64_t* __restrict later, double* __restrict determinant) { double __attribute__((aligned(8))) C[D6_P]; double __attribute__((aligned(8))) D[D6_P]; uint64_t l = 0; // For each update while (l < N_updates) { // C = S^{-1} x U_l for (uint64_t i = 0; i < 6; i++) { C[i] = 0.0f; IVDEP ALIGNED for (uint64_t j = 0; j < D6_P; j++) { C[i] += Slater_inv[i * D6_P + j] * Updates[l * D6_P + j]; } } // Denominator const int cui = Updates_index[l] - 1; double den = 1.0f + C[cui]; if (fabs(den) < breakdown) { // U_l = U_l / 2: split the update in 2 equal halves and save the // second halve in later_updates IVDEP ALIGNED for (uint64_t i = 0; i < D6_P; i++) { later_updates[*later * D6_P + i] = Updates[l * D6_P + i] * 0.5f; C[i] *= 0.5f; } later_index[*later] = Updates_index[l]; (*later)++; den = 1.0f + C[cui]; } // From here onwards we continue with applying the first halve of the // update to Slater_inv double iden = 1.0f / den; if (determinant) *determinant *= den; // D = v^T x S^{-1} : 1 x D6_P IVDEP ALIGNED for (uint64_t j = 0; j < D6_P; j++) { D[j] = Slater_inv[cui * D6_P + j]; } // S^{-1} = S^{-1} - C x D / den for (uint64_t i = 0; i < 6; i++) { IVDEP ALIGNED for (uint64_t j = 0; j < D6_P; j++) { const double update = C[i] * D[j] * iden; Slater_inv[i * D6_P + j] -= update; } } l += 1; } return QMCKL_SUCCESS; } static inline qmckl_exit_code qmckl_slagel_splitting_7( uint64_t N_updates, const double* __restrict Updates, const uint64_t* __restrict Updates_index, const double breakdown, double* __restrict Slater_inv, double* __restrict later_updates, uint64_t* __restrict later_index, uint64_t* __restrict later, double* __restrict determinant) { double __attribute__((aligned(8))) C[D7_P]; double __attribute__((aligned(8))) D[D7_P]; uint64_t l = 0; // For each update while (l < N_updates) { // C = S^{-1} x U_l for (uint64_t i = 0; i < 7; i++) { C[i] = 0.0f; IVDEP ALIGNED for (uint64_t j = 0; j < D7_P; j++) { C[i] += Slater_inv[i * D7_P + j] * Updates[l * D7_P + j]; } } // Denominator const int cui = Updates_index[l] - 1; double den = 1.0f + C[cui]; if (fabs(den) < breakdown) { // U_l = U_l / 2: split the update in 2 equal halves and save the // second halve in later_updates IVDEP ALIGNED for (uint64_t i = 0; i < D7_P; i++) { later_updates[*later * D7_P + i] = Updates[l * D7_P + i] * 0.5f; C[i] *= 0.5f; } later_index[*later] = Updates_index[l]; (*later)++; den = 1.0f + C[cui]; } // From here onwards we continue with applying the first halve of the // update to Slater_inv double iden = 1.0f / den; if (determinant) *determinant *= den; // D = v^T x S^{-1} : 1 x D7_P IVDEP ALIGNED for (uint64_t j = 0; j < D7_P; j++) { D[j] = Slater_inv[cui * D7_P + j]; } // S^{-1} = S^{-1} - C x D / den for (uint64_t i = 0; i < 7; i++) { IVDEP ALIGNED for (uint64_t j = 0; j < D7_P; j++) { const double update = C[i] * D[j] * iden; Slater_inv[i * D7_P + j] -= update; } } l += 1; } return QMCKL_SUCCESS; } static inline qmckl_exit_code qmckl_slagel_splitting_8( uint64_t N_updates, const double* __restrict Updates, const uint64_t* __restrict Updates_index, const double breakdown, double* __restrict Slater_inv, double* __restrict later_updates, uint64_t* __restrict later_index, uint64_t* __restrict later, double* __restrict determinant) { double __attribute__((aligned(8))) C[D8_P]; double __attribute__((aligned(8))) D[D8_P]; uint64_t l = 0; // For each update while (l < N_updates) { // C = S^{-1} x U_l for (uint64_t i = 0; i < 8; i++) { C[i] = 0.0f; IVDEP ALIGNED for (uint64_t j = 0; j < D8_P; j++) { C[i] += Slater_inv[i * D8_P + j] * Updates[l * D8_P + j]; } } // Denominator const int cui = Updates_index[l] - 1; double den = 1.0f + C[cui]; if (fabs(den) < breakdown) { // U_l = U_l / 2: split the update in 2 equal halves and save the // second halve in later_updates IVDEP ALIGNED for (uint64_t i = 0; i < D8_P; i++) { later_updates[*later * D8_P + i] = Updates[l * D8_P + i] * 0.5f; C[i] *= 0.5f; } later_index[*later] = Updates_index[l]; (*later)++; den = 1.0f + C[cui]; } // From here onwards we continue with applying the first halve of the // update to Slater_inv double iden = 1.0f / den; if (determinant) *determinant *= den; // D = v^T x S^{-1} : 1 x D8_P IVDEP ALIGNED for (uint64_t j = 0; j < D8_P; j++) { D[j] = Slater_inv[cui * D8_P + j]; } // S^{-1} = S^{-1} - C x D / den for (uint64_t i = 0; i < 8; i++) { IVDEP ALIGNED for (uint64_t j = 0; j < D8_P; j++) { const double update = C[i] * D[j] * iden; Slater_inv[i * D8_P + j] -= update; } } l += 1; } return QMCKL_SUCCESS; } static inline qmckl_exit_code qmckl_slagel_splitting_9( uint64_t N_updates, const double* __restrict Updates, const uint64_t* __restrict Updates_index, const double breakdown, double* __restrict Slater_inv, double* __restrict later_updates, uint64_t* __restrict later_index, uint64_t* __restrict later, double* __restrict determinant) { double __attribute__((aligned(8))) C[D9_P]; double __attribute__((aligned(8))) D[D9_P]; uint64_t l = 0; // For each update while (l < N_updates) { // C = S^{-1} x U_l for (uint64_t i = 0; i < 9; i++) { C[i] = 0.0f; IVDEP ALIGNED for (uint64_t j = 0; j < D9_P; j++) { C[i] += Slater_inv[i * D9_P + j] * Updates[l * D9_P + j]; } } // Denominator const int cui = Updates_index[l] - 1; double den = 1.0f + C[cui]; if (fabs(den) < breakdown) { // U_l = U_l / 2: split the update in 2 equal halves and save the // second halve in later_updates IVDEP ALIGNED for (uint64_t i = 0; i < D9_P; i++) { later_updates[*later * D9_P + i] = Updates[l * D9_P + i] * 0.5f; C[i] *= 0.5f; } later_index[*later] = Updates_index[l]; (*later)++; den = 1.0f + C[cui]; } // From here onwards we continue with applying the first halve of the // update to Slater_inv double iden = 1.0f / den; if (determinant) *determinant *= den; // D = v^T x S^{-1} : 1 x D9_P IVDEP ALIGNED for (uint64_t j = 0; j < D9_P; j++) { D[j] = Slater_inv[cui * D9_P + j]; } // S^{-1} = S^{-1} - C x D / den for (uint64_t i = 0; i < 9; i++) { IVDEP ALIGNED for (uint64_t j = 0; j < D9_P; j++) { const double update = C[i] * D[j] * iden; Slater_inv[i * D9_P + j] -= update; } } l += 1; } return QMCKL_SUCCESS; } static inline qmckl_exit_code qmckl_slagel_splitting_10( uint64_t N_updates, const double* __restrict Updates, const uint64_t* __restrict Updates_index, const double breakdown, double* __restrict Slater_inv, double* __restrict later_updates, uint64_t* __restrict later_index, uint64_t* __restrict later, double* __restrict determinant) { double __attribute__((aligned(8))) C[D10_P]; double __attribute__((aligned(8))) D[D10_P]; uint64_t l = 0; // For each update while (l < N_updates) { // C = S^{-1} x U_l for (uint64_t i = 0; i < 10; i++) { C[i] = 0.0f; IVDEP ALIGNED for (uint64_t j = 0; j < D10_P; j++) { C[i] += Slater_inv[i * D10_P + j] * Updates[l * D10_P + j]; } } // Denominator const int cui = Updates_index[l] - 1; double den = 1.0f + C[cui]; if (fabs(den) < breakdown) { // U_l = U_l / 2: split the update in 2 equal halves and save the // second halve in later_updates IVDEP ALIGNED for (uint64_t i = 0; i < D10_P; i++) { later_updates[*later * D10_P + i] = Updates[l * D10_P + i] * 0.5f; C[i] *= 0.5f; } later_index[*later] = Updates_index[l]; (*later)++; den = 1.0f + C[cui]; } // From here onwards we continue with applying the first halve of the // update to Slater_inv double iden = 1.0f / den; if (determinant) *determinant *= den; // D = v^T x S^{-1} : 1 x D10_P IVDEP ALIGNED for (uint64_t j = 0; j < D10_P; j++) { D[j] = Slater_inv[cui * D10_P + j]; } // S^{-1} = S^{-1} - C x D / den for (uint64_t i = 0; i < 10; i++) { IVDEP ALIGNED for (uint64_t j = 0; j < D10_P; j++) { const double update = C[i] * D[j] * iden; Slater_inv[i * D10_P + j] -= update; } } l += 1; } return QMCKL_SUCCESS; } static inline qmckl_exit_code qmckl_slagel_splitting_11( uint64_t N_updates, const double* __restrict Updates, const uint64_t* __restrict Updates_index, const double breakdown, double* __restrict Slater_inv, double* __restrict later_updates, uint64_t* __restrict later_index, uint64_t* __restrict later, double* __restrict determinant) { double __attribute__((aligned(8))) C[D11_P]; double __attribute__((aligned(8))) D[D11_P]; uint64_t l = 0; // For each update while (l < N_updates) { // C = S^{-1} x U_l for (uint64_t i = 0; i < 11; i++) { C[i] = 0.0f; IVDEP ALIGNED for (uint64_t j = 0; j < D11_P; j++) { C[i] += Slater_inv[i * D11_P + j] * Updates[l * D11_P + j]; } } // Denominator const int cui = Updates_index[l] - 1; double den = 1.0f + C[cui]; if (fabs(den) < breakdown) { // U_l = U_l / 2: split the update in 2 equal halves and save the // second halve in later_updates IVDEP ALIGNED for (uint64_t i = 0; i < D11_P; i++) { later_updates[*later * D11_P + i] = Updates[l * D11_P + i] * 0.5f; C[i] *= 0.5f; } later_index[*later] = Updates_index[l]; (*later)++; den = 1.0f + C[cui]; } // From here onwards we continue with applying the first halve of the // update to Slater_inv double iden = 1.0f / den; if (determinant) *determinant *= den; // D = v^T x S^{-1} : 1 x D11_P IVDEP ALIGNED for (uint64_t j = 0; j < D11_P; j++) { D[j] = Slater_inv[cui * D11_P + j]; } // S^{-1} = S^{-1} - C x D / den for (uint64_t i = 0; i < 11; i++) { IVDEP ALIGNED for (uint64_t j = 0; j < D11_P; j++) { const double update = C[i] * D[j] * iden; Slater_inv[i * D11_P + j] -= update; } } l += 1; } return QMCKL_SUCCESS; } static inline qmckl_exit_code qmckl_slagel_splitting_12( uint64_t N_updates, const double* __restrict Updates, const uint64_t* __restrict Updates_index, const double breakdown, double* __restrict Slater_inv, double* __restrict later_updates, uint64_t* __restrict later_index, uint64_t* __restrict later, double* __restrict determinant) { double __attribute__((aligned(8))) C[D12_P]; double __attribute__((aligned(8))) D[D12_P]; uint64_t l = 0; // For each update while (l < N_updates) { // C = S^{-1} x U_l for (uint64_t i = 0; i < 12; i++) { C[i] = 0.0f; IVDEP ALIGNED for (uint64_t j = 0; j < D12_P; j++) { C[i] += Slater_inv[i * D12_P + j] * Updates[l * D12_P + j]; } } // Denominator const int cui = Updates_index[l] - 1; double den = 1.0f + C[cui]; if (fabs(den) < breakdown) { // U_l = U_l / 2: split the update in 2 equal halves and save the // second halve in later_updates IVDEP ALIGNED for (uint64_t i = 0; i < D12_P; i++) { later_updates[*later * D12_P + i] = Updates[l * D12_P + i] * 0.5f; C[i] *= 0.5f; } later_index[*later] = Updates_index[l]; (*later)++; den = 1.0f + C[cui]; } // From here onwards we continue with applying the first halve of the // update to Slater_inv double iden = 1.0f / den; if (determinant) *determinant *= den; // D = v^T x S^{-1} : 1 x D12_P IVDEP ALIGNED for (uint64_t j = 0; j < D12_P; j++) { D[j] = Slater_inv[cui * D12_P + j]; } // S^{-1} = S^{-1} - C x D / den for (uint64_t i = 0; i < 12; i++) { IVDEP ALIGNED for (uint64_t j = 0; j < D12_P; j++) { const double update = C[i] * D[j] * iden; Slater_inv[i * D12_P + j] -= update; } } l += 1; } return QMCKL_SUCCESS; } static inline qmckl_exit_code qmckl_slagel_splitting_13( uint64_t N_updates, const double* __restrict Updates, const uint64_t* __restrict Updates_index, const double breakdown, double* __restrict Slater_inv, double* __restrict later_updates, uint64_t* __restrict later_index, uint64_t* __restrict later, double* __restrict determinant) { double __attribute__((aligned(8))) C[D13_P]; double __attribute__((aligned(8))) D[D13_P]; uint64_t l = 0; // For each update while (l < N_updates) { // C = S^{-1} x U_l for (uint64_t i = 0; i < 13; i++) { C[i] = 0.0f; IVDEP ALIGNED for (uint64_t j = 0; j < D13_P; j++) { C[i] += Slater_inv[i * D13_P + j] * Updates[l * D13_P + j]; } } // Denominator const int cui = Updates_index[l] - 1; double den = 1.0f + C[cui]; if (fabs(den) < breakdown) { // U_l = U_l / 2: split the update in 2 equal halves and save the // second halve in later_updates IVDEP ALIGNED for (uint64_t i = 0; i < D13_P; i++) { later_updates[*later * D13_P + i] = Updates[l * D13_P + i] * 0.5f; C[i] *= 0.5f; } later_index[*later] = Updates_index[l]; (*later)++; den = 1.0f + C[cui]; } // From here onwards we continue with applying the first halve of the // update to Slater_inv double iden = 1.0f / den; if (determinant) *determinant *= den; // D = v^T x S^{-1} : 1 x D13_P IVDEP ALIGNED for (uint64_t j = 0; j < D13_P; j++) { D[j] = Slater_inv[cui * D13_P + j]; } // S^{-1} = S^{-1} - C x D / den for (uint64_t i = 0; i < 13; i++) { IVDEP ALIGNED for (uint64_t j = 0; j < D13_P; j++) { const double update = C[i] * D[j] * iden; Slater_inv[i * D13_P + j] -= update; } } l += 1; } return QMCKL_SUCCESS; } static inline qmckl_exit_code qmckl_slagel_splitting_14( uint64_t N_updates, const double* __restrict Updates, const uint64_t* __restrict Updates_index, const double breakdown, double* __restrict Slater_inv, double* __restrict later_updates, uint64_t* __restrict later_index, uint64_t* __restrict later, double* __restrict determinant) { double __attribute__((aligned(8))) C[D14_P]; double __attribute__((aligned(8))) D[D14_P]; uint64_t l = 0; // For each update while (l < N_updates) { // C = S^{-1} x U_l for (uint64_t i = 0; i < 14; i++) { C[i] = 0.0f; IVDEP ALIGNED for (uint64_t j = 0; j < D14_P; j++) { C[i] += Slater_inv[i * D14_P + j] * Updates[l * D14_P + j]; } } // Denominator const int cui = Updates_index[l] - 1; double den = 1.0f + C[cui]; if (fabs(den) < breakdown) { // U_l = U_l / 2: split the update in 2 equal halves and save the // second halve in later_updates IVDEP ALIGNED for (uint64_t i = 0; i < D14_P; i++) { later_updates[*later * D14_P + i] = Updates[l * D14_P + i] * 0.5f; C[i] *= 0.5f; } later_index[*later] = Updates_index[l]; (*later)++; den = 1.0f + C[cui]; } // From here onwards we continue with applying the first halve of the // update to Slater_inv double iden = 1.0f / den; if (determinant) *determinant *= den; // D = v^T x S^{-1} : 1 x D14_P IVDEP ALIGNED for (uint64_t j = 0; j < D14_P; j++) { D[j] = Slater_inv[cui * D14_P + j]; } // S^{-1} = S^{-1} - C x D / den for (uint64_t i = 0; i < 14; i++) { IVDEP ALIGNED for (uint64_t j = 0; j < D14_P; j++) { const double update = C[i] * D[j] * iden; Slater_inv[i * D14_P + j] -= update; } } l += 1; } return QMCKL_SUCCESS; } static inline qmckl_exit_code qmckl_slagel_splitting_15( uint64_t N_updates, const double* __restrict Updates, const uint64_t* __restrict Updates_index, const double breakdown, double* __restrict Slater_inv, double* __restrict later_updates, uint64_t* __restrict later_index, uint64_t* __restrict later, double* __restrict determinant) { double __attribute__((aligned(8))) C[D15_P]; double __attribute__((aligned(8))) D[D15_P]; uint64_t l = 0; // For each update while (l < N_updates) { // C = S^{-1} x U_l for (uint64_t i = 0; i < 15; i++) { C[i] = 0.0f; IVDEP ALIGNED for (uint64_t j = 0; j < D15_P; j++) { C[i] += Slater_inv[i * D15_P + j] * Updates[l * D15_P + j]; } } // Denominator const int cui = Updates_index[l] - 1; double den = 1.0f + C[cui]; if (fabs(den) < breakdown) { // U_l = U_l / 2: split the update in 2 equal halves and save the // second halve in later_updates IVDEP ALIGNED for (uint64_t i = 0; i < D15_P; i++) { later_updates[*later * D15_P + i] = Updates[l * D15_P + i] * 0.5f; C[i] *= 0.5f; } later_index[*later] = Updates_index[l]; (*later)++; den = 1.0f + C[cui]; } // From here onwards we continue with applying the first halve of the // update to Slater_inv double iden = 1.0f / den; if (determinant) *determinant *= den; // D = v^T x S^{-1} : 1 x D15_P IVDEP ALIGNED for (uint64_t j = 0; j < D15_P; j++) { D[j] = Slater_inv[cui * D15_P + j]; } // S^{-1} = S^{-1} - C x D / den for (uint64_t i = 0; i < 15; i++) { IVDEP ALIGNED for (uint64_t j = 0; j < D15_P; j++) { const double update = C[i] * D[j] * iden; Slater_inv[i * D15_P + j] -= update; } } l += 1; } return QMCKL_SUCCESS; } static inline qmckl_exit_code qmckl_slagel_splitting_16( uint64_t N_updates, const double* __restrict Updates, const uint64_t* __restrict Updates_index, const double breakdown, double* __restrict Slater_inv, double* __restrict later_updates, uint64_t* __restrict later_index, uint64_t* __restrict later, double* __restrict determinant) { double __attribute__((aligned(8))) C[D16_P]; double __attribute__((aligned(8))) D[D16_P]; uint64_t l = 0; // For each update while (l < N_updates) { // C = S^{-1} x U_l for (uint64_t i = 0; i < 16; i++) { C[i] = 0.0f; IVDEP ALIGNED for (uint64_t j = 0; j < D16_P; j++) { C[i] += Slater_inv[i * D16_P + j] * Updates[l * D16_P + j]; } } // Denominator const int cui = Updates_index[l] - 1; double den = 1.0f + C[cui]; if (fabs(den) < breakdown) { // U_l = U_l / 2: split the update in 2 equal halves and save the // second halve in later_updates IVDEP ALIGNED for (uint64_t i = 0; i < D16_P; i++) { later_updates[*later * D16_P + i] = Updates[l * D16_P + i] * 0.5f; C[i] *= 0.5f; } later_index[*later] = Updates_index[l]; (*later)++; den = 1.0f + C[cui]; } // From here onwards we continue with applying the first halve of the // update to Slater_inv double iden = 1.0f / den; if (determinant) *determinant *= den; // D = v^T x S^{-1} : 1 x D16_P IVDEP ALIGNED for (uint64_t j = 0; j < D16_P; j++) { D[j] = Slater_inv[cui * D16_P + j]; } // S^{-1} = S^{-1} - C x D / den for (uint64_t i = 0; i < 16; i++) { IVDEP ALIGNED for (uint64_t j = 0; j < D16_P; j++) { const double update = C[i] * D[j] * iden; Slater_inv[i * D16_P + j] -= update; } } l += 1; } return QMCKL_SUCCESS; } static inline qmckl_exit_code qmckl_slagel_splitting_17( uint64_t N_updates, const double* __restrict Updates, const uint64_t* __restrict Updates_index, const double breakdown, double* __restrict Slater_inv, double* __restrict later_updates, uint64_t* __restrict later_index, uint64_t* __restrict later, double* __restrict determinant) { double __attribute__((aligned(8))) C[D17_P]; double __attribute__((aligned(8))) D[D17_P]; uint64_t l = 0; // For each update while (l < N_updates) { // C = S^{-1} x U_l for (uint64_t i = 0; i < 17; i++) { C[i] = 0.0f; IVDEP ALIGNED for (uint64_t j = 0; j < D17_P; j++) { C[i] += Slater_inv[i * D17_P + j] * Updates[l * D17_P + j]; } } // Denominator const int cui = Updates_index[l] - 1; double den = 1.0f + C[cui]; if (fabs(den) < breakdown) { // U_l = U_l / 2: split the update in 2 equal halves and save the // second halve in later_updates IVDEP ALIGNED for (uint64_t i = 0; i < D17_P; i++) { later_updates[*later * D17_P + i] = Updates[l * D17_P + i] * 0.5f; C[i] *= 0.5f; } later_index[*later] = Updates_index[l]; (*later)++; den = 1.0f + C[cui]; } // From here onwards we continue with applying the first halve of the // update to Slater_inv double iden = 1.0f / den; if (determinant) *determinant *= den; // D = v^T x S^{-1} : 1 x D17_P IVDEP ALIGNED for (uint64_t j = 0; j < D17_P; j++) { D[j] = Slater_inv[cui * D17_P + j]; } // S^{-1} = S^{-1} - C x D / den for (uint64_t i = 0; i < 17; i++) { IVDEP ALIGNED for (uint64_t j = 0; j < D17_P; j++) { const double update = C[i] * D[j] * iden; Slater_inv[i * D17_P + j] -= update; } } l += 1; } return QMCKL_SUCCESS; } static inline qmckl_exit_code qmckl_slagel_splitting_18( uint64_t N_updates, const double* __restrict Updates, const uint64_t* __restrict Updates_index, const double breakdown, double* __restrict Slater_inv, double* __restrict later_updates, uint64_t* __restrict later_index, uint64_t* __restrict later, double* __restrict determinant) { double __attribute__((aligned(8))) C[D18_P]; double __attribute__((aligned(8))) D[D18_P]; uint64_t l = 0; // For each update while (l < N_updates) { // C = S^{-1} x U_l for (uint64_t i = 0; i < 18; i++) { C[i] = 0.0f; IVDEP ALIGNED for (uint64_t j = 0; j < D18_P; j++) { C[i] += Slater_inv[i * D18_P + j] * Updates[l * D18_P + j]; } } // Denominator const int cui = Updates_index[l] - 1; double den = 1.0f + C[cui]; if (fabs(den) < breakdown) { // U_l = U_l / 2: split the update in 2 equal halves and save the // second halve in later_updates IVDEP ALIGNED for (uint64_t i = 0; i < D18_P; i++) { later_updates[*later * D18_P + i] = Updates[l * D18_P + i] * 0.5f; C[i] *= 0.5f; } later_index[*later] = Updates_index[l]; (*later)++; den = 1.0f + C[cui]; } // From here onwards we continue with applying the first halve of the // update to Slater_inv double iden = 1.0f / den; if (determinant) *determinant *= den; // D = v^T x S^{-1} : 1 x D18_P IVDEP ALIGNED for (uint64_t j = 0; j < D18_P; j++) { D[j] = Slater_inv[cui * D18_P + j]; } // S^{-1} = S^{-1} - C x D / den for (uint64_t i = 0; i < 18; i++) { IVDEP ALIGNED for (uint64_t j = 0; j < D18_P; j++) { const double update = C[i] * D[j] * iden; Slater_inv[i * D18_P + j] -= update; } } l += 1; } return QMCKL_SUCCESS; } static inline qmckl_exit_code qmckl_slagel_splitting_19( uint64_t N_updates, const double* __restrict Updates, const uint64_t* __restrict Updates_index, const double breakdown, double* __restrict Slater_inv, double* __restrict later_updates, uint64_t* __restrict later_index, uint64_t* __restrict later, double* __restrict determinant) { double __attribute__((aligned(8))) C[D19_P]; double __attribute__((aligned(8))) D[D19_P]; uint64_t l = 0; // For each update while (l < N_updates) { // C = S^{-1} x U_l for (uint64_t i = 0; i < 19; i++) { C[i] = 0.0f; IVDEP ALIGNED for (uint64_t j = 0; j < D19_P; j++) { C[i] += Slater_inv[i * D19_P + j] * Updates[l * D19_P + j]; } } // Denominator const int cui = Updates_index[l] - 1; double den = 1.0f + C[cui]; if (fabs(den) < breakdown) { // U_l = U_l / 2: split the update in 2 equal halves and save the // second halve in later_updates IVDEP ALIGNED for (uint64_t i = 0; i < D19_P; i++) { later_updates[*later * D19_P + i] = Updates[l * D19_P + i] * 0.5f; C[i] *= 0.5f; } later_index[*later] = Updates_index[l]; (*later)++; den = 1.0f + C[cui]; } // From here onwards we continue with applying the first halve of the // update to Slater_inv double iden = 1.0f / den; if (determinant) *determinant *= den; // D = v^T x S^{-1} : 1 x D19_P IVDEP ALIGNED for (uint64_t j = 0; j < D19_P; j++) { D[j] = Slater_inv[cui * D19_P + j]; } // S^{-1} = S^{-1} - C x D / den for (uint64_t i = 0; i < 19; i++) { IVDEP ALIGNED for (uint64_t j = 0; j < D19_P; j++) { const double update = C[i] * D[j] * iden; Slater_inv[i * D19_P + j] -= update; } } l += 1; } return QMCKL_SUCCESS; } static inline qmckl_exit_code qmckl_slagel_splitting_20( uint64_t N_updates, const double* __restrict Updates, const uint64_t* __restrict Updates_index, const double breakdown, double* __restrict Slater_inv, double* __restrict later_updates, uint64_t* __restrict later_index, uint64_t* __restrict later, double* __restrict determinant) { double __attribute__((aligned(8))) C[D20_P]; double __attribute__((aligned(8))) D[D20_P]; uint64_t l = 0; // For each update while (l < N_updates) { // C = S^{-1} x U_l for (uint64_t i = 0; i < 20; i++) { C[i] = 0.0f; IVDEP ALIGNED for (uint64_t j = 0; j < D20_P; j++) { C[i] += Slater_inv[i * D20_P + j] * Updates[l * D20_P + j]; } } // Denominator const int cui = Updates_index[l] - 1; double den = 1.0f + C[cui]; if (fabs(den) < breakdown) { // U_l = U_l / 2: split the update in 2 equal halves and save the // second halve in later_updates IVDEP ALIGNED for (uint64_t i = 0; i < D20_P; i++) { later_updates[*later * D20_P + i] = Updates[l * D20_P + i] * 0.5f; C[i] *= 0.5f; } later_index[*later] = Updates_index[l]; (*later)++; den = 1.0f + C[cui]; } // From here onwards we continue with applying the first halve of the // update to Slater_inv double iden = 1.0f / den; if (determinant) *determinant *= den; // D = v^T x S^{-1} : 1 x D20_P IVDEP ALIGNED for (uint64_t j = 0; j < D20_P; j++) { D[j] = Slater_inv[cui * D20_P + j]; } // S^{-1} = S^{-1} - C x D / den for (uint64_t i = 0; i < 20; i++) { IVDEP ALIGNED for (uint64_t j = 0; j < D20_P; j++) { const double update = C[i] * D[j] * iden; Slater_inv[i * D20_P + j] -= update; } } l += 1; } return QMCKL_SUCCESS; } static inline qmckl_exit_code qmckl_slagel_splitting_21( uint64_t N_updates, const double* __restrict Updates, const uint64_t* __restrict Updates_index, const double breakdown, double* __restrict Slater_inv, double* __restrict later_updates, uint64_t* __restrict later_index, uint64_t* __restrict later, double* __restrict determinant) { double __attribute__((aligned(8))) C[D21_P]; double __attribute__((aligned(8))) D[D21_P]; uint64_t l = 0; // For each update while (l < N_updates) { // C = S^{-1} x U_l for (uint64_t i = 0; i < 21; i++) { C[i] = 0.0f; IVDEP ALIGNED for (uint64_t j = 0; j < D21_P; j++) { C[i] += Slater_inv[i * D21_P + j] * Updates[l * D21_P + j]; } } // Denominator const int cui = Updates_index[l] - 1; double den = 1.0f + C[cui]; if (fabs(den) < breakdown) { // U_l = U_l / 2: split the update in 2 equal halves and save the // second halve in later_updates IVDEP ALIGNED for (uint64_t i = 0; i < D21_P; i++) { later_updates[*later * D21_P + i] = Updates[l * D21_P + i] * 0.5f; C[i] *= 0.5f; } later_index[*later] = Updates_index[l]; (*later)++; den = 1.0f + C[cui]; } // From here onwards we continue with applying the first halve of the // update to Slater_inv double iden = 1.0f / den; if (determinant) *determinant *= den; // D = v^T x S^{-1} : 1 x D21_P IVDEP ALIGNED for (uint64_t j = 0; j < D21_P; j++) { D[j] = Slater_inv[cui * D21_P + j]; } // S^{-1} = S^{-1} - C x D / den for (uint64_t i = 0; i < 21; i++) { IVDEP ALIGNED for (uint64_t j = 0; j < D21_P; j++) { const double update = C[i] * D[j] * iden; Slater_inv[i * D21_P + j] -= update; } } l += 1; } return QMCKL_SUCCESS; } qmckl_exit_code qmckl_slagel_splitting( const uint64_t LDS, const uint64_t Dim, const uint64_t N_updates, const double* Updates, const uint64_t* Updates_index, const double breakdown, double* Slater_inv, double* later_updates, uint64_t* later_index, uint64_t* later, double* determinant) { if (LDS == (1+(Dim-1)/SIMD_LENGTH)*SIMD_LENGTH) { // Most cases switch (Dim) { case 2: return qmckl_slagel_splitting_2( N_updates, Updates, Updates_index, breakdown, Slater_inv, later_updates, later_index, later, determinant); case 3: return qmckl_slagel_splitting_3( N_updates, Updates, Updates_index, breakdown, Slater_inv, later_updates, later_index, later, determinant); case 4: return qmckl_slagel_splitting_4( N_updates, Updates, Updates_index, breakdown, Slater_inv, later_updates, later_index, later, determinant); case 5: return qmckl_slagel_splitting_5( N_updates, Updates, Updates_index, breakdown, Slater_inv, later_updates, later_index, later, determinant); case 6: return qmckl_slagel_splitting_6( N_updates, Updates, Updates_index, breakdown, Slater_inv, later_updates, later_index, later, determinant); case 7: return qmckl_slagel_splitting_7( N_updates, Updates, Updates_index, breakdown, Slater_inv, later_updates, later_index, later, determinant); case 8: return qmckl_slagel_splitting_8( N_updates, Updates, Updates_index, breakdown, Slater_inv, later_updates, later_index, later, determinant); case 9: return qmckl_slagel_splitting_9( N_updates, Updates, Updates_index, breakdown, Slater_inv, later_updates, later_index, later, determinant); case 10: return qmckl_slagel_splitting_10( N_updates, Updates, Updates_index, breakdown, Slater_inv, later_updates, later_index, later, determinant); case 11: return qmckl_slagel_splitting_11( N_updates, Updates, Updates_index, breakdown, Slater_inv, later_updates, later_index, later, determinant); case 12: return qmckl_slagel_splitting_12( N_updates, Updates, Updates_index, breakdown, Slater_inv, later_updates, later_index, later, determinant); case 13: return qmckl_slagel_splitting_13( N_updates, Updates, Updates_index, breakdown, Slater_inv, later_updates, later_index, later, determinant); case 14: return qmckl_slagel_splitting_14( N_updates, Updates, Updates_index, breakdown, Slater_inv, later_updates, later_index, later, determinant); case 15: return qmckl_slagel_splitting_15( N_updates, Updates, Updates_index, breakdown, Slater_inv, later_updates, later_index, later, determinant); case 16: return qmckl_slagel_splitting_16( N_updates, Updates, Updates_index, breakdown, Slater_inv, later_updates, later_index, later, determinant); case 17: return qmckl_slagel_splitting_17( N_updates, Updates, Updates_index, breakdown, Slater_inv, later_updates, later_index, later, determinant); case 18: return qmckl_slagel_splitting_18( N_updates, Updates, Updates_index, breakdown, Slater_inv, later_updates, later_index, later, determinant); case 19: return qmckl_slagel_splitting_19( N_updates, Updates, Updates_index, breakdown, Slater_inv, later_updates, later_index, later, determinant); case 20: return qmckl_slagel_splitting_20( N_updates, Updates, Updates_index, breakdown, Slater_inv, later_updates, later_index, later, determinant); case 21: return qmckl_slagel_splitting_21( N_updates, Updates, Updates_index, breakdown, Slater_inv, later_updates, later_index, later, determinant); } } else { // When SIMD_LENGTH > 1, called with LDS == Dim AND Dim != (1+(Dim-1)/SIMD_LENGTH)*SIMD_LENGTH) return qmckl_slagel_splitting_hpc( LDS, Dim, N_updates, Updates, Updates_index, breakdown, Slater_inv, later_updates, later_index, later, determinant); } return QMCKL_FAILURE; }
7.1.4 Performance
This function cannot be used by itself and is used in Sherman-Morrison with update splitting and Woodbury 3x3 and 2x2 with Sherman-Morrison and update splitting. Please look at the performance reccomendations for those two kernels.
8 End of files
assert (qmckl_context_destroy(context) == QMCKL_SUCCESS); return 0; }