-
Notifications
You must be signed in to change notification settings - Fork 12
/
Copy pathex02_h2_comparison.py
181 lines (164 loc) · 5.67 KB
/
ex02_h2_comparison.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
"""Compare results for relaxation of H2 molecule
using VaspInteractive vs Vasp internal vs Vasp SinglePoint + BFGS
using the same force stop criteria the energy results should be almost identical
Even if the WAVECAR is reload during the relaxation, VaspInteractive is still faster
than classic Vasp due to shorter spin-up time
"""
import numpy as np
import os
import tempfile
from time import time
from ase.atoms import Atoms
from ase.optimize import BFGS
from vasp_interactive import VaspInteractive
from vasp_interactive.vasp_interactive import parse_outcar_iterations
from ase.calculators.vasp import Vasp
from pathlib import Path
d = 0.9575
h2_root = Atoms("H2", positions=[(d, 0, 0), (0, 0, 0)], cell=[8, 8, 8], pbc=True)
rootdir = Path(__file__).resolve().parents[1] / "sandbox"
fmax = 0.005
ediff = 1e-7
def run_with_vasp_interactive():
"""Relaxation using interactive VASP"""
print("*" * 40)
print("Running relaxation using VaspInteractive")
print("*" * 40)
h2 = h2_root.copy()
# with tempfile.TemporaryDirectory() as tmpdir:
with tempfile.TemporaryDirectory() as tmpdir:
calc = VaspInteractive(
istart=0,
ismear=0,
ediff=ediff,
xc="pbe",
kpts=(1, 1, 1), # not important, just keeps it faster
directory=tmpdir,
# Only relevant for vasp5
parse_vaspout=True,
)
# Best practice of VaspInteractive is to use it as ContextManager
t_ = time()
with calc:
h2.calc = calc
dyn = BFGS(h2)
# Now ASE-BFGS controls the relaxation, not VASP
dyn.run(fmax=fmax)
print(
"Final energy using VaspInteractive: ",
h2.get_potential_energy(),
)
# Read the iterations, should be outside the context
# Omit the last ionic step since it's a dummy
n_ion, n_elec = calc.read_all_iterations()
print(f"Ionic steps: {n_ion - 1}")
print(f"Electronic scf per ionic cycle: {n_elec[:-1]}")
print(f"Wall time: {time() - t_:.2f} s")
print("")
def run_with_vasp():
"""Relaxation using VASP internal routines"""
print("*" * 40)
print("Running relaxation using internal VASP routines")
print("*" * 40)
h2 = h2_root.copy()
with tempfile.TemporaryDirectory() as tmpdir:
calc = Vasp(
istart=0,
ismear=0,
xc="pbe",
kpts=(1, 1, 1),
directory=tmpdir,
ibrion=2,
nsw=10,
ediff=ediff,
ediffg=-fmax,
)
# Best practice of VaspInteractive is to use it as ContextManager
h2.calc = calc
t_ = time()
print(
"Final energy from Vasp internal routines: ",
h2.get_potential_energy(),
)
# Read the iterations
n_ion, n_elec = parse_outcar_iterations(calc.load_file("OUTCAR"))
print(f"Ionic steps: {n_ion}")
print(f"Electronic scf per ionic cycle: {n_elec}")
print(f"Wall time: {time() - t_:.2f} s")
print("")
def run_with_vasp_bfgs():
"""Relaxation using BFGS + classic Vasp calculator"""
print("*" * 40)
print("Running relaxation using BFGS + classic Vasp (no WAVECAR reload)")
print("*" * 40)
h2 = h2_root.copy()
with tempfile.TemporaryDirectory() as tmpdir:
calc = Vasp(
ismear=0,
istart=0,
xc="pbe",
ediff=ediff,
kpts=(1, 1, 1), # not important, just keeps it faster
directory=tmpdir,
ibrion=-1,
nsw=0,
)
t_ = time()
h2.calc = calc
dyn = BFGS(h2)
# Use manual force threshold in order to read the iterations
n_elec = []
n_ion = 1
f = np.abs(h2.get_forces()).max()
n_elec.append(calc.read_number_of_iterations())
while f > fmax:
dyn.step()
n_ion += 1
f = np.abs(h2.get_forces()).max()
n_elec.append(calc.read_number_of_iterations())
print("Final energy from Vasp + BFGS: ", h2.get_potential_energy())
# Read the iterations
print(f"Ionic steps: {n_ion}")
print(f"Electronic scf per ionic cycle: {n_elec}")
print(f"Wall time: {time() - t_:.2f} s")
print("")
def run_with_vasp_bfgs_cache():
"""Relaxation using BFGS + classic Vasp calculator"""
print("*" * 40)
print("Running relaxation using BFGS + classic Vasp (with WAVECAR reload)")
print("*" * 40)
h2 = h2_root.copy()
with tempfile.TemporaryDirectory() as tmpdir:
calc = Vasp(
ismear=0,
xc="pbe",
ediff=ediff,
kpts=(1, 1, 1), # not important, just keeps it faster
directory=tmpdir,
ibrion=-1,
nsw=0,
)
t_ = time()
h2.calc = calc
dyn = BFGS(h2)
# Use manual force threshold in order to read the iterations
n_elec = []
n_ion = 1
f = np.abs(h2.get_forces()).max()
n_elec.append(calc.read_number_of_iterations())
while f > fmax:
dyn.step()
n_ion += 1
f = np.abs(h2.get_forces()).max()
n_elec.append(calc.read_number_of_iterations())
print("Final energy from Vasp + BFGS: ", h2.get_potential_energy())
# Read the iterations
print(f"Ionic steps: {n_ion}")
print(f"Electronic scf per ionic cycle: {n_elec}")
print(f"Wall time: {time() - t_:.2f} s")
print("")
if __name__ == "__main__":
run_with_vasp_interactive()
run_with_vasp()
run_with_vasp_bfgs()
run_with_vasp_bfgs_cache()