forked from RudolfWeeber/espresso-virtual-sites
-
Notifications
You must be signed in to change notification settings - Fork 0
/
energy.h
441 lines (397 loc) · 12.4 KB
/
energy.h
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
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
/* This file is part of the ESPResSo distribution (http://www.espresso.mpg.de).
It is therefore subject to the ESPResSo license agreement which you accepted upon receiving the distribution
and by which you are legally bound while utilizing this file in any form or way.
There is NO WARRANTY, not even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
You should have received a copy of that license along with this program;
if not, refer to http://www.espresso.mpg.de/license.html where its current version can be found, or
write to Max-Planck-Institute for Polymer Research, Theory Group, PO Box 3148, 55021 Mainz, Germany.
Copyright (c) 2002-2006; all rights reserved unless otherwise stated. */
/** \file energy.h
Implementation of the energy calculation.
*/
#ifndef ENERGY_H
#define ENERGY_H
#include "utils.h"
#include "integrate.h"
#include "statistics.h"
#include "thermostat.h"
#include "communication.h"
/* include the energy files */
#include "p3m.h"
#include "lj.h"
#include "ljgen.h"
#include "steppot.h"
#include "hertzian.h"
#include "bmhtf-nacl.h"
#include "buckingham.h"
#include "soft_sphere.h"
#include "ljcos.h"
#include "ljcos2.h"
#include "ljangle.h"
#include "tab.h"
#include "overlap.h"
#include "gb.h"
#include "fene.h"
#include "harmonic.h"
#include "subt_lj.h"
#include "angle.h"
#include "angledist.h"
#include "dihedral.h"
#include "debye_hueckel.h"
#include "endangledist.h"
#include "reaction_field.h"
#include "mmm1d.h"
#include "mmm2d.h"
#include "maggs.h"
#include "morse.h"
#include "ewald.h"
#include "elc.h"
#include "mdlc_correction.h"
/** \name Exported Variables */
/************************************************************/
/*@{*/
///
extern Observable_stat energy, total_energy;
/*@}*/
/** \name Exported Functions */
/************************************************************/
/*@{*/
/** parallel energy calculation.
@param result non-zero only on master node; will contain the cumulative over all nodes. */
void energy_calc(double *result);
/** Calculate non bonded energies between a pair of particles.
@param p1 pointer to particle 1.
@param p2 pointer to particle 2.
@param ia_params the interaction parameters between the two particles
@param d vector between p1 and p2.
@param dist distance between p1 and p2.
@param dist2 distance squared between p1 and p2.
@return the short ranged interaction energy between the two particles
*/
MDINLINE double calc_non_bonded_pair_energy(Particle *p1, Particle *p2,
IA_parameters *ia_params,
double d[3], double dist, double dist2)
{
double ret = 0;
#ifdef NO_INTRA_NB
if (p1->p.mol_id==p2->p.mol_id) return 0;
#endif
#ifdef MOL_CUT
//You may want to put a correction factor for smoothing function else then theta
if (checkIfParticlesInteractViaMolCut(p1,p2,ia_params)==0) return 0;
#endif
#ifdef LENNARD_JONES
/* lennard jones */
ret += lj_pair_energy(p1,p2,ia_params,d,dist);
#endif
#ifdef LENNARD_JONES_GENERIC
/* Generic lennard jones */
ret += ljgen_pair_energy(p1,p2,ia_params,d,dist);
#endif
#ifdef LJ_ANGLE
/* Directional LJ */
ret += ljangle_pair_energy(p1,p2,ia_params,d,dist);
#endif
#ifdef SMOOTH_STEP
/* smooth step */
ret += SmSt_pair_energy(p1,p2,ia_params,d,dist,dist2);
#endif
#ifdef HERTZIAN
/* Hertzian potential */
ret += hertzian_pair_energy(p1,p2,ia_params,d,dist,dist2);
#endif
#ifdef BMHTF_NACL
/* BMHTF NaCl */
ret += BMHTF_pair_energy(p1,p2,ia_params,d,dist,dist2);
#endif
#ifdef MORSE
/* morse */
ret +=morse_pair_energy(p1,p2,ia_params,d,dist);
#endif
#ifdef BUCKINGHAM
/* lennard jones */
ret += buck_pair_energy(p1,p2,ia_params,d,dist);
#endif
#ifdef SOFT_SPHERE
/* soft-sphere */
ret += soft_pair_energy(p1,p2,ia_params,d,dist);
#endif
#ifdef LJCOS2
/* lennard jones */
ret += ljcos2_pair_energy(p1,p2,ia_params,d,dist);
#endif
#ifdef TABULATED
/* tabulated */
ret += tabulated_pair_energy(p1,p2,ia_params,d,dist);
#endif
#ifdef LJCOS
/* lennard jones cosine */
ret += ljcos_pair_energy(p1,p2,ia_params,d,dist);
#endif
#ifdef GAY_BERNE
/* Gay-Berne */
ret += gb_pair_energy(p1,p2,ia_params,d,dist);
#endif
#ifdef INTER_RF
ret += interrf_pair_energy(p1,p2,ia_params,dist);
#endif
return ret;
}
/** Add non bonded energies and short range coulomb between a pair of particles.
@param p1 pointer to particle 1.
@param p2 pointer to particle 2.
@param d vector between p1 and p2.
@param dist distance between p1 and p2.
@param dist2 distance squared between p1 and p2. */
MDINLINE void add_non_bonded_pair_energy(Particle *p1, Particle *p2, double d[3],
double dist, double dist2)
{
IA_parameters *ia_params = get_ia_param(p1->p.type,p2->p.type);
#if defined(ELECTROSTATICS) || defined(MAGNETOSTATICS)
double ret = 0;
#endif
*obsstat_nonbonded(&energy, p1->p.type, p2->p.type) +=
calc_non_bonded_pair_energy(p1, p2, ia_params, d, dist, dist2);
#ifdef ELECTROSTATICS
if (coulomb.method != COULOMB_NONE) {
/* real space coulomb */
switch (coulomb.method) {
#ifdef ELP3M
case COULOMB_P3M:
ret = p3m_coulomb_pair_energy(p1->p.q*p2->p.q,d,dist2,dist);
break;
case COULOMB_ELC_P3M:
ret = p3m_coulomb_pair_energy(p1->p.q*p2->p.q,d,dist2,dist);
if (elc_params.dielectric_contrast_on)
ret += 0.5*ELC_P3M_dielectric_layers_energy_contribution(p1,p2);
break;
#endif
case COULOMB_EWALD:
ret = ewald_coulomb_pair_energy(p1,p2,d,dist2,dist);
break;
case COULOMB_DH:
ret = dh_coulomb_pair_energy(p1,p2,dist);
break;
case COULOMB_RF:
ret = rf_coulomb_pair_energy(p1,p2,dist);
break;
case COULOMB_INTER_RF:
//this is done above as interaction
ret = 0;
break;
case COULOMB_MMM1D:
ret = mmm1d_coulomb_pair_energy(p1,p2,d,dist2,dist);
break;
case COULOMB_MMM2D:
ret = mmm2d_coulomb_pair_energy(p1->p.q*p2->p.q,d,dist2,dist);
break;
default :
ret = 0.;
}
energy.coulomb[0] += ret;
}
#endif
#ifdef MAGNETOSTATICS
if (coulomb.Dmethod != DIPOLAR_NONE) {
ret=0;
switch (coulomb.Dmethod) {
#ifdef ELP3M
case DIPOLAR_MDLC_P3M:
//fall trough
case DIPOLAR_P3M:
ret = p3m_dipolar_pair_energy(p1,p2,d,dist2,dist);
break;
#endif
}
energy.dipolar[0] += ret;
}
#endif
}
/** Calculate bonded energies for one particle.
@param p1 particle for which to calculate energies
*/
MDINLINE void add_bonded_energy(Particle *p1)
{
char *errtxt;
Particle *p2, *p3 = NULL, *p4 = NULL;
Bonded_ia_parameters *iaparams;
int i, type_num, type, n_partners, bond_broken;
double ret=0, dx[3] = {0, 0, 0};
i = 0;
while(i<p1->bl.n) {
type_num = p1->bl.e[i++];
iaparams = &bonded_ia_params[type_num];
type = iaparams->type;
n_partners = iaparams->num;
/* fetch particle 2, which is always needed */
p2 = local_particles[p1->bl.e[i++]];
if (!p2) {
errtxt = runtime_error(128 + 2*TCL_INTEGER_SPACE);
ERROR_SPRINTF(errtxt,"{069 bond broken between particles %d and %d (particles not stored on the same node)} ",
p1->p.identity, p1->bl.e[i-1]);
return;
}
/* fetch particle 3 eventually */
if (n_partners >= 2) {
p3 = local_particles[p1->bl.e[i++]];
if (!p3) {
errtxt = runtime_error(128 + 3*TCL_INTEGER_SPACE);
ERROR_SPRINTF(errtxt,"{070 bond broken between particles %d, %d and %d (particles not stored on the same node)} ",
p1->p.identity, p1->bl.e[i-2], p1->bl.e[i-1]);
return;
}
}
/* fetch particle 4 eventually */
if (n_partners >= 3) {
p4 = local_particles[p1->bl.e[i++]];
if (!p4) {
errtxt = runtime_error(128 + 4*TCL_INTEGER_SPACE);
ERROR_SPRINTF(errtxt,"{071 bond broken between particles %d, %d, %d and %d (particles not stored on the same node)} ",
p1->p.identity, p1->bl.e[i-3], p1->bl.e[i-2], p1->bl.e[i-1]);
return;
}
}
/* similar to the force, we prepare the center-center vector */
if (n_partners == 1)
get_mi_vector(dx, p1->r.p, p2->r.p);
switch(type) {
case BONDED_IA_FENE:
bond_broken = fene_pair_energy(p1, p2, iaparams, dx, &ret);
break;
case BONDED_IA_HARMONIC:
bond_broken = harmonic_pair_energy(p1, p2, iaparams, dx, &ret);
break;
#ifdef LENNARD_JONES
case BONDED_IA_SUBT_LJ:
bond_broken = subt_lj_pair_energy(p1, p2, iaparams, dx, &ret);
break;
#endif
#ifdef BOND_ANGLE
case BONDED_IA_ANGLE:
bond_broken = angle_energy(p1, p2, p3, iaparams, &ret);
break;
#endif
#ifdef BOND_ANGLEDIST
case BONDED_IA_ANGLEDIST:
bond_broken = angledist_energy(p1, p2, p3, iaparams, &ret);
break;
#endif
#ifdef BOND_ENDANGLEDIST
case BONDED_IA_ENDANGLEDIST:
bond_broken = endangledist_pair_energy(p1, p2, iaparams, dx, &ret);
break;
#endif
case BONDED_IA_DIHEDRAL:
bond_broken = dihedral_energy(p2, p1, p3, p4, iaparams, &ret);
break;
#ifdef BOND_CONSTRAINT
case BONDED_IA_RIGID_BOND:
bond_broken = 0;
ret = 0;
break;
#endif
#ifdef TABULATED
case BONDED_IA_TABULATED:
switch(iaparams->p.tab.type) {
case TAB_BOND_LENGTH:
bond_broken = tab_bond_energy(p1, p2, iaparams, dx, &ret);
break;
case TAB_BOND_ANGLE:
bond_broken = tab_angle_energy(p1, p2, p3, iaparams, &ret);
break;
case TAB_BOND_DIHEDRAL:
bond_broken = tab_dihedral_energy(p2, p1, p3, p4, iaparams, &ret);
break;
default :
errtxt = runtime_error(128 + TCL_INTEGER_SPACE);
ERROR_SPRINTF(errtxt,"{072 add_bonded_energy: tabulated bond type of atom %d unknown\n", p1->p.identity);
return;
}
break;
#endif
#ifdef OVERLAPPED
case BONDED_IA_OVERLAPPED:
switch(iaparams->p.overlap.type) {
case OVERLAP_BOND_LENGTH:
bond_broken = overlap_bond_energy(p1, p2, iaparams, dx, &ret);
break;
case OVERLAP_BOND_ANGLE:
bond_broken = overlap_angle_energy(p1, p2, p3, iaparams, &ret);
break;
case OVERLAP_BOND_DIHEDRAL:
bond_broken = overlap_dihedral_energy(p2, p1, p3, p4, iaparams, &ret);
break;
default :
errtxt = runtime_error(128 + TCL_INTEGER_SPACE);
ERROR_SPRINTF(errtxt,"{072 add_bonded_energy: overlapped bond type of atom %d unknown\n", p1->p.identity);
return;
}
break;
#endif
#ifdef BOND_VIRTUAL
case BONDED_IA_VIRTUAL_BOND:
bond_broken = 0;
ret = 0;
break;
#endif
default :
errtxt = runtime_error(128 + TCL_INTEGER_SPACE);
ERROR_SPRINTF(errtxt,"{073 add_bonded_energy: bond type of atom %d unknown\n", p1->p.identity);
return;
}
switch (n_partners) {
case 1:
if (bond_broken) {
char *errtext = runtime_error(128 + 2*TCL_INTEGER_SPACE);
ERROR_SPRINTF(errtext,"{074 bond broken between particles %d and %d} ",
p1->p.identity, p2->p.identity);
continue;
}
break;
case 2:
if (bond_broken) {
char *errtext = runtime_error(128 + 3*TCL_INTEGER_SPACE);
ERROR_SPRINTF(errtext,"{075 bond broken between particles %d, %d and %d} ",
p1->p.identity, p2->p.identity, p3->p.identity);
continue;
}
break;
case 3:
if (bond_broken) {
char *errtext = runtime_error(128 + 4*TCL_INTEGER_SPACE);
ERROR_SPRINTF(errtext,"{076 bond broken between particles %d, %d, %d and %d} ",
p1->p.identity, p2->p.identity, p3->p.identity, p4->p.identity);
continue;
}
}
*obsstat_bonded(&energy, type_num) += ret;
}
}
/** Calculate kinetic energies for one particle.
@param p1 particle for which to calculate energies
*/
MDINLINE void add_kinetic_energy(Particle *p1)
{
#ifdef VIRTUAL_SITES
if (ifParticleIsVirtual(p1)) return;
#endif
/* kinetic energy */
energy.data.e[0] += (SQR(p1->m.v[0]) + SQR(p1->m.v[1]) + SQR(p1->m.v[2]))*PMASS(*p1);
#ifdef ROTATION
#ifdef ROTATIONAL_INERTIA
/* the rotational part is added to the total kinetic energy;
Here we use the rotational inertia */
energy.data.e[0] += (SQR(p1->m.omega[0])*p1->p.rinertia[0] +
SQR(p1->m.omega[1])*p1->p.rinertia[1] +
SQR(p1->m.omega[2])*p1->p.rinertia[2])*time_step*time_step;
#else
/* the rotational part is added to the total kinetic energy;
at the moment, we assume unit inertia tensor I=(1,1,1) */
energy.data.e[0] += (SQR(p1->m.omega[0]) + SQR(p1->m.omega[1]) + SQR(p1->m.omega[2]))*time_step*time_step;
#endif
#endif
}
/** implementation of analyze energy */
int parse_and_print_energy(Tcl_Interp *interp, int argc, char **argv);
/*@}*/
#endif