10
1
mirror of https://gitlab.com/scemama/QCaml.git synced 2025-01-03 10:05:40 +01:00
QCaml/gaussian/lib/primitive_shell_pair.ml

170 lines
3.6 KiB
OCaml
Raw Permalink Normal View History

2024-01-17 10:30:24 +01:00
[@@@landmark "auto-off"]
2020-10-09 09:47:57 +02:00
open Common
2018-03-15 15:25:49 +01:00
open Constants
type t = {
2019-02-26 12:47:23 +01:00
norm_scales : float array lazy_t;
2019-02-20 19:43:16 +01:00
exponent : float; (* {% $\alpha + \beta$ %} *)
exponent_inv : float; (* {% $1/(\alpha + \beta)$ %} *)
a_minus_b_sq : float; (* {% $|A-B|^2$ %} *)
normalization : float; (* [norm_coef_a * norm_coef_b * g], with
{% $g = (\pi/(\alpha+\beta))^(3/2) \exp (-|A-B|^2 \alpha\beta/(\alpha+\beta))$ %} *)
2019-02-26 12:47:23 +01:00
center : Coordinate.t; (* {% $P = (\alpha A + \beta B)/(\alpha+\beta)$ %} *)
center_minus_a : Coordinate.t; (* {% $P - A$ %} *)
a_minus_b : Coordinate.t; (* {% $A - B$ %} *)
2024-01-17 10:30:24 +01:00
ang_mom : Angular_momentum.t;
2020-09-26 12:02:53 +02:00
shell_a : Primitive_shell.t;
shell_b : Primitive_shell.t;
2018-03-15 15:25:49 +01:00
}
2020-09-26 12:02:53 +02:00
module Am = Angular_momentum
2018-03-15 15:25:49 +01:00
module Co = Coordinate
2020-09-26 12:02:53 +02:00
module Ps = Primitive_shell
2018-03-15 15:25:49 +01:00
2024-01-17 10:30:24 +01:00
let ang_mom x = x.ang_mom
let a_minus_b x = x.a_minus_b
let a_minus_b_sq x = x.a_minus_b_sq
let center_minus_a x = x.center_minus_a
let normalization x = x.normalization
let exponent x = x.exponent
let center x = x.center
let shell_a x = x.shell_a
let shell_b x = x.shell_b
2018-03-15 19:11:59 +01:00
2024-01-17 10:30:24 +01:00
let hash a =
2018-03-15 15:25:49 +01:00
Hashtbl.hash a
2018-03-15 19:11:59 +01:00
2018-03-15 15:25:49 +01:00
let equivalent a b =
2018-03-20 15:16:24 +01:00
a.exponent = b.exponent &&
2018-03-21 15:01:39 +01:00
a.ang_mom = b.ang_mom &&
2018-03-20 15:16:24 +01:00
a.normalization = b.normalization &&
2018-03-15 19:11:59 +01:00
a.center = b.center &&
a.center_minus_a = b.center_minus_a &&
a.a_minus_b = b.a_minus_b
2018-03-15 15:25:49 +01:00
let cmp a b =
hash a - hash b
2024-01-17 10:30:24 +01:00
let[@landmark] compute_norm_scales p_a p_b =
2023-06-17 00:32:39 +02:00
Array.map (fun v1 ->
Array.map (fun v2 -> v1 *. v2) (Ps.norm_scales p_b)
) (Ps.norm_scales p_a)
|> Array.to_list
|> Array.concat
2024-01-17 10:30:24 +01:00
let[@landmark] create_make_of p_a p_b =
2018-03-15 15:25:49 +01:00
2024-01-17 10:30:24 +01:00
let a_minus_b =
2018-03-15 15:25:49 +01:00
Co.( Ps.center p_a |- Ps.center p_b )
in
let a_minus_b_sq =
Co.dot a_minus_b a_minus_b
in
2023-06-17 00:32:39 +02:00
let norm_scales =
lazy (compute_norm_scales p_a p_b)
in
2018-03-15 15:25:49 +01:00
2018-03-21 15:01:39 +01:00
let ang_mom =
Am.( Ps.ang_mom p_a + Ps.ang_mom p_b )
2018-03-15 15:25:49 +01:00
in
2024-01-17 10:30:24 +01:00
function p_a ->
2018-03-15 15:25:49 +01:00
2018-03-22 00:29:14 +01:00
let norm_coef_a =
Ps.normalization p_a
in
2018-03-15 15:25:49 +01:00
2024-01-17 10:30:24 +01:00
let alfa_a =
2018-03-22 00:29:14 +01:00
Co.( Ps.exponent p_a |. Ps.center p_a )
in
2018-03-15 15:25:49 +01:00
2024-01-17 10:30:24 +01:00
function p_b ->
2018-03-15 15:25:49 +01:00
2018-03-22 00:29:14 +01:00
let normalization =
norm_coef_a *. Ps.normalization p_b
in
2018-03-15 15:25:49 +01:00
2018-03-22 00:29:14 +01:00
let exponent =
Ps.exponent p_a +. Ps.exponent p_b
in
2018-03-15 15:25:49 +01:00
2018-03-22 00:29:14 +01:00
let exponent_inv = 1. /. exponent in
2018-03-15 15:25:49 +01:00
2024-01-17 10:30:24 +01:00
let normalization =
let argexpo =
2018-03-22 00:29:14 +01:00
Ps.exponent p_a *. Ps.exponent p_b *. a_minus_b_sq *. exponent_inv
in
normalization *. (pi *. exponent_inv)**1.5 *. exp (-. argexpo)
in
2018-03-15 15:25:49 +01:00
2018-03-22 00:29:14 +01:00
function cutoff ->
2018-03-15 15:25:49 +01:00
2018-03-22 00:29:14 +01:00
if abs_float normalization > cutoff then (
2018-03-15 15:25:49 +01:00
2024-01-17 10:30:24 +01:00
let beta_b =
2018-03-22 00:29:14 +01:00
Co.( Ps.exponent p_b |. Ps.center p_b )
in
2018-03-15 15:25:49 +01:00
2018-03-22 00:29:14 +01:00
let center =
2019-02-20 19:43:16 +01:00
Co.(exponent_inv |. (alfa_a |+ beta_b))
2018-03-22 00:29:14 +01:00
in
2018-03-15 15:25:49 +01:00
2024-01-17 10:30:24 +01:00
let center_minus_a =
2018-03-22 00:29:14 +01:00
Co.(center |- Ps.center p_a)
in
2018-03-15 15:25:49 +01:00
2018-03-22 00:29:14 +01:00
Some {
2024-01-17 10:30:24 +01:00
ang_mom ;
exponent ; exponent_inv ; center ; center_minus_a ; a_minus_b ;
2018-03-22 00:29:14 +01:00
a_minus_b_sq ; normalization ; norm_scales ; shell_a = p_a;
shell_b = p_b }
2018-03-15 15:25:49 +01:00
2018-03-22 00:29:14 +01:00
)
else None
2018-03-15 15:25:49 +01:00
2024-01-17 10:30:24 +01:00
let[@landmark] make p_a p_b =
2018-03-15 15:25:49 +01:00
let f =
create_make_of p_a p_b
in
2018-03-15 19:11:59 +01:00
match f p_a p_b 0. with
2018-03-15 15:25:49 +01:00
| Some result -> result
| None -> assert false
2024-01-17 10:30:24 +01:00
let[@landmark] norm_scales x =
2023-06-17 00:32:39 +02:00
try
Lazy.force x.norm_scales
2023-06-17 02:07:35 +02:00
with Lazy.Undefined -> compute_norm_scales x.shell_a x.shell_b
2018-03-15 15:25:49 +01:00
2018-03-20 15:16:24 +01:00
let exponent_inv x = x.exponent_inv
2018-03-15 15:25:49 +01:00
let monocentric x =
Ps.center x.shell_a = Ps.center x.shell_b
2018-03-21 15:01:39 +01:00
2024-01-17 10:30:24 +01:00
let[@landmark] zkey_array x =
Am.zkey_array (Am.Doublet
2018-03-21 15:01:39 +01:00
Ps.(ang_mom x.shell_a, ang_mom x.shell_b)
)