10
1
mirror of https://gitlab.com/scemama/QCaml.git synced 2025-01-03 01:55:40 +01:00

Reading both XYZ and Zmatrix

This commit is contained in:
Anthony Scemama 2018-01-17 15:56:57 +01:00
parent e909bb9abb
commit dad44067f7
16 changed files with 222 additions and 89 deletions

View File

@ -1,11 +1,15 @@
(** General basis set read from a file *)
type primitive = { type primitive = {
exponent: float ; exponent: float ;
coefficient: float coefficient: float
} }
type contracted_shell = Angular_momentum.t * (primitive array) type element_name = string
type gaussian_basis_set = string * (contracted_shell array) type general_contracted_shell = Angular_momentum.t * (primitive array)
type basis_set = element_name * (general_contracted_shell array)
let string_of_primitive ?id prim = let string_of_primitive ?id prim =
@ -30,6 +34,9 @@ let string_of_contracted_shell_array a =
|> Array.to_list |> Array.to_list
|> String.concat "\n" |> String.concat "\n"
let to_string (name, contracted_shell_array) = let to_string (name, contracted_shell_array) =
Printf.sprintf "%s\n%s" name (string_of_contracted_shell_array contracted_shell_array) Printf.sprintf "%s\n%s" name (string_of_contracted_shell_array contracted_shell_array)

View File

@ -1,7 +1,7 @@
{ {
exception SyntaxError of string exception SyntaxError of string
open Basis_parser open Gamess_parser
} }

View File

@ -1,4 +1,4 @@
/* Parses basis sets in GAMESS format */ /* Parses basis sets GAMESS format */
%{ %{
@ -13,7 +13,7 @@
%token EOF %token EOF
%start input %start input
%type <Basis.gaussian_basis_set> input %type <Basis.basis_set> input
%% /* Grammar rules and actions follow */ %% /* Grammar rules and actions follow */
@ -46,3 +46,4 @@ primitive:
| INTEGER FLOAT FLOAT EOL { Basis.{exponent=$2 ; coefficient=$3 } } | INTEGER FLOAT FLOAT EOL { Basis.{exponent=$2 ; coefficient=$3 } }

21
Basis/Gamess_reader.ml Normal file
View File

@ -0,0 +1,21 @@
(** Read a basis set file in GAMESS format and return an association list where the key is an
Element.t and the value is the parsed basis set. *)
let read ~filename =
let lexbuf =
let ic = open_in filename in
Lexing.from_channel ic
in
let rec aux accu =
try
let element, basis =
Gamess_parser.input Basis_lexer.read_all lexbuf
in
let key =
Element.of_string element
in
aux ((key, basis)::accu)
with
| Parsing.Parse_error -> List.rev accu
in
aux []

View File

@ -13,7 +13,9 @@ ALL_EXE=$(patsubst %.ml,%.native,$(wildcard qp_*.ml))
.PHONY: default .PHONY: default
default: $(ALL_EXE) #default: $(ALL_EXE)
default: test.byte
tests: $(ALL_TESTS) tests: $(ALL_TESTS)
qpackage.odocl: $(MLIFILES) qpackage.odocl: $(MLIFILES)

View File

@ -79,9 +79,8 @@ let to_long_string = function
| Sb -> "Antimony" | Te -> "Tellurium" | I -> "Iodine" | Sb -> "Antimony" | Te -> "Tellurium" | I -> "Iodine"
| Xe -> "Xenon" | Pt -> "Platinum" | Xe -> "Xenon" | Pt -> "Platinum"
let to_charge c =
begin let to_int = function
match c with
| X -> 0 | H -> 1 | He -> 2 | Li -> 3 | X -> 0 | H -> 1 | He -> 2 | Li -> 3
| Be -> 4 | B -> 5 | C -> 6 | N -> 7 | Be -> 4 | B -> 5 | C -> 6 | N -> 7
| O -> 8 | F -> 9 | Ne -> 10 | Na -> 11 | O -> 8 | F -> 9 | Ne -> 10 | Na -> 11
@ -96,12 +95,13 @@ let to_charge c =
| Ru -> 44 | Rh -> 45 | Pd -> 46 | Ag -> 47 | Ru -> 44 | Rh -> 45 | Pd -> 46 | Ag -> 47
| Cd -> 48 | In -> 49 | Sn -> 50 | Sb -> 51 | Cd -> 48 | In -> 49 | Sn -> 50 | Sb -> 51
| Te -> 52 | I -> 53 | Xe -> 54 | Pt -> 78 | Te -> 52 | I -> 53 | Xe -> 54 | Pt -> 78
end
|> Charge.of_int
let of_charge c = let to_charge c =
match Charge.to_int c with to_int c |> Charge.of_int
let of_int = function
| 0 -> X | 1 -> H | 2 -> He | 3 -> Li | 0 -> X | 1 -> H | 2 -> He | 3 -> Li
| 4 -> Be | 5 -> B | 6 -> C | 7 -> N | 4 -> Be | 5 -> B | 6 -> C | 7 -> N
| 8 -> O | 9 -> F | 10 -> Ne | 11 -> Na | 8 -> O | 9 -> F | 10 -> Ne | 11 -> Na
@ -119,6 +119,10 @@ let of_charge c =
| x -> raise (ElementError ("Element of charge "^(string_of_int x)^" unknown")) | x -> raise (ElementError ("Element of charge "^(string_of_int x)^" unknown"))
let of_charge c =
Charge.to_int c |> of_int
let covalent_radius x = let covalent_radius x =
let result = function let result = function
| X -> 0. | H -> 0.37 | He -> 0.70 | Li -> 1.23 | X -> 0. | H -> 0.37 | He -> 0.70 | Li -> 1.23

View File

@ -1,6 +1,14 @@
exception ElementError of string exception ElementError of string
type t type t =
|X
|H |He
|Li|Be |B |C |N |O |F |Ne
|Na|Mg |Al|Si|P |S |Cl|Ar
|K |Ca|Sc|Ti|V |Cr|Mn|Fe|Co|Ni|Cu|Zn|Ga|Ge|As|Se|Br|Kr
|Rb|Sr|Y |Zr|Nb|Mo|Tc|Ru|Rh|Pd|Ag|Cd|In|Sn|Sb|Te|I |Xe
|Pt
(** String conversion functions *) (** String conversion functions *)
val of_string : string -> t val of_string : string -> t
@ -8,6 +16,8 @@ val to_string : t -> string
val to_long_string : t -> string val to_long_string : t -> string
(** Properties *) (** Properties *)
val to_int : t -> int
val of_int : int -> t
val to_charge : t -> Charge.t val to_charge : t -> Charge.t
val of_charge : Charge.t -> t val of_charge : Charge.t -> t
val covalent_radius : t -> Radius.t val covalent_radius : t -> Radius.t

View File

@ -1 +1,25 @@
type xyz_input = string * ( (Element.t * (float array)) array) type coordinates = ( (Element.t * (float array)) array)
let of_xyz_file ~filename =
let lexbuf =
let ic = open_in filename in
Lexing.from_channel ic
in
Xyz_parser.input Nuclei_lexer.read_all lexbuf
let of_zmt_file ~filename =
let ic = open_in filename in
let rec aux accu =
try
let line = input_line ic in
aux (line::accu)
with End_of_file ->
close_in ic;
List.rev accu
|> String.concat "\n"
in aux []
|> Zmatrix.of_string
|> Zmatrix.to_xyz
|> Array.map (fun (e,x,y,z) -> (e, [|x;y;z|]))

View File

@ -1,5 +1,5 @@
{ {
open Nuclei_parser open Xyz_parser
} }
let eol = ['\n'] let eol = ['\n']

View File

@ -1,4 +1,4 @@
/* Parses basis sets in GAMESS format */ /* Parses nuclear coordinates in xyz format */
%{ %{
exception InputError of string exception InputError of string
@ -11,19 +11,19 @@ exception InputError of string
%token <float> FLOAT %token <float> FLOAT
%token EOF %token EOF
%start input_xyz %start input
%type <Nuclei.xyz_input> input_xyz %type <(Element.t * (float array)) array> input
%% /* Grammar rules and actions follow */ %% /* Grammar rules and actions follow */
input_xyz: input:
| integer title atoms_xyz { | integer title atoms_xyz {
let len = List.length $3 in let len = List.length $3 in
if len <> $1 then if len <> $1 then
let error_msg = Printf.sprintf "%d atoms entered, expected %d" len $1 in let error_msg = Printf.sprintf "%d atoms entered, expected %d" len $1 in
raise (InputError error_msg) raise (InputError error_msg)
else else
($2,Array.of_list $3) Array.of_list $3
} }
integer: integer:
@ -51,21 +51,8 @@ atoms_list:
| { [] } | { [] }
| atoms_list WORD SPACE FLOAT SPACE FLOAT SPACE FLOAT EOL { (Element.of_string $2, [| $4;$6;$8 |]) :: $1 } | atoms_list WORD SPACE FLOAT SPACE FLOAT SPACE FLOAT EOL { (Element.of_string $2, [| $4;$6;$8 |]) :: $1 }
| atoms_list WORD SPACE FLOAT SPACE FLOAT SPACE FLOAT SPACE EOL { (Element.of_string $2, [| $4;$6;$8 |]) :: $1 } | atoms_list WORD SPACE FLOAT SPACE FLOAT SPACE FLOAT SPACE EOL { (Element.of_string $2, [| $4;$6;$8 |]) :: $1 }
| atoms_list INTEGER SPACE FLOAT SPACE FLOAT SPACE FLOAT EOL { (Element.of_int $2, [| $4;$6;$8 |]) :: $1 }
label: | atoms_list INTEGER SPACE FLOAT SPACE FLOAT SPACE FLOAT SPACE EOL { (Element.of_int $2, [| $4;$6;$8 |]) :: $1 }
| FLOAT { Zmatrix.Value $1 }
| WORD { Zmatrix.Label $1 }
first_line:
| WORD { $1 }
second_line:
| WORD INTEGER label { ($1,$2,$3) }
third_line:
| WORD INTEGER label INTEGER label { ($1,$2,$3,$4,$5) }
nth_line:
| WORD INTEGER label INTEGER label INTEGER label { ($1,$2,$3,$4,$5,$6,$7) }

View File

@ -200,20 +200,6 @@ let rotation_matrix axis angle =
let apply_rotation_matrix rot u = let apply_rotation_matrix rot u =
(dot rot.(0) u, dot rot.(1) u, dot rot.(2) u) (dot rot.(0) u, dot rot.(1) u, dot rot.(2) u)
let center_of_mass l =
let (x,y,z) =
let sum_mass, com =
Array.fold_left (fun (s,com) (e,x,y,z) ->
let mass =
Positive_float.to_float @@ Element.mass e
in
(s +. mass, ( mass |. (x,y,z) ) |+ com) )
(0., (0.,0.,0.)) l
in
(1. /. sum_mass) |. com
in
Printf.printf "%f %f %f\n" x y z ; (x,y,z)
let to_xyz (z,map) = let to_xyz (z,map) =
let result = let result =
Array.make (Array.length z) None Array.make (Array.length z) None
@ -300,14 +286,16 @@ let to_xyz (z,map) =
| Some x -> x | Some x -> x
| None -> failwith "Some atoms were not defined" ) result | None -> failwith "Some atoms were not defined" ) result
in in
Array.to_list result result
let to_xyz_string (l,map) = let to_xyz_string (l,map) =
String.concat "\n" String.concat "\n"
( to_xyz (l,map) ( to_xyz (l,map)
|> List.map (fun (e,x,y,z) -> |> Array.map (fun (e,x,y,z) ->
Printf.sprintf "%s %f %f %f\n" (Element.to_string e) x y z) ) Printf.sprintf "%s %f %f %f\n" (Element.to_string e) x y z)
|> Array.to_list
)

View File

@ -1,7 +1,22 @@
type units = type units =
| Bohr | Bohr
| Angstrom | Angstrom
;;
type angle_units =
| Degree
| Radian
let pi = acos (-1.)
let to_degree x =
assert (x <= 2.*.pi);
assert (x >= -2.*.pi);
x *. 180. /. pi
let to_radian x =
assert (x <= 360.);
assert (x >= -360.);
x *. pi /. 180.
let angstrom_to_bohr = 1. /. 0.52917721092 let angstrom_to_bohr = 1. /. 0.52917721092
let bohr_to_angstrom = 0.52917721092 let bohr_to_angstrom = 0.52917721092

View File

@ -1,7 +1,8 @@
type units = Bohr | Angstrom type units = Bohr | Angstrom
type angle_units = Degree | Radian
val to_radian : float -> float
val to_degree : float -> float
(** Conversion functions *)
val angstrom_to_bohr : float val angstrom_to_bohr : float
val bohr_to_angstrom : float val bohr_to_angstrom : float

73
Utils/Util.ml Normal file
View File

@ -0,0 +1,73 @@
(** Constants *)
let pi = acos (-1.)
let pi_inv = 1. /. pi
let two_over_sq_pi = 2. /. (sqrt pi)
let factmax = 150
(** Generalized Boys function. Uses GSL's incomplete Gamma function.
maxm : Maximum total angular momentum
*)
let boys_function ~maxm t =
begin
if t = 0. then
Array.init (maxm+1) (fun m -> 1. /. float_of_int (m+m+1))
else
let incomplete_gamma ~alpha x =
Gsl.Sf.gamma alpha *. ( Gsl.Sf.gamma_inc_P alpha x )
in
let t_inv =
1. /. t
in
let factor =
Array.make (maxm+1) (0.5, sqrt t_inv);
in
for i=1 to maxm
do
let (dm, f) = factor.(i-1) in
factor.(i) <- (dm +. 1., f *. t_inv);
done;
Array.map (fun (dm, f) -> (incomplete_gamma dm t ) *. 0.5 *. f) factor
end
let fact_memo =
let rec aux accu_l accu = function
| 0 -> aux [1.] 1. 1
| i when (i = factmax) ->
let x = (float_of_int factmax) *. accu in
List.rev (x::accu_l)
| i -> let x = (float_of_int i) *. accu in
aux (x::accu_l) x (i+1)
in
aux [] 0. 0
|> Array.of_list
(** Factorial function.
@raise Invalid_argument for negative or arguments >100. *)
let fact = function
| i when (i < 0) ->
raise (Invalid_argument "Argument of factorial should be non-negative")
| i when (i > 150) ->
raise (Invalid_argument "Result of factorial is infinite")
| i -> fact_memo.(i)
(** Integer powers of floats *)
let rec pow a = function
| 0 -> 1.
| 1 -> a
| 2 -> a *. a
| 3 -> a *. a *. a
| -1 -> 1. /. a
| n when (n<0) -> pow (1./.a) (-n)
| n ->
let b = pow a (n / 2) in
b *. b *. (if n mod 2 = 0 then 1. else a)
;;

2
_tags
View File

@ -1 +1 @@
true: package(str) true: package(str,gsl,zarith)