10
1
mirror of https://gitlab.com/scemama/QCaml.git synced 2025-01-05 02:48:37 +01:00
QCaml/SCF/Fock.ml

245 lines
6.6 KiB
OCaml
Raw Normal View History

2018-02-23 18:44:31 +01:00
open Lacaml.D
open Simulation
2018-03-13 18:24:00 +01:00
open Constants
2018-06-27 13:13:59 +02:00
open Util
2018-02-23 18:44:31 +01:00
2018-03-26 16:47:08 +02:00
2018-05-30 18:07:05 +02:00
type t =
{
fock : Mat.t ;
core : Mat.t ;
coulomb : Mat.t ;
exchange : Mat.t ;
}
2018-02-23 18:44:31 +01:00
2019-02-26 11:58:53 +01:00
let fock t = t.fock
let core t = t.core
let coulomb t = t.coulomb
let exchange t = t.exchange
2018-06-13 17:49:58 +02:00
2018-06-29 16:04:40 +02:00
2019-02-26 11:58:53 +01:00
module Ao = AOBasis
2018-06-29 16:04:40 +02:00
2019-02-26 11:58:53 +01:00
let make_rhf ~density ?(threshold=Constants.epsilon) ao_basis =
2018-02-23 18:44:31 +01:00
let m_P = density
2019-02-20 18:24:44 +01:00
and m_T = Ao.kin_ints ao_basis |> KinInt.matrix
and m_V = Ao.eN_ints ao_basis |> NucInt.matrix
and m_G = Ao.ee_ints ao_basis
2018-02-23 18:44:31 +01:00
in
let nBas = Mat.dim1 m_T
in
2018-05-30 18:07:05 +02:00
let m_Hc = Mat.add m_T m_V
2018-05-31 18:50:34 +02:00
and m_J = Array.make_matrix nBas nBas 0.
and m_K = Array.make_matrix nBas nBas 0.
2018-05-30 18:07:05 +02:00
in
2018-07-04 18:42:26 +02:00
for sigma = 1 to nBas do
let m_Ksigma = m_K.(sigma-1) in
for nu = 1 to nBas do
let m_Jnu = m_J.(nu-1) in
for lambda = 1 to nBas do
let pJ = m_P.{lambda,sigma}
and pK = 0.5 *. m_P.{lambda,nu}
in
match (abs_float pJ > threshold , abs_float pK > threshold, nu < sigma) with
2018-07-04 18:42:26 +02:00
| (false, false, _) -> ()
| (true , true , true) ->
2019-02-26 11:58:53 +01:00
begin
for mu = 1 to nu do
let integral =
ERI.get_phys m_G mu lambda nu sigma
in
if (integral <> 0.) then begin
m_Jnu.(mu-1) <- m_Jnu.(mu-1) +. pJ *. integral;
2020-04-20 02:06:49 +02:00
m_Ksigma.(mu-1) <- m_Ksigma.(mu-1) +. pK *. integral
2019-02-26 11:58:53 +01:00
end
done;
for mu = nu+1 to sigma do
2020-04-20 02:06:49 +02:00
m_Ksigma.(mu-1) <- m_Ksigma.(mu-1) +. pK *.
2019-02-26 11:58:53 +01:00
ERI.get_phys m_G mu lambda nu sigma
done
end
2018-07-04 18:42:26 +02:00
| (true , true , false) ->
2019-02-26 11:58:53 +01:00
begin
2018-07-04 18:42:26 +02:00
for mu = 1 to sigma do
2019-02-26 11:58:53 +01:00
let integral =
ERI.get_phys m_G mu lambda nu sigma
in
if (integral <> 0.) then begin
m_Jnu.(mu-1) <- m_Jnu.(mu-1) +. pJ *. integral;
2020-04-20 02:06:49 +02:00
m_Ksigma.(mu-1) <- m_Ksigma.(mu-1) +. pK *. integral
2019-02-26 11:58:53 +01:00
end
done;
for mu = sigma+1 to nu do
m_Jnu.(mu-1) <-
m_Jnu.(mu-1) +. pJ *. ERI.get_phys m_G mu lambda nu sigma
2018-07-04 18:42:26 +02:00
done
2019-02-26 11:58:53 +01:00
end
| (false, true , _) ->
for mu = 1 to sigma do
m_Ksigma.(mu-1) <-
2020-04-20 02:06:49 +02:00
m_Ksigma.(mu-1) +. pK *. ERI.get_phys m_G mu lambda nu sigma
2019-02-26 11:58:53 +01:00
done
2018-07-04 18:42:26 +02:00
| (true , false, _) ->
2019-02-26 11:58:53 +01:00
for mu = 1 to nu do
m_Jnu.(mu-1) <-
m_Jnu.(mu-1) +. pJ *. ERI.get_phys m_G mu lambda nu sigma
done
done
done;
for mu = 1 to sigma-1 do
m_K.(mu-1).(sigma-1) <- m_Ksigma.(mu-1);
done
done;
for nu = 1 to nBas do
let m_Jnu = m_J.(nu-1) in
for mu = 1 to nu-1 do
m_J.(mu-1).(nu-1) <- m_Jnu.(mu-1)
done
done;
let m_J = Mat.of_array m_J
and m_K = Mat.of_array m_K
in
2020-04-20 02:06:49 +02:00
{ fock = Mat.add m_Hc (Mat.sub m_J m_K) ;
2019-02-26 11:58:53 +01:00
core = m_Hc ; coulomb = m_J ; exchange = m_K }
let make_uhf ~density_same ~density_other ?(threshold=Constants.epsilon) ao_basis =
let m_P_a = density_same
and m_P_b = density_other
and m_T = Ao.kin_ints ao_basis |> KinInt.matrix
and m_V = Ao.eN_ints ao_basis |> NucInt.matrix
and m_G = Ao.ee_ints ao_basis
in
let nBas = Mat.dim1 m_T
in
let m_Hc = Mat.add m_T m_V
and m_J = Array.make_matrix nBas nBas 0.
and m_K = Array.make_matrix nBas nBas 0.
in
for sigma = 1 to nBas do
let m_Ksigma = m_K.(sigma-1) in
for nu = 1 to nBas do
let m_Jnu = m_J.(nu-1) in
for lambda = 1 to nBas do
let pJ = m_P_a.{lambda,sigma} +. m_P_b.{lambda,sigma}
and pK = m_P_a.{lambda,nu}
in
match (abs_float pJ > threshold , abs_float pK > threshold, nu < sigma) with
| (false, false, _) -> ()
| (true , true , true) ->
begin
2018-07-04 18:42:26 +02:00
for mu = 1 to nu do
2019-02-26 11:58:53 +01:00
let integral =
ERI.get_phys m_G mu lambda nu sigma
in
if (integral <> 0.) then begin
m_Jnu.(mu-1) <- m_Jnu.(mu-1) +. pJ *. integral;
2020-04-20 02:06:49 +02:00
m_Ksigma.(mu-1) <- m_Ksigma.(mu-1) +. pK *. integral
2019-02-26 11:58:53 +01:00
end
done;
for mu = nu+1 to sigma do
2020-04-20 02:06:49 +02:00
m_Ksigma.(mu-1) <- m_Ksigma.(mu-1) +. pK *.
2019-02-26 11:58:53 +01:00
ERI.get_phys m_G mu lambda nu sigma
done
end
| (true , true , false) ->
begin
for mu = 1 to sigma do
let integral =
ERI.get_phys m_G mu lambda nu sigma
in
if (integral <> 0.) then begin
m_Jnu.(mu-1) <- m_Jnu.(mu-1) +. pJ *. integral;
2020-04-20 02:06:49 +02:00
m_Ksigma.(mu-1) <- m_Ksigma.(mu-1) +. pK *. integral
2019-02-26 11:58:53 +01:00
end
done;
for mu = sigma+1 to nu do
m_Jnu.(mu-1) <-
m_Jnu.(mu-1) +. pJ *. ERI.get_phys m_G mu lambda nu sigma
2018-07-04 18:42:26 +02:00
done
2019-02-26 11:58:53 +01:00
end
| (false, true , _) ->
for mu = 1 to sigma do
m_Ksigma.(mu-1) <-
2020-04-20 02:06:49 +02:00
m_Ksigma.(mu-1) +. pK *. ERI.get_phys m_G mu lambda nu sigma
2019-02-26 11:58:53 +01:00
done
| (true , false, _) ->
for mu = 1 to nu do
m_Jnu.(mu-1) <-
m_Jnu.(mu-1) +. pJ *. ERI.get_phys m_G mu lambda nu sigma
done
2018-07-04 18:42:26 +02:00
done
done;
for mu = 1 to sigma-1 do
m_K.(mu-1).(sigma-1) <- m_Ksigma.(mu-1);
done
done;
for nu = 1 to nBas do
let m_Jnu = m_J.(nu-1) in
for mu = 1 to nu-1 do
m_J.(mu-1).(nu-1) <- m_Jnu.(mu-1)
done
done;
2018-05-31 18:50:34 +02:00
let m_J = Mat.of_array m_J
and m_K = Mat.of_array m_K
in
2020-04-20 02:06:49 +02:00
{ fock = Mat.add m_Hc (Mat.sub m_J m_K) ;
2018-05-30 18:07:05 +02:00
core = m_Hc ; coulomb = m_J ; exchange = m_K }
2018-02-23 18:44:31 +01:00
2018-06-27 13:13:59 +02:00
2019-02-26 11:58:53 +01:00
let op ~f f1 f2 =
2019-02-26 11:58:53 +01:00
assert (f1.core = f2.core);
let m_Hc = f1.core
and m_J = f f1.coulomb f2.coulomb
and m_K = f f1.exchange f2.exchange
in
{
2020-04-20 02:06:49 +02:00
fock = Mat.add m_Hc (Mat.sub m_J m_K);
core = m_Hc;
coulomb = m_J;
exchange = m_K;
}
let add = op ~f:(fun a b -> Mat.add a b)
2019-02-26 11:58:53 +01:00
let sub = op ~f:(fun a b -> Mat.sub a b)
2019-02-26 11:58:53 +01:00
let scale alpha f1 =
let m_Hc = f1.core
and m_J = lacpy f1.coulomb
and m_K = lacpy f1.exchange
in
Mat.scal alpha m_J;
Mat.scal alpha m_K;
{
2020-04-20 02:06:49 +02:00
fock = Mat.add m_Hc (Mat.sub m_J m_K);
2019-02-26 11:58:53 +01:00
core = m_Hc;
coulomb = m_J;
exchange = m_K;
}
2019-12-02 14:58:48 +01:00
let pp ppf a =
2018-06-27 13:13:59 +02:00
Format.fprintf ppf "@[<2>";
Format.fprintf ppf "@[ Fock matrix:@[<2>@[%a@]@.]@]" pp_matrix a.fock;
Format.fprintf ppf "@[ Core Hamiltonian:@[<2>@[%a@]@.]@]" pp_matrix a.core;
Format.fprintf ppf "@[ Coulomb matrix:@[<2>@[%a@]@.]@]" pp_matrix a.coulomb;
Format.fprintf ppf "@[ Exchange matrix:@[<2>@[%a@]@.]@]" pp_matrix a.exchange;
Format.fprintf ppf "@]"