This repository has been archived by the owner on Mar 5, 2018. It is now read-only.
-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathAtomsInterface.jl
343 lines (244 loc) · 11.6 KB
/
AtomsInterface.jl
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
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
"""
Add Documentation for `AtomsInterface`; at the moment just a collection of
notes.
* Position arrays in `Atoms.jl` should always be 3 x N and modules ought
to convert automatically
*
"""
module AtomsInterface
using Prototypes
export AbstractAtoms,
positions, set_positions!,
neighbours, set_neighbours!,
AbstractCalculator,
potential_energy, forces, potential_energy_d,
r_sum, r_dot,
Preconditioner,
set_array!, get_array,
maxforce
# Lexicon.update! is something that many users will have installed,
try
import Lexicon.update!
catch
nothing
end
#######################################################################
# AbstractAtoms
#######################################################################
"""`AbstractAtoms`: the abstract supertype for storing atomistic
configurations. A basic implementation might simply store a list of positions.
"""
abstract AbstractAtoms
# the following are all dummy method definitions that just throw an error if a
# method hasn't been implemented
import Base.length
"Return number of atoms"
@protofun length(::AbstractAtoms)
import Base.getindex
"Return position(s) of atom(s) `i`"
@protofun getindex(::AbstractAtoms, i)
import Base.setindex!
"Set position(s) of atom(s) `i`"
@protofun setindex!(::AbstractAtoms, x, i)
"Return reference to positions of all atoms as a `3 x N` array."
@protofun get_positions(::AbstractAtoms)
"alias for `get_positions`"
positions(a::AbstractAtoms) = get_positions(a)
"Set positions of all atoms as a `3 x N` array."
@protofun set_positions!(::AbstractAtoms, ::Any)
# #######################################################################
# # AbstractAtomsX
# # a still abstract implementation of much of the functionality
# # of AbstractAtoms, assuming that atom positions are just standard
# # vectors and are stored in a field .X
# #######################################################################
# """`AbstractAtomsX`: implementations of this abstract sub-type
# *must* have a field `.X`, with d rows and nX columns, where nX is the
# number of atoms and d the space dimension in which the atoms live, normally
# d = 3. With this guaranteed, much of the functionality of the
# `AbstractAtoms` interface can already be implemented.
# """
# abstract AbstractAtomsX <: AbstractAtoms
# # note: this is not part of the interface, but I am finding it very useful
# # maybe it could be included in the general interface as well?
# # note the default for update!(atm::AbstractAtomsX, i) is that
# # the "update-all" version is called, but this can be overloaded to
# # prevent that.
# "Tell the 'listeners' that the atom configuration has changed."
# update!(atm::AbstractAtomsX) = nothing
# update!(atm::AbstractAtomsX, i::Integer) = update!(atm::AbstractAtomsX)
# # implementations of the standard interface
# length(atm::AbstractAtomsX) = size(atm.X, 2)
# getindex(atm::AbstractAtomsX, i) = atm.X[:, i]
# setindex!(atm::AbstractAtomsX, x, i) = begin atm.X[:, i] = x; update!(atm, i); end
# get_positions(atm::AbstractAtomsX) = atm.X
# set_positions!(atm::AbstractAtomsX, X) = begin atm.X = X; update!(atm) end
# ddim(atm::AbstractAtomsX) = size(atm.X, 1)
# rdim(atm::AbstractAtomsX) = size(atm.X, 1)
#######################################################################
# NEIGHBOURLIST
#######################################################################
"Abstract supertype of neighbourlists."
abstract AbstractNeighbourList
##### for now, we don't attach neighbourlist to atoms,
##### so I just comment these out for now.
############################################################
# # each atoms object should probably have a neighbourlist attached even
# # if it is just a trivial "all atoms are neighbours" thing
# "Return the neighbourlist attached to the atoms object."
# @protofun get_neighbourlist(::AbstractAtoms)
# "alias for `get_neighbourlist`"
# neighbourlist(a::AbstractAtoms) = get_neighbourlist(a)
# "attach a neighbourlist to an atoms object"
# @protofun set_neighbourlist!(::AbstractAtoms, ::AbstractNeighbourList)
"tell the neighbourlist that the atoms object has changed"
@protofun update!(::AbstractNeighbourList, ::AbstractAtoms)
"""`get_neigs(n, a; rcut==-1) -> Ineig, r`: returns a list of
indices of neighbours of `n` and distances `r`. if rcut == -1 (default), then
the cut-off radius originally supplied to the neighbourlist is used.
If rcut > nlist.rcut, then an error is thrown.
(Note that `r` is computed anyhow, so it is passed by default; it can be
discarded if not needed.)
"""
get_neighbours(n, atm::AbstractAtoms; rcut=-1) =
get_neighbours(n, neighbourlist(atm), atm; rcut=-1)
"alias for `get_neighbours`"
neighbours(n, atm::AbstractAtoms; rcut=-1) = get_neighbours(n, atm; rcut=rcut)
# DISCUSS: maybe it would be useful to also return the list of
# direction-vectors? It was probably computed anyhow?
"""Returns `(Ineig, r)` where `Ineig is an integer vector with indices
of neighbour atoms of `n` and r the a Float vector with their distances."""
@protofun get_neighbours(n::Integer, neigs::AbstractNeighbourList,
atm::AbstractAtoms; rcut=-1)
neighbours(n::Integer, neigs::AbstractNeighbourList,
atm::AbstractAtoms; rcut=-1) =
get_neighbours(n, neigs, atm; rcut=rcut)
#######################################################################
# CONSTRAINTS
#######################################################################
# # constraints implement boundary conditions, or other types
# # of constraints; the details of this interface are still a bit fuzzy for me
# """Abstract supertype for constraints; these are objects that implement boundary
# conditions."""
# abstract AbstractConstraints
# "Return constraints attached to the atoms object (if one exists)"
# @protofun get_constraints(::AbstractAtoms)
# "Attach a constraints object to the atoms object"
# @protofun set_constraints!(::AbstractAtoms, ::AbstractConstraints)
# """Returns a bare `Vector{T <: FloatingPoint}` object containing the degrees of
# freedom describing the state of the simulation. This function should be
# overloaded for concrete implementions of `AbstractAtoms` and
# `AbstractConstraints`.
# Alternative wrapper function
# get_dofs(atm::AbstractAtoms) = get_dofs(atm, get_constraints(atm))
# """
# @protofun get_dofs(a::AbstractAtoms, c::AbstractConstraints)
# get_dofs(atm::AbstractAtoms) = get_dofs(atm, get_constraints(atm))
# """Takes a \"dual\" array (3 x lenght) and applies the dual constraints
# to obtain effective forces acting on dofs. Returns a vector of the same
# length as dofs."""
# @protofun forces_to_dofs{T <: AbstractFloat}(f::Matrix{T}, con::AbstractConstraints)
#######################################################################
# CALCULATOR
#######################################################################
"""`AbstractCalculator`: the abstract supertype of calculators. These
store model information, and are linked to the implementation of energy, forces,
and so forth. """
abstract AbstractCalculator
# The following two functions are, in my view, superfluous for the
# general interface since - typically - a calculator need not be attached
# to a specific atoms object. However, it does turn out to be convenient to
# have the calculator attached to the atoms object; see multiple convenience
# wrapper functions below.
"Return calculator attached to the atoms object (if one exists)"
@protofun calculator(::AbstractAtoms)
"Attach a calculator to the atoms object"
@protofun set_calculator!(::AbstractAtoms)
"Returns the cut-off radius of the potential."
@protofun cutoff(::AbstractCalculator)
## ==============================
## get_E and has_E : total energy
"""Return the total energy of a configuration of atoms `a`, using the calculator
`c`. Alternatively can call `get_E(a) = get_E(a, get_calculator(a))` """
@protofun potential_energy(a::AbstractAtoms, c::AbstractCalculator)
potential_energy(a::AbstractAtoms) =
potential_energy(a, get_calculator(a))
# "Returns `true` if the calculator `c` can compute total energies."
# @protofun has_E(c::AbstractCalculator)
# has_E(a::AbstractAtoms) = has_E(get_calculator(a))
## ==============================
## get_Es and has_Es : site energy
# """`site_energies(idx, a::AbstractAtoms, c::AbstractCalculator)`:
# Returns an `Vector{Float64}` of site energies of a configuration of
# atoms `a`, using the calculator `c`. If idx==[] (default), then *all*
# site energies are returned, otherwise those corresponding to the list
# of indices idx.
# """
# @protofun site_energies(idx, a::AbstractAtoms, c::AbstractCalculator)
# get_Es(idx, a::AbstractAtoms) = get_Es(idx, a, get_calculator(a))
# "Returns `true` if the calculator `c` can compute site energies."
# @protofun has_Es(c::AbstractCalculator)
# has_Es(a::AbstractAtoms) = has_Es(get_calculator(a))
# """same calling convention as get_Es.
# Returns a tuple `(dEs, Ineigs)`, where `dEs` is d x nneigs and
# `Ineigs` is the list of neighbours for which the forces have been computed
# """
# @protofun get_dEs(idx, a::AbstractAtoms, c::AbstractCalculator)
# get_dEs(idx, a::AbstractAtoms) = get_dEs(idx, a, get_calculator(calc))
# ==========================================================
# get_dE
# (every calculator needs this, so there is no has_dE())
"""Returns the negative gradient of the total energy in the format `3 x length`.
Alternatively, one can call the simplified form
forces(a::AbstractAtoms) = forces(a, calculator(a))
provided that a has an attached calculator is avilable."""
@protofun forces(a::AbstractAtoms, c::AbstractCalculator)
forces(a::AbstractAtoms) = forces(a, calculator(a))
potential_energy_d(a,c) = - forces(a, c)
# """Returns the gradient of the total energy in the format `3 x length`.
# Alternatively, one can call the simplified form
# get_dE(a::AbstractAtoms) = get_dE(a, get_calculator(a))
# provided that a.calc is avilable."""
# @protofun get_dE(a::AbstractAtoms, c::AbstractCalculator)
# get_dE(a::AbstractAtoms) = get_dE(a, get_calculator(a))
# "Return gradient of total energy taken w.r.t. dofs, i.e., as a long vector. "
# get_dE_dofs(a::AbstractAtoms, calc::AbstractCalculator, con::AbstractConstraints) =
# forces_to_dofs(get_dE(a, calc), con)
# get_dE_dofs(a::AbstractAtoms) = get_dE(a, get_calculator(a),
# get_constraints(a) )
# ==========================================================
### Some useful utility functions
############################################################
### Robust Summation and Dot Products
############################################################
## Should this go into LjOptim?
"Robust summation. Uses `sum_kbn`."
r_sum(a) = sum_kbn(a)
## NOTE: if I see this correctly, then r_dot allocates a temporary
## vector, which is likely quite a performance overhead.
## probably, we want to re-implement this.
## TODO: new version without the intermediate allocation
"Robust inner product. Defined as `r_dot(a, b) = r_sum(a .* b)`"
r_dot(a, b) = r_sum(a[:] .* b[:])
# ==================================================
# more abstract types that eventually need to be
# arranged in a better way
abstract Preconditioner
"store an array; assumes that obj.arrays exists"
function set_array!(obj::Any, key, val)
obj.arrays[key] = val
end
"""retrieve an array; instead of raising an exception if `key` does not exist,
this function returns `nothing`"""
function get_array(obj::Any, key)
if haskey(obj.arrays, key)
return obj.arrays[key]
else
return nothing
end
end
"""`max_force(f::AbstractArray{2})`:
For a 3 x N array `f`, return the maximum of `|f[:,n]|₂`.
"""
maxforce(f) = sqrt(maximum(sumabs2(f, 1)))
end