mirror of
https://github.com/QuantumPackage/qp2.git
synced 2025-01-10 20:18:12 +01:00
2cffbdcc9d
instead of real/imag parts read separately, use ezfio to read/write complex arrays with extra dimension of size 2 converter needs to be tested (might need to transpose some axes in arrays) converter has extra garbage that needs to be removed after testing
325 lines
9.4 KiB
OCaml
325 lines
9.4 KiB
OCaml
open Qptypes
|
|
open Qputils
|
|
open Sexplib.Std
|
|
|
|
module Mo_basis : sig
|
|
type t =
|
|
{ mo_num : MO_number.t ;
|
|
mo_label : MO_label.t;
|
|
mo_class : MO_class.t array;
|
|
mo_occ : MO_occ.t array;
|
|
mo_coef : (MO_coef.t array) array;
|
|
ao_md5 : MD5.t;
|
|
} [@@deriving sexp]
|
|
val read : unit -> t option
|
|
val write : t -> unit
|
|
val reorder : t -> int array -> t
|
|
val to_string : t -> string
|
|
val to_rst : t -> Rst_string.t
|
|
end = struct
|
|
type t =
|
|
{ mo_num : MO_number.t ;
|
|
mo_label : MO_label.t;
|
|
mo_class : MO_class.t array;
|
|
mo_occ : MO_occ.t array;
|
|
mo_coef : (MO_coef.t array) array;
|
|
ao_md5 : MD5.t;
|
|
} [@@deriving sexp]
|
|
|
|
let get_default = Qpackage.get_ezfio_default "mo_basis"
|
|
|
|
let is_complex = lazy (Ezfio.get_nuclei_is_complex () )
|
|
|
|
let read_mo_label () =
|
|
if not (Ezfio.has_mo_basis_mo_label ()) then
|
|
Ezfio.set_mo_basis_mo_label "None"
|
|
;
|
|
Ezfio.get_mo_basis_mo_label ()
|
|
|> MO_label.of_string
|
|
|
|
|
|
let reorder b ordering =
|
|
{ b with
|
|
mo_coef = Array.map (fun mo ->
|
|
Array.init (Array.length mo)
|
|
(fun i -> mo.(ordering.(i)))
|
|
) b.mo_coef
|
|
}
|
|
|
|
let read_ao_md5 () =
|
|
let ao_md5 =
|
|
match (Input_ao_basis.Ao_basis.read ()) with
|
|
| None -> failwith "Unable to read AO basis"
|
|
| Some result -> Input_ao_basis.Ao_basis.to_md5 result
|
|
in
|
|
let result =
|
|
if not (Ezfio.has_mo_basis_ao_md5 ()) then
|
|
begin
|
|
MD5.to_string ao_md5
|
|
|> Ezfio.set_mo_basis_ao_md5
|
|
end;
|
|
Ezfio.get_mo_basis_ao_md5 ()
|
|
|> MD5.of_string
|
|
in
|
|
if (ao_md5 <> result) then
|
|
begin
|
|
Printf.eprintf ":%s:\n:%s:\n%!" (MD5.to_string ao_md5) (MD5.to_string result);
|
|
failwith "The current MOs don't correspond to the current AOs."
|
|
end;
|
|
result
|
|
|
|
|
|
let read_mo_num () =
|
|
let elec_alpha_num =
|
|
Ezfio.get_electrons_elec_alpha_num ()
|
|
in
|
|
let result =
|
|
Ezfio.get_mo_basis_mo_num ()
|
|
in
|
|
if result < elec_alpha_num then
|
|
failwith "More alpha electrons than MOs";
|
|
MO_number.of_int result
|
|
|
|
|
|
let read_mo_class () =
|
|
if not (Ezfio.has_mo_basis_mo_class ()) then
|
|
begin
|
|
let mo_num = MO_number.to_int (read_mo_num ()) in
|
|
let data =
|
|
Array.init mo_num (fun _ -> MO_class.(to_string (Active [])))
|
|
|> Array.to_list
|
|
in
|
|
Ezfio.ezfio_array_of_list ~rank:1
|
|
~dim:[| mo_num |] ~data:data
|
|
|> Ezfio.set_mo_basis_mo_class
|
|
end;
|
|
Ezfio.flattened_ezfio (Ezfio.get_mo_basis_mo_class () )
|
|
|> Array.map MO_class.of_string
|
|
|
|
|
|
let read_mo_occ () =
|
|
if not (Ezfio.has_mo_basis_mo_label ()) then
|
|
begin
|
|
let elec_alpha_num = Ezfio.get_electrons_elec_alpha_num ()
|
|
and elec_beta_num = Ezfio.get_electrons_elec_beta_num ()
|
|
and mo_num = MO_number.to_int (read_mo_num ()) in
|
|
let data = Array.init mo_num (fun i ->
|
|
if (i<elec_beta_num) then 2.
|
|
else if (i < elec_alpha_num) then 1.
|
|
else 0.) |> Array.to_list in
|
|
Ezfio.ezfio_array_of_list ~rank:1
|
|
~dim:[| mo_num |] ~data:data
|
|
|> Ezfio.set_mo_basis_mo_occ
|
|
end;
|
|
Ezfio.flattened_ezfio (Ezfio.get_mo_basis_mo_occ () )
|
|
|> Array.map MO_occ.of_float
|
|
|
|
|
|
let read_mo_coef () =
|
|
let a =
|
|
(
|
|
if Lazy.force is_complex then
|
|
Ezfio.get_mo_basis_mo_coef_complex ()
|
|
else
|
|
Ezfio.get_mo_basis_mo_coef ()
|
|
)
|
|
|> Ezfio.flattened_ezfio
|
|
|> Array.map MO_coef.of_float
|
|
in
|
|
let mo_num = read_mo_num () |> MO_number.to_int in
|
|
let ao_num = (Array.length a)/mo_num in
|
|
Array.init mo_num (fun j ->
|
|
Array.sub a (j*ao_num) (ao_num)
|
|
)
|
|
|
|
|
|
let read () =
|
|
if (Ezfio.has_mo_basis_mo_num ()) then
|
|
Some
|
|
{ mo_num = read_mo_num ();
|
|
mo_label = read_mo_label () ;
|
|
mo_class = read_mo_class ();
|
|
mo_occ = read_mo_occ ();
|
|
mo_coef = read_mo_coef ();
|
|
ao_md5 = read_ao_md5 ();
|
|
}
|
|
else
|
|
None
|
|
|
|
|
|
let mo_coef_to_string mo_coef =
|
|
let ao_num = Array.length mo_coef.(0)
|
|
and mo_num = Array.length mo_coef in
|
|
let rec print_five imin imax =
|
|
match (imax-imin+1) with
|
|
| 1 ->
|
|
let header = [ Printf.sprintf " #%15d" (imin+1) ; ] in
|
|
let new_lines =
|
|
List.init ao_num (fun i ->
|
|
Printf.sprintf " %3d %15.10f " (i+1)
|
|
(MO_coef.to_float mo_coef.(imin ).(i)) )
|
|
in header @ new_lines
|
|
| 2 ->
|
|
let header = [ Printf.sprintf " #%15d %15d" (imin+1) (imin+2) ; ] in
|
|
let new_lines =
|
|
List.init ao_num (fun i ->
|
|
Printf.sprintf " %3d %15.10f %15.10f" (i+1)
|
|
(MO_coef.to_float mo_coef.(imin ).(i))
|
|
(MO_coef.to_float mo_coef.(imin+1).(i)) )
|
|
in header @ new_lines
|
|
| 3 ->
|
|
let header = [ Printf.sprintf " #%15d %15d %15d"
|
|
(imin+1) (imin+2) (imin+3); ] in
|
|
let new_lines =
|
|
List.init ao_num (fun i ->
|
|
Printf.sprintf " %3d %15.10f %15.10f %15.10f" (i+1)
|
|
(MO_coef.to_float mo_coef.(imin ).(i))
|
|
(MO_coef.to_float mo_coef.(imin+1).(i))
|
|
(MO_coef.to_float mo_coef.(imin+2).(i)) )
|
|
in header @ new_lines
|
|
| 4 ->
|
|
let header = [ Printf.sprintf " #%15d %15d %15d %15d"
|
|
(imin+1) (imin+2) (imin+3) (imin+4) ; ] in
|
|
let new_lines =
|
|
List.init ao_num (fun i ->
|
|
Printf.sprintf " %3d %15.10f %15.10f %15.10f %15.10f" (i+1)
|
|
(MO_coef.to_float mo_coef.(imin ).(i))
|
|
(MO_coef.to_float mo_coef.(imin+1).(i))
|
|
(MO_coef.to_float mo_coef.(imin+2).(i))
|
|
(MO_coef.to_float mo_coef.(imin+3).(i)) )
|
|
in header @ new_lines
|
|
| 5 ->
|
|
let header = [ Printf.sprintf " #%15d %15d %15d %15d %15d"
|
|
(imin+1) (imin+2) (imin+3) (imin+4) (imin+5) ; ] in
|
|
let new_lines =
|
|
List.init ao_num (fun i ->
|
|
Printf.sprintf " %3d %15.10f %15.10f %15.10f %15.10f %15.10f" (i+1)
|
|
(MO_coef.to_float mo_coef.(imin ).(i))
|
|
(MO_coef.to_float mo_coef.(imin+1).(i))
|
|
(MO_coef.to_float mo_coef.(imin+2).(i))
|
|
(MO_coef.to_float mo_coef.(imin+3).(i))
|
|
(MO_coef.to_float mo_coef.(imin+4).(i)) )
|
|
in header @ new_lines
|
|
| _ -> assert false
|
|
in
|
|
let rec create_list accu i =
|
|
if (i+4 < mo_num) then
|
|
create_list ( (print_five i (i+3) |> String.concat "\n")::accu ) (i+4)
|
|
else
|
|
(print_five i (mo_num-1) |> String.concat "\n")::accu |> List.rev
|
|
in
|
|
create_list [] 0 |> String.concat "\n\n"
|
|
|
|
|
|
let to_rst b =
|
|
Printf.sprintf "
|
|
Label of the molecular orbitals ::
|
|
|
|
mo_label = %s
|
|
|
|
Total number of MOs ::
|
|
|
|
mo_num = %s
|
|
|
|
MO coefficients ::
|
|
|
|
%s
|
|
"
|
|
(MO_label.to_string b.mo_label)
|
|
(MO_number.to_string b.mo_num)
|
|
(mo_coef_to_string b.mo_coef)
|
|
|> Rst_string.of_string
|
|
|
|
|
|
|
|
let to_string b =
|
|
Printf.sprintf "
|
|
mo_label = \"%s\"
|
|
mo_num = %s
|
|
mo_clas = %s
|
|
mo_occ = %s
|
|
mo_coef = %s
|
|
"
|
|
(MO_label.to_string b.mo_label)
|
|
(MO_number.to_string b.mo_num)
|
|
(b.mo_class |> Array.to_list |> List.map
|
|
(MO_class.to_string) |> String.concat ", " )
|
|
(b.mo_occ |> Array.to_list |> List.map
|
|
(MO_occ.to_string) |> String.concat ", " )
|
|
(b.mo_coef |> Array.map
|
|
(fun x-> Array.map MO_coef.to_string x |>
|
|
Array.to_list |> String.concat "," ) |>
|
|
Array.to_list |> String.concat "\n" )
|
|
|
|
|
|
let write_mo_num n =
|
|
MO_number.to_int n
|
|
|> Ezfio.set_mo_basis_mo_num
|
|
|
|
|
|
let write_mo_label a =
|
|
MO_label.to_string a
|
|
|> Ezfio.set_mo_basis_mo_label
|
|
|
|
|
|
let write_mo_class a =
|
|
let mo_num = Array.length a in
|
|
let data = Array.map MO_class.to_string a
|
|
|> Array.to_list
|
|
in Ezfio.ezfio_array_of_list ~rank:1 ~dim:[| mo_num |] ~data
|
|
|> Ezfio.set_mo_basis_mo_class
|
|
|
|
|
|
let write_mo_occ a =
|
|
let mo_num = Array.length a in
|
|
let data = Array.map MO_occ.to_float a
|
|
|> Array.to_list
|
|
in Ezfio.ezfio_array_of_list ~rank:1 ~dim:[| mo_num |] ~data
|
|
|> Ezfio.set_mo_basis_mo_occ
|
|
|
|
|
|
let write_md5 a =
|
|
MD5.to_string a
|
|
|> Ezfio.set_mo_basis_ao_md5
|
|
|
|
|
|
let write_mo_coef a =
|
|
let mo_num = Array.length a in
|
|
let ao_num =
|
|
let x = Array.length a.(0) in
|
|
if Lazy.force is_complex then x/2 else x
|
|
in
|
|
let data =
|
|
Array.map (fun mo -> Array.map MO_coef.to_float mo
|
|
|> Array.to_list) a
|
|
|> Array.to_list
|
|
|> List.concat
|
|
in
|
|
if Lazy.force is_complex then
|
|
(Ezfio.ezfio_array_of_list ~rank:3 ~dim:[| 2 ; ao_num ; mo_num |] ~data
|
|
|> Ezfio.set_mo_basis_mo_coef_complex )
|
|
else
|
|
(Ezfio.ezfio_array_of_list ~rank:2 ~dim:[| ao_num ; mo_num |] ~data
|
|
|> Ezfio.set_mo_basis_mo_coef )
|
|
|
|
|
|
let write
|
|
{ mo_num : MO_number.t ;
|
|
mo_label : MO_label.t;
|
|
mo_class : MO_class.t array;
|
|
mo_occ : MO_occ.t array;
|
|
mo_coef : (MO_coef.t array) array;
|
|
ao_md5 : MD5.t;
|
|
} =
|
|
write_mo_num mo_num;
|
|
write_mo_label mo_label;
|
|
write_mo_class mo_class;
|
|
write_mo_occ mo_occ;
|
|
write_mo_coef mo_coef;
|
|
write_md5 ao_md5
|
|
|
|
|
|
end
|
|
|
|
|