mirror of
https://gitlab.com/scemama/QCaml.git
synced 2025-01-03 10:05:40 +01:00
198 lines
5.4 KiB
OCaml
198 lines
5.4 KiB
OCaml
type t =
|
|
{
|
|
size : int;
|
|
contracted_shells : Contracted_shell.t array ;
|
|
atomic_shells : Atomic_shell.t array lazy_t;
|
|
general_basis : General_basis.t ;
|
|
}
|
|
|
|
module As = Atomic_shell
|
|
module Cs = Contracted_shell
|
|
module Gb = General_basis
|
|
module Ps = Primitive_shell
|
|
|
|
let general_basis t = t.general_basis
|
|
|
|
(** Returns an array of the basis set per atom *)
|
|
let of_nuclei_and_general_basis nucl bas =
|
|
let index_ = ref 0 in
|
|
let contracted_shells =
|
|
Array.map (fun (e, center) ->
|
|
List.assoc e bas
|
|
|> Array.map (fun (ang_mom, shell) ->
|
|
let lc =
|
|
Array.map (fun Gb.{exponent ; coefficient} ->
|
|
coefficient, Ps.make ang_mom center exponent) shell
|
|
in
|
|
let result = Cs.make ~index:!index_ lc in
|
|
index_ := !index_ + Cs.size_of_shell result;
|
|
result
|
|
)
|
|
) nucl
|
|
|> Array.to_list
|
|
|> Array.concat
|
|
in
|
|
let atomic_shells = lazy(
|
|
let uniq_center_angmom =
|
|
Array.map (fun x -> Cs.center x, Cs.ang_mom x) contracted_shells
|
|
|> Array.to_list
|
|
|> List.sort_uniq compare
|
|
in
|
|
let csl =
|
|
Array.to_list contracted_shells
|
|
in
|
|
List.map (fun (center, ang_mom) ->
|
|
let a =
|
|
List.filter (fun x -> Cs.center x = center && Cs.ang_mom x = ang_mom) csl
|
|
|> Array.of_list
|
|
in
|
|
As.make ~index:(Cs.index a.(0)) a
|
|
) uniq_center_angmom
|
|
|> List.sort (fun x y -> compare (As.index x) (As.index y))
|
|
|> Array.of_list
|
|
) in
|
|
{ contracted_shells ; atomic_shells ; size = !index_;
|
|
general_basis = bas }
|
|
|
|
|
|
let size t = t.size
|
|
|
|
let atomic_shells t = Lazy.force t.atomic_shells
|
|
|
|
let contracted_shells t = t.contracted_shells
|
|
|
|
let to_string b =
|
|
let b = atomic_shells b in
|
|
let line ="
|
|
-----------------------------------------------------------------------
|
|
" in
|
|
"
|
|
Atomic Basis set
|
|
----------------
|
|
|
|
-----------------------------------------------------------------------
|
|
# Angular Coordinates (Bohr) Exponents Coefficients
|
|
Momentum X Y Z
|
|
-----------------------------------------------------------------------
|
|
"
|
|
^
|
|
( Array.map (fun p -> Format.(fprintf str_formatter "%a" As.pp p;
|
|
flush_str_formatter ())) b
|
|
|> Array.to_list
|
|
|> String.concat line
|
|
)
|
|
^ line
|
|
|
|
|
|
|
|
let of_nuclei_and_basis_filename ~nuclei filename =
|
|
let general_basis =
|
|
General_basis.read filename
|
|
in
|
|
of_nuclei_and_general_basis nuclei general_basis
|
|
|
|
let of_nuclei_and_basis_string ~nuclei str =
|
|
let general_basis =
|
|
General_basis.of_string str
|
|
in
|
|
of_nuclei_and_general_basis nuclei general_basis
|
|
|
|
|
|
let of_nuclei_and_basis_filenames ~nuclei filenames =
|
|
let general_basis =
|
|
General_basis.read_many filenames
|
|
in
|
|
of_nuclei_and_general_basis nuclei general_basis
|
|
|
|
|
|
let of_trexio f =
|
|
let nuclei = Particles.Nuclei.of_trexio f in
|
|
let shell_ang_mom = Trexio.read_basis_shell_ang_mom f
|
|
|> Array.map Common.Angular_momentum.of_int
|
|
in
|
|
let shell_index = Trexio.read_basis_shell_index f
|
|
|> Array.to_list
|
|
in
|
|
let nucleus_index = Trexio.read_basis_nucleus_index f
|
|
|> Array.to_list
|
|
in
|
|
let exponent = Trexio.read_basis_exponent f in
|
|
let coefficient = Trexio.read_basis_coefficient f in
|
|
|
|
let filter_indices ref_index full_array =
|
|
List.mapi (fun j idx ->
|
|
if idx = ref_index then Some j
|
|
else None) full_array
|
|
|> List.filter (function
|
|
| None -> false
|
|
| _ -> true)
|
|
|> List.map (function
|
|
| Some x -> x
|
|
| None -> assert false)
|
|
in
|
|
|
|
let index_ = ref 0 in
|
|
let contracted_shells =
|
|
Array.mapi (fun i (_,center) ->
|
|
let shell_indices = filter_indices i nucleus_index in
|
|
List.map (fun ishell ->
|
|
let prim_indices = filter_indices ishell shell_index in
|
|
let lc = List.map (fun k ->
|
|
coefficient.(k),
|
|
Ps.make shell_ang_mom.(ishell) center exponent.(k)) prim_indices
|
|
|> Array.of_list
|
|
in
|
|
let result = Cs.make ~index:!index_ lc in
|
|
index_ := !index_ + Cs.size_of_shell result;
|
|
result
|
|
) shell_indices
|
|
|> Array.of_list
|
|
) nuclei
|
|
|> Array.to_list
|
|
|> Array.concat
|
|
in
|
|
let atomic_shells = lazy(
|
|
let uniq_center_angmom =
|
|
Array.map (fun x -> Cs.center x, Cs.ang_mom x) contracted_shells
|
|
|> Array.to_list
|
|
|> List.sort_uniq compare
|
|
in
|
|
let csl =
|
|
Array.to_list contracted_shells
|
|
in
|
|
List.map (fun (center, ang_mom) ->
|
|
let a =
|
|
List.filter (fun x -> Cs.center x = center && Cs.ang_mom x = ang_mom) csl
|
|
|> Array.of_list
|
|
in
|
|
As.make ~index:(Cs.index a.(0)) a
|
|
) uniq_center_angmom
|
|
|> List.sort (fun x y -> compare (As.index x) (As.index y))
|
|
|> Array.of_list
|
|
) in
|
|
{ contracted_shells ; atomic_shells ; size = !index_;
|
|
general_basis = [];
|
|
}
|
|
|
|
let to_trexio _f _t =
|
|
failwith "Not implemented"
|
|
(*
|
|
Trexio.write_basis_type f "Gaussian" ;
|
|
Trexio.write_basis_shell_num f t.size;
|
|
let prim_num =
|
|
Array.fold_left (fun accu i -> accu + Array.length i) 0 t.contracted_shells
|
|
in
|
|
Trexio.read_basis_prim_num f prim_num;
|
|
let nucleus_index =
|
|
Array.map
|
|
in
|
|
let shell_factor = Trexio.read_basis_shell_factor f in
|
|
let prim_factor =
|
|
Trexio.read_basis_prim_factor f in
|
|
()
|
|
*)
|
|
|
|
let pp ppf t =
|
|
Format.fprintf ppf "@[%s@]" (to_string t)
|
|
|