2 * This file is part of the GROMACS molecular simulation package.
4 * Copyright (c) 1991-2000, University of Groningen, The Netherlands.
5 * Copyright (c) 2001-2004, The GROMACS development team.
6 * Copyright (c) 2013,2014,2015,2016,2017 by the GROMACS development team.
7 * Copyright (c) 2018,2019,2020, by the GROMACS development team, led by
8 * Mark Abraham, David van der Spoel, Berk Hess, and Erik Lindahl,
9 * and including many others, as listed in the AUTHORS file in the
10 * top-level source directory and at http://www.gromacs.org.
12 * GROMACS is free software; you can redistribute it and/or
13 * modify it under the terms of the GNU Lesser General Public License
14 * as published by the Free Software Foundation; either version 2.1
15 * of the License, or (at your option) any later version.
17 * GROMACS is distributed in the hope that it will be useful,
18 * but WITHOUT ANY WARRANTY; without even the implied warranty of
19 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
20 * Lesser General Public License for more details.
22 * You should have received a copy of the GNU Lesser General Public
23 * License along with GROMACS; if not, see
24 * http://www.gnu.org/licenses, or write to the Free Software Foundation,
25 * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
27 * If you want to redistribute modifications to GROMACS, please
28 * consider that scientific software is very special. Version
29 * control is crucial - bugs must be traceable. We will be happy to
30 * consider code for inclusion in the official distribution, but
31 * derived work must not be called official GROMACS. Details are found
32 * in the README & COPYING files - if they are missing, get the
33 * official version at http://www.gromacs.org.
35 * To help us fund GROMACS development, we humbly ask that you cite
36 * the research papers on the package. Check out http://www.gromacs.org.
39 * \brief Defines code that writes energy-like quantities.
41 * \author Mark Abraham <mark.j.abraham@gmail.com>
42 * \author Paul Bauer <paul.bauer.q@gmail.com>
43 * \author Artem Zhmurov <zhmurov@gmail.com>
45 * \ingroup module_mdlib
49 #include "energyoutput.h"
58 #include "gromacs/awh/awh.h"
59 #include "gromacs/fileio/enxio.h"
60 #include "gromacs/fileio/gmxfio.h"
61 #include "gromacs/fileio/xvgr.h"
62 #include "gromacs/gmxlib/network.h"
63 #include "gromacs/listed_forces/disre.h"
64 #include "gromacs/listed_forces/orires.h"
65 #include "gromacs/math/functions.h"
66 #include "gromacs/math/units.h"
67 #include "gromacs/math/vec.h"
68 #include "gromacs/mdlib/constr.h"
69 #include "gromacs/mdlib/ebin.h"
70 #include "gromacs/mdlib/mdebin_bar.h"
71 #include "gromacs/mdrunutility/handlerestart.h"
72 #include "gromacs/mdtypes/energyhistory.h"
73 #include "gromacs/mdtypes/fcdata.h"
74 #include "gromacs/mdtypes/group.h"
75 #include "gromacs/mdtypes/inputrec.h"
76 #include "gromacs/mdtypes/md_enums.h"
77 #include "gromacs/mdtypes/state.h"
78 #include "gromacs/pbcutil/pbc.h"
79 #include "gromacs/pulling/pull.h"
80 #include "gromacs/topology/mtop_util.h"
81 #include "gromacs/trajectory/energyframe.h"
82 #include "gromacs/utility/arraysize.h"
83 #include "gromacs/utility/fatalerror.h"
84 #include "gromacs/utility/mdmodulenotification.h"
85 #include "gromacs/utility/smalloc.h"
86 #include "gromacs/utility/stringutil.h"
88 //! Labels for energy file quantities
90 static const char* conrmsd_nm
[] = { "Constr. rmsd", "Constr.2 rmsd" };
92 static std::array
<const char*, 3> boxs_nm
= { "Box-X", "Box-Y", "Box-Z" };
94 static std::array
<const char*, 6> tricl_boxs_nm
= { "Box-XX", "Box-YY", "Box-ZZ",
95 "Box-YX", "Box-ZX", "Box-ZY" };
97 static const char* vol_nm
[] = { "Volume" };
99 static const char* dens_nm
[] = { "Density" };
101 static const char* pv_nm
[] = { "pV" };
103 static const char* enthalpy_nm
[] = { "Enthalpy" };
105 static std::array
<const char*, 6> boxvel_nm
= { "Box-Vel-XX", "Box-Vel-YY", "Box-Vel-ZZ",
106 "Box-Vel-YX", "Box-Vel-ZX", "Box-Vel-ZY" };
108 const char* egrp_nm
[egNR
+ 1] = { "Coul-SR", "LJ-SR", "Buck-SR", "Coul-14", "LJ-14", nullptr };
114 /*! \brief Energy output class
116 * This is the collection of energy averages collected during mdrun, and to
117 * be written out to the .edr file.
119 * \todo Use more std containers.
120 * \todo Remove GMX_CONSTRAINTVIR
121 * \todo Write free-energy output also to energy file (after adding more tests)
123 EnergyOutput::EnergyOutput(ener_file
* fp_ene
,
124 const gmx_mtop_t
* mtop
,
125 const t_inputrec
* ir
,
126 const pull_t
* pull_work
,
129 const StartingBehavior startingBehavior
,
130 const MdModulesNotifier
& mdModulesNotifier
)
132 const char* ener_nm
[F_NRE
];
133 static const char* vir_nm
[] = { "Vir-XX", "Vir-XY", "Vir-XZ", "Vir-YX", "Vir-YY",
134 "Vir-YZ", "Vir-ZX", "Vir-ZY", "Vir-ZZ" };
135 static const char* sv_nm
[] = { "ShakeVir-XX", "ShakeVir-XY", "ShakeVir-XZ",
136 "ShakeVir-YX", "ShakeVir-YY", "ShakeVir-YZ",
137 "ShakeVir-ZX", "ShakeVir-ZY", "ShakeVir-ZZ" };
138 static const char* fv_nm
[] = { "ForceVir-XX", "ForceVir-XY", "ForceVir-XZ",
139 "ForceVir-YX", "ForceVir-YY", "ForceVir-YZ",
140 "ForceVir-ZX", "ForceVir-ZY", "ForceVir-ZZ" };
141 static const char* pres_nm
[] = { "Pres-XX", "Pres-XY", "Pres-XZ", "Pres-YX", "Pres-YY",
142 "Pres-YZ", "Pres-ZX", "Pres-ZY", "Pres-ZZ" };
143 static const char* surft_nm
[] = { "#Surf*SurfTen" };
144 static const char* mu_nm
[] = { "Mu-X", "Mu-Y", "Mu-Z" };
145 static const char* vcos_nm
[] = { "2CosZ*Vel-X" };
146 static const char* visc_nm
[] = { "1/Viscosity" };
147 static const char* baro_nm
[] = { "Barostat" };
149 const SimulationGroups
* groups
;
153 int i
, j
, ni
, nj
, n
, k
, kk
, ncon
, nset
;
156 if (EI_DYNAMICS(ir
->eI
))
158 delta_t_
= ir
->delta_t
;
165 groups
= &mtop
->groups
;
167 bBHAM
= (mtop
->ffparams
.numTypes() > 0) && (mtop
->ffparams
.functype
[0] == F_BHAM
);
168 b14
= (gmx_mtop_ftype_count(mtop
, F_LJ14
) > 0 || gmx_mtop_ftype_count(mtop
, F_LJC14_Q
) > 0);
170 ncon
= gmx_mtop_ftype_count(mtop
, F_CONSTR
);
171 nset
= gmx_mtop_ftype_count(mtop
, F_SETTLE
);
172 bool bConstr
= (ncon
> 0 || nset
> 0) && !isRerun
;
177 if (ncon
> 0 && ir
->eConstrAlg
== econtLINCS
)
181 bConstrVir_
= (getenv("GMX_CONSTRAINTVIR") != nullptr);
188 /* Energy monitoring */
189 for (i
= 0; i
< egNR
; i
++)
194 // Setting true only to those energy terms, that have active interactions and
195 // are not vsite terms (not VSITE2, VSITE3, VSITE3FD, VSITE3FAD, VSITE3OUT, VSITE4FD, VSITE4FDN, or VSITEN)
196 for (i
= 0; i
< F_NRE
; i
++)
198 bEner_
[i
] = (gmx_mtop_ftype_count(mtop
, i
) > 0)
199 && ((interaction_function
[i
].flags
& IF_VSITE
) == 0);
204 bEner_
[F_EKIN
] = EI_DYNAMICS(ir
->eI
);
205 bEner_
[F_ETOT
] = EI_DYNAMICS(ir
->eI
);
206 bEner_
[F_TEMP
] = EI_DYNAMICS(ir
->eI
);
208 bEner_
[F_ECONSERVED
] = integratorHasConservedEnergyQuantity(ir
);
209 bEner_
[F_PDISPCORR
] = (ir
->eDispCorr
!= edispcNO
);
210 bEner_
[F_PRES
] = true;
213 bEner_
[F_LJ
] = !bBHAM
;
214 bEner_
[F_BHAM
] = bBHAM
;
215 bEner_
[F_EQM
] = ir
->bQMMM
;
216 bEner_
[F_RF_EXCL
] = (EEL_RF(ir
->coulombtype
) && ir
->cutoff_scheme
== ecutsGROUP
);
217 bEner_
[F_COUL_RECIP
] = EEL_FULL(ir
->coulombtype
);
218 bEner_
[F_LJ_RECIP
] = EVDW_PME(ir
->vdwtype
);
219 bEner_
[F_LJ14
] = b14
;
220 bEner_
[F_COUL14
] = b14
;
221 bEner_
[F_LJC14_Q
] = false;
222 bEner_
[F_LJC_PAIRS_NB
] = false;
225 bEner_
[F_DVDL_COUL
] = (ir
->efep
!= efepNO
) && ir
->fepvals
->separate_dvdl
[efptCOUL
];
226 bEner_
[F_DVDL_VDW
] = (ir
->efep
!= efepNO
) && ir
->fepvals
->separate_dvdl
[efptVDW
];
227 bEner_
[F_DVDL_BONDED
] = (ir
->efep
!= efepNO
) && ir
->fepvals
->separate_dvdl
[efptBONDED
];
228 bEner_
[F_DVDL_RESTRAINT
] = (ir
->efep
!= efepNO
) && ir
->fepvals
->separate_dvdl
[efptRESTRAINT
];
229 bEner_
[F_DKDL
] = (ir
->efep
!= efepNO
) && ir
->fepvals
->separate_dvdl
[efptMASS
];
230 bEner_
[F_DVDL
] = (ir
->efep
!= efepNO
) && ir
->fepvals
->separate_dvdl
[efptFEP
];
232 bEner_
[F_CONSTR
] = false;
233 bEner_
[F_CONSTRNC
] = false;
234 bEner_
[F_SETTLE
] = false;
236 bEner_
[F_COUL_SR
] = true;
237 bEner_
[F_EPOT
] = true;
239 bEner_
[F_DISPCORR
] = (ir
->eDispCorr
!= edispcNO
);
240 bEner_
[F_DISRESVIOL
] = (gmx_mtop_ftype_count(mtop
, F_DISRES
) > 0);
241 bEner_
[F_ORIRESDEV
] = (gmx_mtop_ftype_count(mtop
, F_ORIRES
) > 0);
242 bEner_
[F_COM_PULL
] = ((ir
->bPull
&& pull_have_potential(pull_work
)) || ir
->bRot
);
244 MdModulesEnergyOutputToDensityFittingRequestChecker mdModulesAddOutputToDensityFittingFieldRequest
;
245 mdModulesNotifier
.simulationSetupNotifications_
.notify(&mdModulesAddOutputToDensityFittingFieldRequest
);
247 bEner_
[F_DENSITYFITTING
] = mdModulesAddOutputToDensityFittingFieldRequest
.energyOutputToDensityFitting_
;
250 // Counting the energy terms that will be printed and saving their names
252 for (i
= 0; i
< F_NRE
; i
++)
256 ener_nm
[f_nre_
] = interaction_function
[i
].longname
;
261 epc_
= isRerun
? epcNO
: ir
->epc
;
262 bDiagPres_
= !TRICLINIC(ir
->ref_p
) && !isRerun
;
263 ref_p_
= (ir
->ref_p
[XX
][XX
] + ir
->ref_p
[YY
][YY
] + ir
->ref_p
[ZZ
][ZZ
]) / DIM
;
264 bTricl_
= TRICLINIC(ir
->compress
) || TRICLINIC(ir
->deform
);
265 bDynBox_
= inputrecDynamicBox(ir
);
266 etc_
= isRerun
? etcNO
: ir
->etc
;
267 bNHC_trotter_
= inputrecNvtTrotter(ir
) && !isRerun
;
268 bPrintNHChains_
= ir
->bPrintNHChains
&& !isRerun
;
269 bMTTK_
= (inputrecNptTrotter(ir
) || inputrecNphTrotter(ir
)) && !isRerun
;
270 bMu_
= inputrecNeedMutot(ir
);
274 /* Pass NULL for unit to let get_ebin_space determine the units
275 * for interaction_function[i].longname
277 ie_
= get_ebin_space(ebin_
, f_nre_
, ener_nm
, nullptr);
280 /* This should be called directly after the call for ie_,
281 * such that iconrmsd_ follows directly in the list.
283 iconrmsd_
= get_ebin_space(ebin_
, nCrmsd_
, conrmsd_nm
, "");
287 ib_
= get_ebin_space(ebin_
, bTricl_
? tricl_boxs_nm
.size() : boxs_nm
.size(),
288 bTricl_
? tricl_boxs_nm
.data() : boxs_nm
.data(), unit_length
);
289 ivol_
= get_ebin_space(ebin_
, 1, vol_nm
, unit_volume
);
290 idens_
= get_ebin_space(ebin_
, 1, dens_nm
, unit_density_SI
);
293 ipv_
= get_ebin_space(ebin_
, 1, pv_nm
, unit_energy
);
294 ienthalpy_
= get_ebin_space(ebin_
, 1, enthalpy_nm
, unit_energy
);
299 isvir_
= get_ebin_space(ebin_
, asize(sv_nm
), sv_nm
, unit_energy
);
300 ifvir_
= get_ebin_space(ebin_
, asize(fv_nm
), fv_nm
, unit_energy
);
304 ivir_
= get_ebin_space(ebin_
, asize(vir_nm
), vir_nm
, unit_energy
);
305 ipres_
= get_ebin_space(ebin_
, asize(pres_nm
), pres_nm
, unit_pres_bar
);
306 isurft_
= get_ebin_space(ebin_
, asize(surft_nm
), surft_nm
, unit_surft_bar
);
308 if (epc_
== epcPARRINELLORAHMAN
|| epc_
== epcMTTK
)
310 ipc_
= get_ebin_space(ebin_
, bTricl_
? boxvel_nm
.size() : DIM
, boxvel_nm
.data(), unit_vel
);
314 imu_
= get_ebin_space(ebin_
, asize(mu_nm
), mu_nm
, unit_dipole_D
);
316 if (ir
->cos_accel
!= 0)
318 ivcos_
= get_ebin_space(ebin_
, asize(vcos_nm
), vcos_nm
, unit_vel
);
319 ivisc_
= get_ebin_space(ebin_
, asize(visc_nm
), visc_nm
, unit_invvisc_SI
);
322 /* Energy monitoring */
323 for (i
= 0; i
< egNR
; i
++)
327 bEInd_
[egCOULSR
] = true;
328 bEInd_
[egLJSR
] = true;
332 bEInd_
[egLJSR
] = false;
333 bEInd_
[egBHAMSR
] = true;
337 bEInd_
[egLJ14
] = true;
338 bEInd_
[egCOUL14
] = true;
341 for (i
= 0; (i
< egNR
); i
++)
348 n
= groups
->groups
[SimulationAtomGroupType::EnergyOutput
].size();
350 nE_
= (n
* (n
+ 1)) / 2;
357 for (k
= 0; (k
< nEc_
); k
++)
359 snew(gnm
[k
], STRLEN
);
361 for (i
= 0; (i
< gmx::ssize(groups
->groups
[SimulationAtomGroupType::EnergyOutput
])); i
++)
363 ni
= groups
->groups
[SimulationAtomGroupType::EnergyOutput
][i
];
364 for (j
= i
; (j
< gmx::ssize(groups
->groups
[SimulationAtomGroupType::EnergyOutput
])); j
++)
366 nj
= groups
->groups
[SimulationAtomGroupType::EnergyOutput
][j
];
367 for (k
= kk
= 0; (k
< egNR
); k
++)
371 sprintf(gnm
[kk
], "%s:%s-%s", egrp_nm
[k
], *(groups
->groupNames
[ni
]),
372 *(groups
->groupNames
[nj
]));
376 igrp_
[n
] = get_ebin_space(ebin_
, nEc_
, gnm
, unit_energy
);
380 for (k
= 0; (k
< nEc_
); k
++)
388 gmx_incons("Number of energy terms wrong");
392 nTC_
= isRerun
? 0 : groups
->groups
[SimulationAtomGroupType::TemperatureCoupling
].size();
393 nNHC_
= ir
->opts
.nhchainlength
; /* shorthand for number of NH chains */
396 nTCP_
= 1; /* assume only one possible coupling system for barostat
403 if (etc_
== etcNOSEHOOVER
)
407 mde_n_
= 2 * nNHC_
* nTC_
;
415 mdeb_n_
= 2 * nNHC_
* nTCP_
;
424 snew(tmp_r_
, mde_n_
);
425 // TODO redo the group name memory management to make it more clear
427 snew(grpnms
, std::max(mde_n_
, mdeb_n_
)); // Just in case mdeb_n_ > mde_n_
429 for (i
= 0; (i
< nTC_
); i
++)
431 ni
= groups
->groups
[SimulationAtomGroupType::TemperatureCoupling
][i
];
432 sprintf(buf
, "T-%s", *(groups
->groupNames
[ni
]));
433 grpnms
[i
] = gmx_strdup(buf
);
435 itemp_
= get_ebin_space(ebin_
, nTC_
, grpnms
, unit_temp_K
);
436 for (i
= 0; i
< nTC_
; i
++)
442 if (etc_
== etcNOSEHOOVER
)
448 for (i
= 0; (i
< nTC_
); i
++)
450 ni
= groups
->groups
[SimulationAtomGroupType::TemperatureCoupling
][i
];
451 bufi
= *(groups
->groupNames
[ni
]);
452 for (j
= 0; (j
< nNHC_
); j
++)
454 sprintf(buf
, "Xi-%d-%s", j
, bufi
);
455 grpnms
[2 * (i
* nNHC_
+ j
)] = gmx_strdup(buf
);
456 sprintf(buf
, "vXi-%d-%s", j
, bufi
);
457 grpnms
[2 * (i
* nNHC_
+ j
) + 1] = gmx_strdup(buf
);
460 itc_
= get_ebin_space(ebin_
, mde_n_
, grpnms
, unit_invtime
);
464 for (i
= 0; (i
< nTCP_
); i
++)
466 bufi
= baro_nm
[0]; /* All barostat DOF's together for now. */
467 for (j
= 0; (j
< nNHC_
); j
++)
469 sprintf(buf
, "Xi-%d-%s", j
, bufi
);
470 grpnms
[2 * (i
* nNHC_
+ j
)] = gmx_strdup(buf
);
471 sprintf(buf
, "vXi-%d-%s", j
, bufi
);
472 grpnms
[2 * (i
* nNHC_
+ j
) + 1] = gmx_strdup(buf
);
475 itcb_
= get_ebin_space(ebin_
, mdeb_n_
, grpnms
, unit_invtime
);
481 for (i
= 0; (i
< nTC_
); i
++)
483 ni
= groups
->groups
[SimulationAtomGroupType::TemperatureCoupling
][i
];
484 bufi
= *(groups
->groupNames
[ni
]);
485 sprintf(buf
, "Xi-%s", bufi
);
486 grpnms
[2 * i
] = gmx_strdup(buf
);
487 sprintf(buf
, "vXi-%s", bufi
);
488 grpnms
[2 * i
+ 1] = gmx_strdup(buf
);
490 itc_
= get_ebin_space(ebin_
, mde_n_
, grpnms
, unit_invtime
);
495 else if (etc_
== etcBERENDSEN
|| etc_
== etcYES
|| etc_
== etcVRESCALE
)
497 for (i
= 0; (i
< nTC_
); i
++)
499 ni
= groups
->groups
[SimulationAtomGroupType::TemperatureCoupling
][i
];
500 sprintf(buf
, "Lamb-%s", *(groups
->groupNames
[ni
]));
501 grpnms
[i
] = gmx_strdup(buf
);
503 itc_
= get_ebin_space(ebin_
, mde_n_
, grpnms
, "");
507 for (i
= 0; i
< allocated
; i
++)
513 nU_
= groups
->groups
[SimulationAtomGroupType::Acceleration
].size();
517 snew(grpnms
, 3 * nU_
);
518 for (i
= 0; (i
< nU_
); i
++)
520 ni
= groups
->groups
[SimulationAtomGroupType::Acceleration
][i
];
521 sprintf(buf
, "Ux-%s", *(groups
->groupNames
[ni
]));
522 grpnms
[3 * i
+ XX
] = gmx_strdup(buf
);
523 sprintf(buf
, "Uy-%s", *(groups
->groupNames
[ni
]));
524 grpnms
[3 * i
+ YY
] = gmx_strdup(buf
);
525 sprintf(buf
, "Uz-%s", *(groups
->groupNames
[ni
]));
526 grpnms
[3 * i
+ ZZ
] = gmx_strdup(buf
);
528 iu_
= get_ebin_space(ebin_
, 3 * nU_
, grpnms
, unit_vel
);
529 for (i
= 0; i
< 3 * nU_
; i
++)
536 /* Note that fp_ene should be valid on the master rank and null otherwise */
537 if (fp_ene
!= nullptr && startingBehavior
!= StartingBehavior::RestartWithAppending
)
539 do_enxnms(fp_ene
, &ebin_
->nener
, &ebin_
->enm
);
542 /* check whether we're going to write dh histograms */
544 if (ir
->fepvals
->separate_dhdl_file
== esepdhdlfileNO
)
546 /* Currently dh histograms are only written with dynamics */
547 if (EI_DYNAMICS(ir
->eI
))
551 mde_delta_h_coll_init(dhc_
, ir
);
554 snew(dE_
, ir
->fepvals
->n_lambda
);
559 snew(dE_
, ir
->fepvals
->n_lambda
);
564 snew(temperatures_
, ir
->fepvals
->n_lambda
);
565 numTemperatures_
= ir
->fepvals
->n_lambda
;
566 for (i
= 0; i
< ir
->fepvals
->n_lambda
; i
++)
568 temperatures_
[i
] = ir
->simtempvals
->temperatures
[i
];
573 numTemperatures_
= 0;
577 EnergyOutput::~EnergyOutput()
583 done_mde_delta_h_coll(dhc_
);
585 if (numTemperatures_
> 0)
587 sfree(temperatures_
);
593 /*! \brief Print a lambda vector to a string
595 * \param[in] fep The inputrec's FEP input data
596 * \param[in] i The index of the lambda vector
597 * \param[in] get_native_lambda Whether to print the native lambda
598 * \param[in] get_names Whether to print the names rather than the values
599 * \param[in,out] str The pre-allocated string buffer to print to.
601 static void print_lambda_vector(t_lambda
* fep
, int i
, bool get_native_lambda
, bool get_names
, char* str
)
606 for (j
= 0; j
< efptNR
; j
++)
608 if (fep
->separate_dvdl
[j
])
613 str
[0] = 0; /* reset the string */
616 str
+= sprintf(str
, "("); /* set the opening parenthesis*/
618 for (j
= 0; j
< efptNR
; j
++)
620 if (fep
->separate_dvdl
[j
])
624 if (get_native_lambda
&& fep
->init_lambda
>= 0)
626 str
+= sprintf(str
, "%.4f", fep
->init_lambda
);
630 str
+= sprintf(str
, "%.4f", fep
->all_lambda
[j
][i
]);
635 str
+= sprintf(str
, "%s", efpt_singular_names
[j
]);
637 /* print comma for the next item */
640 str
+= sprintf(str
, ", ");
647 /* and add the closing parenthesis */
652 FILE* open_dhdl(const char* filename
, const t_inputrec
* ir
, const gmx_output_env_t
* oenv
)
655 const char *dhdl
= "dH/d\\lambda", *deltag
= "\\DeltaH", *lambda
= "\\lambda",
656 *lambdastate
= "\\lambda state";
657 int i
, nsets
, nsets_de
, nsetsbegin
;
658 int n_lambda_terms
= 0;
659 t_lambda
* fep
= ir
->fepvals
; /* for simplicity */
660 t_expanded
* expand
= ir
->expandedvals
;
661 char lambda_vec_str
[STRLEN
], lambda_name_str
[STRLEN
];
666 bool write_pV
= false;
668 /* count the number of different lambda terms */
669 for (i
= 0; i
< efptNR
; i
++)
671 if (fep
->separate_dvdl
[i
])
677 std::string title
, label_x
, label_y
;
678 if (fep
->n_lambda
== 0)
680 title
= gmx::formatString("%s", dhdl
);
681 label_x
= gmx::formatString("Time (ps)");
682 label_y
= gmx::formatString("%s (%s %s)", dhdl
, unit_energy
, "[\\lambda]\\S-1\\N");
686 title
= gmx::formatString("%s and %s", dhdl
, deltag
);
687 label_x
= gmx::formatString("Time (ps)");
688 label_y
= gmx::formatString("%s and %s (%s %s)", dhdl
, deltag
, unit_energy
,
689 "[\\8l\\4]\\S-1\\N");
691 fp
= gmx_fio_fopen(filename
, "w+");
692 xvgr_header(fp
, title
.c_str(), label_x
, label_y
, exvggtXNY
, oenv
);
697 buf
= gmx::formatString("T = %g (K) ", ir
->opts
.ref_t
[0]);
699 if ((ir
->efep
!= efepSLOWGROWTH
) && (ir
->efep
!= efepEXPANDED
))
701 if ((fep
->init_lambda
>= 0) && (n_lambda_terms
== 1))
703 /* compatibility output */
704 buf
+= gmx::formatString("%s = %.4f", lambda
, fep
->init_lambda
);
708 print_lambda_vector(fep
, fep
->init_fep_state
, true, false, lambda_vec_str
);
709 print_lambda_vector(fep
, fep
->init_fep_state
, true, true, lambda_name_str
);
710 buf
+= gmx::formatString("%s %d: %s = %s", lambdastate
, fep
->init_fep_state
,
711 lambda_name_str
, lambda_vec_str
);
714 xvgr_subtitle(fp
, buf
.c_str(), oenv
);
718 if (fep
->dhdl_derivatives
== edhdlderivativesYES
)
720 nsets_dhdl
= n_lambda_terms
;
722 /* count the number of delta_g states */
723 nsets_de
= fep
->lambda_stop_n
- fep
->lambda_start_n
;
725 nsets
= nsets_dhdl
+ nsets_de
; /* dhdl + fep differences */
727 if (fep
->n_lambda
> 0 && (expand
->elmcmove
> elmcmoveNO
))
729 nsets
+= 1; /*add fep state for expanded ensemble */
732 if (fep
->edHdLPrintEnergy
!= edHdLPrintEnergyNO
)
734 nsets
+= 1; /* add energy to the dhdl as well */
738 if ((ir
->epc
!= epcNO
) && (fep
->n_lambda
> 0) && (fep
->init_lambda
< 0))
740 nsetsextend
+= 1; /* for PV term, other terms possible if required for
741 the reduced potential (only needed with foreign
742 lambda, and only output when init_lambda is not
743 set in order to maintain compatibility of the
747 std::vector
<std::string
> setname(nsetsextend
);
749 if (expand
->elmcmove
> elmcmoveNO
)
751 /* state for the fep_vals, if we have alchemical sampling */
752 setname
[s
++] = "Thermodynamic state";
755 if (fep
->edHdLPrintEnergy
!= edHdLPrintEnergyNO
)
758 switch (fep
->edHdLPrintEnergy
)
760 case edHdLPrintEnergyPOTENTIAL
:
761 energy
= gmx::formatString("%s (%s)", "Potential Energy", unit_energy
);
763 case edHdLPrintEnergyTOTAL
:
764 case edHdLPrintEnergyYES
:
765 default: energy
= gmx::formatString("%s (%s)", "Total Energy", unit_energy
);
767 setname
[s
++] = energy
;
770 if (fep
->dhdl_derivatives
== edhdlderivativesYES
)
772 for (i
= 0; i
< efptNR
; i
++)
774 if (fep
->separate_dvdl
[i
])
776 std::string derivative
;
777 if ((fep
->init_lambda
>= 0) && (n_lambda_terms
== 1))
779 /* compatibility output */
780 derivative
= gmx::formatString("%s %s %.4f", dhdl
, lambda
, fep
->init_lambda
);
784 double lam
= fep
->init_lambda
;
785 if (fep
->init_lambda
< 0)
787 lam
= fep
->all_lambda
[i
][fep
->init_fep_state
];
789 derivative
= gmx::formatString("%s %s = %.4f", dhdl
, efpt_singular_names
[i
], lam
);
791 setname
[s
++] = derivative
;
796 if (fep
->n_lambda
> 0)
798 /* g_bar has to determine the lambda values used in this simulation
799 * from this xvg legend.
802 if (expand
->elmcmove
> elmcmoveNO
)
804 nsetsbegin
= 1; /* for including the expanded ensemble */
811 if (fep
->edHdLPrintEnergy
!= edHdLPrintEnergyNO
)
815 nsetsbegin
+= nsets_dhdl
;
817 for (i
= fep
->lambda_start_n
; i
< fep
->lambda_stop_n
; i
++)
819 print_lambda_vector(fep
, i
, false, false, lambda_vec_str
);
821 if ((fep
->init_lambda
>= 0) && (n_lambda_terms
== 1))
823 /* for compatible dhdl.xvg files */
824 buf
= gmx::formatString("%s %s %s", deltag
, lambda
, lambda_vec_str
);
828 buf
= gmx::formatString("%s %s to %s", deltag
, lambda
, lambda_vec_str
);
833 /* print the temperature for this state if doing simulated annealing */
834 buf
+= gmx::formatString(
835 "T = %g (%s)", ir
->simtempvals
->temperatures
[s
- (nsetsbegin
)], unit_temp_K
);
841 setname
[s
++] = gmx::formatString("pV (%s)", unit_energy
);
844 xvgrLegend(fp
, setname
, oenv
);
853 void EnergyOutput::addDataAtEnergyStep(bool bDoDHDL
,
857 const gmx_enerdata_t
* enerd
,
858 const t_state
* state
,
860 const t_expanded
* expand
,
866 const gmx_ekindata_t
* ekind
,
868 const gmx::Constraints
* constr
)
870 int j
, k
, kk
, n
, gid
;
871 real crmsd
[2], tmp6
[6];
872 real bs
[tricl_boxs_nm
.size()], vol
, dens
, pv
, enthalpy
;
874 double store_dhdl
[efptNR
];
875 real store_energy
= 0;
878 /* Do NOT use the box in the state variable, but the separate box provided
879 * as an argument. This is because we sometimes need to write the box from
880 * the last timestep to match the trajectory frames.
882 add_ebin_indexed(ebin_
, ie_
, gmx::ArrayRef
<bool>(bEner_
), enerd
->term
, bSum
);
885 crmsd
[0] = constr
->rmsd();
886 add_ebin(ebin_
, iconrmsd_
, nCrmsd_
, crmsd
, false);
899 nboxs
= tricl_boxs_nm
.size();
906 nboxs
= boxs_nm
.size();
908 vol
= box
[XX
][XX
] * box
[YY
][YY
] * box
[ZZ
][ZZ
];
909 dens
= (tmass
* AMU
) / (vol
* NANO
* NANO
* NANO
);
910 add_ebin(ebin_
, ib_
, nboxs
, bs
, bSum
);
911 add_ebin(ebin_
, ivol_
, 1, &vol
, bSum
);
912 add_ebin(ebin_
, idens_
, 1, &dens
, bSum
);
916 /* This is pV (in kJ/mol). The pressure is the reference pressure,
917 not the instantaneous pressure */
918 pv
= vol
* ref_p_
/ PRESFAC
;
920 add_ebin(ebin_
, ipv_
, 1, &pv
, bSum
);
921 enthalpy
= pv
+ enerd
->term
[F_ETOT
];
922 add_ebin(ebin_
, ienthalpy_
, 1, &enthalpy
, bSum
);
927 add_ebin(ebin_
, isvir_
, 9, svir
[0], bSum
);
928 add_ebin(ebin_
, ifvir_
, 9, fvir
[0], bSum
);
932 add_ebin(ebin_
, ivir_
, 9, vir
[0], bSum
);
933 add_ebin(ebin_
, ipres_
, 9, pres
[0], bSum
);
934 tmp
= (pres
[ZZ
][ZZ
] - (pres
[XX
][XX
] + pres
[YY
][YY
]) * 0.5) * box
[ZZ
][ZZ
];
935 add_ebin(ebin_
, isurft_
, 1, &tmp
, bSum
);
937 if (epc_
== epcPARRINELLORAHMAN
|| epc_
== epcMTTK
)
939 tmp6
[0] = state
->boxv
[XX
][XX
];
940 tmp6
[1] = state
->boxv
[YY
][YY
];
941 tmp6
[2] = state
->boxv
[ZZ
][ZZ
];
942 tmp6
[3] = state
->boxv
[YY
][XX
];
943 tmp6
[4] = state
->boxv
[ZZ
][XX
];
944 tmp6
[5] = state
->boxv
[ZZ
][YY
];
945 add_ebin(ebin_
, ipc_
, bTricl_
? 6 : 3, tmp6
, bSum
);
949 add_ebin(ebin_
, imu_
, 3, mu_tot
, bSum
);
951 if (ekind
&& ekind
->cosacc
.cos_accel
!= 0)
953 vol
= box
[XX
][XX
] * box
[YY
][YY
] * box
[ZZ
][ZZ
];
954 dens
= (tmass
* AMU
) / (vol
* NANO
* NANO
* NANO
);
955 add_ebin(ebin_
, ivcos_
, 1, &(ekind
->cosacc
.vcos
), bSum
);
956 /* 1/viscosity, unit 1/(kg m^-1 s^-1) */
958 / (ekind
->cosacc
.cos_accel
/ (ekind
->cosacc
.vcos
* PICO
) * dens
959 * gmx::square(box
[ZZ
][ZZ
] * NANO
/ (2 * M_PI
)));
960 add_ebin(ebin_
, ivisc_
, 1, &tmp
, bSum
);
965 for (int i
= 0; (i
< nEg_
); i
++)
967 for (j
= i
; (j
< nEg_
); j
++)
969 gid
= GID(i
, j
, nEg_
);
970 for (k
= kk
= 0; (k
< egNR
); k
++)
974 eee
[kk
++] = enerd
->grpp
.ener
[k
][gid
];
977 add_ebin(ebin_
, igrp_
[n
], nEc_
, eee
, bSum
);
985 for (int i
= 0; (i
< nTC_
); i
++)
987 tmp_r_
[i
] = ekind
->tcstat
[i
].T
;
989 add_ebin(ebin_
, itemp_
, nTC_
, tmp_r_
, bSum
);
991 if (etc_
== etcNOSEHOOVER
)
993 /* whether to print Nose-Hoover chains: */
998 for (int i
= 0; (i
< nTC_
); i
++)
1000 for (j
= 0; j
< nNHC_
; j
++)
1003 tmp_r_
[2 * k
] = state
->nosehoover_xi
[k
];
1004 tmp_r_
[2 * k
+ 1] = state
->nosehoover_vxi
[k
];
1007 add_ebin(ebin_
, itc_
, mde_n_
, tmp_r_
, bSum
);
1011 for (int i
= 0; (i
< nTCP_
); i
++)
1013 for (j
= 0; j
< nNHC_
; j
++)
1016 tmp_r_
[2 * k
] = state
->nhpres_xi
[k
];
1017 tmp_r_
[2 * k
+ 1] = state
->nhpres_vxi
[k
];
1020 add_ebin(ebin_
, itcb_
, mdeb_n_
, tmp_r_
, bSum
);
1025 for (int i
= 0; (i
< nTC_
); i
++)
1027 tmp_r_
[2 * i
] = state
->nosehoover_xi
[i
];
1028 tmp_r_
[2 * i
+ 1] = state
->nosehoover_vxi
[i
];
1030 add_ebin(ebin_
, itc_
, mde_n_
, tmp_r_
, bSum
);
1034 else if (etc_
== etcBERENDSEN
|| etc_
== etcYES
|| etc_
== etcVRESCALE
)
1036 for (int i
= 0; (i
< nTC_
); i
++)
1038 tmp_r_
[i
] = ekind
->tcstat
[i
].lambda
;
1040 add_ebin(ebin_
, itc_
, nTC_
, tmp_r_
, bSum
);
1044 if (ekind
&& nU_
> 1)
1046 for (int i
= 0; (i
< nU_
); i
++)
1048 copy_rvec(ekind
->grpstat
[i
].u
, tmp_v_
[i
]);
1050 add_ebin(ebin_
, iu_
, 3 * nU_
, tmp_v_
[0], bSum
);
1053 ebin_increase_count(1, ebin_
, bSum
);
1055 // BAR + thermodynamic integration values
1056 if ((fp_dhdl_
|| dhc_
) && bDoDHDL
)
1058 const auto& foreignTerms
= enerd
->foreignLambdaTerms
;
1059 for (int i
= 0; i
< foreignTerms
.numLambdas(); i
++)
1061 /* zero for simulated tempering */
1062 dE_
[i
] = foreignTerms
.deltaH(i
);
1063 if (numTemperatures_
> 0)
1065 GMX_RELEASE_ASSERT(numTemperatures_
> state
->fep_state
,
1066 "Number of lambdas in state is bigger then in input record");
1068 numTemperatures_
>= foreignTerms
.numLambdas(),
1069 "Number of lambdas in energy data is bigger then in input record");
1070 /* MRS: is this right, given the way we have defined the exchange probabilities? */
1071 /* is this even useful to have at all? */
1072 dE_
[i
] += (temperatures_
[i
] / temperatures_
[state
->fep_state
] - 1.0) * enerd
->term
[F_EKIN
];
1078 fprintf(fp_dhdl_
, "%.4f", time
);
1079 /* the current free energy state */
1081 /* print the current state if we are doing expanded ensemble */
1082 if (expand
->elmcmove
> elmcmoveNO
)
1084 fprintf(fp_dhdl_
, " %4d", state
->fep_state
);
1086 /* total energy (for if the temperature changes */
1088 if (fep
->edHdLPrintEnergy
!= edHdLPrintEnergyNO
)
1090 switch (fep
->edHdLPrintEnergy
)
1092 case edHdLPrintEnergyPOTENTIAL
: store_energy
= enerd
->term
[F_EPOT
]; break;
1093 case edHdLPrintEnergyTOTAL
:
1094 case edHdLPrintEnergyYES
:
1095 default: store_energy
= enerd
->term
[F_ETOT
];
1097 fprintf(fp_dhdl_
, " %#.8g", store_energy
);
1100 if (fep
->dhdl_derivatives
== edhdlderivativesYES
)
1102 for (int i
= 0; i
< efptNR
; i
++)
1104 if (fep
->separate_dvdl
[i
])
1106 /* assumes F_DVDL is first */
1107 fprintf(fp_dhdl_
, " %#.8g", enerd
->term
[F_DVDL
+ i
]);
1111 for (int i
= fep
->lambda_start_n
; i
< fep
->lambda_stop_n
; i
++)
1113 fprintf(fp_dhdl_
, " %#.8g", dE_
[i
]);
1115 if (bDynBox_
&& bDiagPres_
&& (epc_
!= epcNO
) && foreignTerms
.numLambdas() > 0
1116 && (fep
->init_lambda
< 0))
1118 fprintf(fp_dhdl_
, " %#.8g", pv
); /* PV term only needed when
1119 there are alternate state
1120 lambda and we're not in
1121 compatibility mode */
1123 fprintf(fp_dhdl_
, "\n");
1124 /* and the binary free energy output */
1126 if (dhc_
&& bDoDHDL
)
1129 for (int i
= 0; i
< efptNR
; i
++)
1131 if (fep
->separate_dvdl
[i
])
1133 /* assumes F_DVDL is first */
1134 store_dhdl
[idhdl
] = enerd
->term
[F_DVDL
+ i
];
1138 store_energy
= enerd
->term
[F_ETOT
];
1139 /* store_dh is dE */
1140 mde_delta_h_coll_add_dh(dhc_
, static_cast<double>(state
->fep_state
), store_energy
, pv
,
1141 store_dhdl
, dE_
+ fep
->lambda_start_n
, time
);
1146 void EnergyOutput::recordNonEnergyStep()
1148 ebin_increase_count(1, ebin_
, false);
1151 void EnergyOutput::printHeader(FILE* log
, int64_t steps
, double time
)
1158 "Step", "Time", gmx_step_str(steps
, buf
), time
);
1161 void EnergyOutput::printStepToEnergyFile(ener_file
* fp_ene
,
1175 fr
.nsteps
= ebin_
->nsteps
;
1177 fr
.nsum
= ebin_
->nsum
;
1178 fr
.nre
= (bEne
) ? ebin_
->nener
: 0;
1180 int ndisre
= bDR
? fcd
->disres
->npair
: 0;
1181 /* these are for the old-style blocks (1 subblock, only reals), because
1182 there can be only one per ID for these */
1186 /* Optional additional old-style (real-only) blocks. */
1187 for (int i
= 0; i
< enxNR
; i
++)
1192 if (bOR
&& fcd
->orires
->nr
> 0)
1194 t_oriresdata
& orires
= *fcd
->orires
;
1195 diagonalize_orires_tensors(&orires
);
1196 nr
[enxOR
] = orires
.nr
;
1197 block
[enxOR
] = orires
.otav
;
1199 nr
[enxORI
] = (orires
.oinsl
!= orires
.otav
) ? orires
.nr
: 0;
1200 block
[enxORI
] = orires
.oinsl
;
1201 id
[enxORI
] = enxORI
;
1202 nr
[enxORT
] = orires
.nex
* 12;
1203 block
[enxORT
] = orires
.eig
;
1204 id
[enxORT
] = enxORT
;
1207 /* whether we are going to write anything out: */
1208 if (fr
.nre
|| ndisre
|| nr
[enxOR
] || nr
[enxORI
])
1210 /* the old-style blocks go first */
1212 for (int i
= 0; i
< enxNR
; i
++)
1219 add_blocks_enxframe(&fr
, fr
.nblock
);
1220 for (int b
= 0; b
< fr
.nblock
; b
++)
1222 add_subblocks_enxblock(&(fr
.block
[b
]), 1);
1223 fr
.block
[b
].id
= id
[b
];
1224 fr
.block
[b
].sub
[0].nr
= nr
[b
];
1226 fr
.block
[b
].sub
[0].type
= xdr_datatype_float
;
1227 fr
.block
[b
].sub
[0].fval
= block
[b
];
1229 fr
.block
[b
].sub
[0].type
= xdr_datatype_double
;
1230 fr
.block
[b
].sub
[0].dval
= block
[b
];
1234 /* check for disre block & fill it. */
1239 add_blocks_enxframe(&fr
, fr
.nblock
);
1241 add_subblocks_enxblock(&(fr
.block
[db
]), 2);
1242 const t_disresdata
& disres
= *fcd
->disres
;
1243 fr
.block
[db
].id
= enxDISRE
;
1244 fr
.block
[db
].sub
[0].nr
= ndisre
;
1245 fr
.block
[db
].sub
[1].nr
= ndisre
;
1247 fr
.block
[db
].sub
[0].type
= xdr_datatype_float
;
1248 fr
.block
[db
].sub
[1].type
= xdr_datatype_float
;
1249 fr
.block
[db
].sub
[0].fval
= disres
.rt
;
1250 fr
.block
[db
].sub
[1].fval
= disres
.rm3tav
;
1252 fr
.block
[db
].sub
[0].type
= xdr_datatype_double
;
1253 fr
.block
[db
].sub
[1].type
= xdr_datatype_double
;
1254 fr
.block
[db
].sub
[0].dval
= disres
.rt
;
1255 fr
.block
[db
].sub
[1].dval
= disres
.rm3tav
;
1258 /* here we can put new-style blocks */
1260 /* Free energy perturbation blocks */
1263 mde_delta_h_coll_handle_block(dhc_
, &fr
, fr
.nblock
);
1266 /* we can now free & reset the data in the blocks */
1269 mde_delta_h_coll_reset(dhc_
);
1272 /* AWH bias blocks. */
1273 if (awh
!= nullptr) // TODO: add boolean flag.
1275 awh
->writeToEnergyFrame(step
, &fr
);
1278 /* do the actual I/O */
1279 do_enx(fp_ene
, &fr
);
1282 /* We have stored the sums, so reset the sum history */
1283 reset_ebin_sums(ebin_
);
1289 if (bOR
&& fcd
->orires
->nr
> 0)
1291 print_orires_log(log
, fcd
->orires
);
1294 fprintf(log
, " Energies (%s)\n", unit_energy
);
1295 pr_ebin(log
, ebin_
, ie_
, f_nre_
+ nCrmsd_
, 5, eprNORMAL
, true);
1300 void EnergyOutput::printAnnealingTemperatures(FILE* log
, const SimulationGroups
* groups
, t_grpopts
* opts
)
1306 for (int i
= 0; i
< opts
->ngtc
; i
++)
1308 if (opts
->annealing
[i
] != eannNO
)
1310 fprintf(log
, "Current ref_t for group %s: %8.1f\n",
1311 *(groups
->groupNames
[groups
->groups
[SimulationAtomGroupType::TemperatureCoupling
][i
]]),
1320 void EnergyOutput::printAverages(FILE* log
, const SimulationGroups
* groups
)
1322 if (ebin_
->nsum_sim
<= 0)
1326 fprintf(log
, "Not enough data recorded to report energy averages\n");
1333 char buf1
[22], buf2
[22];
1335 fprintf(log
, "\t<====== ############### ==>\n");
1336 fprintf(log
, "\t<==== A V E R A G E S ====>\n");
1337 fprintf(log
, "\t<== ############### ======>\n\n");
1339 fprintf(log
, "\tStatistics over %s steps using %s frames\n",
1340 gmx_step_str(ebin_
->nsteps_sim
, buf1
), gmx_step_str(ebin_
->nsum_sim
, buf2
));
1343 fprintf(log
, " Energies (%s)\n", unit_energy
);
1344 pr_ebin(log
, ebin_
, ie_
, f_nre_
+ nCrmsd_
, 5, eprAVER
, true);
1349 pr_ebin(log
, ebin_
, ib_
, bTricl_
? tricl_boxs_nm
.size() : boxs_nm
.size(), 5, eprAVER
, true);
1354 fprintf(log
, " Constraint Virial (%s)\n", unit_energy
);
1355 pr_ebin(log
, ebin_
, isvir_
, 9, 3, eprAVER
, false);
1357 fprintf(log
, " Force Virial (%s)\n", unit_energy
);
1358 pr_ebin(log
, ebin_
, ifvir_
, 9, 3, eprAVER
, false);
1363 fprintf(log
, " Total Virial (%s)\n", unit_energy
);
1364 pr_ebin(log
, ebin_
, ivir_
, 9, 3, eprAVER
, false);
1366 fprintf(log
, " Pressure (%s)\n", unit_pres_bar
);
1367 pr_ebin(log
, ebin_
, ipres_
, 9, 3, eprAVER
, false);
1372 fprintf(log
, " Total Dipole (%s)\n", unit_dipole_D
);
1373 pr_ebin(log
, ebin_
, imu_
, 3, 3, eprAVER
, false);
1379 int padding
= 8 - strlen(unit_energy
);
1380 fprintf(log
, "%*sEpot (%s) ", padding
, "", unit_energy
);
1381 for (int i
= 0; (i
< egNR
); i
++)
1385 fprintf(log
, "%12s ", egrp_nm
[i
]);
1391 for (int i
= 0; (i
< nEg_
); i
++)
1393 int ni
= groups
->groups
[SimulationAtomGroupType::EnergyOutput
][i
];
1394 for (int j
= i
; (j
< nEg_
); j
++)
1396 int nj
= groups
->groups
[SimulationAtomGroupType::EnergyOutput
][j
];
1398 14 - (strlen(*(groups
->groupNames
[ni
])) + strlen(*(groups
->groupNames
[nj
])));
1399 fprintf(log
, "%*s%s-%s", padding
, "", *(groups
->groupNames
[ni
]),
1400 *(groups
->groupNames
[nj
]));
1401 pr_ebin(log
, ebin_
, igrp_
[n
], nEc_
, nEc_
, eprAVER
, false);
1409 pr_ebin(log
, ebin_
, itemp_
, nTC_
, 4, eprAVER
, true);
1414 fprintf(log
, "%15s %12s %12s %12s\n", "Group", "Ux", "Uy", "Uz");
1415 for (int i
= 0; (i
< nU_
); i
++)
1417 int ni
= groups
->groups
[SimulationAtomGroupType::Acceleration
][i
];
1418 fprintf(log
, "%15s", *groups
->groupNames
[ni
]);
1419 pr_ebin(log
, ebin_
, iu_
+ 3 * i
, 3, 3, eprAVER
, false);
1426 void EnergyOutput::fillEnergyHistory(energyhistory_t
* enerhist
) const
1428 const t_ebin
* const ebin
= ebin_
;
1430 enerhist
->nsteps
= ebin
->nsteps
;
1431 enerhist
->nsum
= ebin
->nsum
;
1432 enerhist
->nsteps_sim
= ebin
->nsteps_sim
;
1433 enerhist
->nsum_sim
= ebin
->nsum_sim
;
1437 /* This will only actually resize the first time */
1438 enerhist
->ener_ave
.resize(ebin
->nener
);
1439 enerhist
->ener_sum
.resize(ebin
->nener
);
1441 for (int i
= 0; i
< ebin
->nener
; i
++)
1443 enerhist
->ener_ave
[i
] = ebin
->e
[i
].eav
;
1444 enerhist
->ener_sum
[i
] = ebin
->e
[i
].esum
;
1448 if (ebin
->nsum_sim
> 0)
1450 /* This will only actually resize the first time */
1451 enerhist
->ener_sum_sim
.resize(ebin
->nener
);
1453 for (int i
= 0; i
< ebin
->nener
; i
++)
1455 enerhist
->ener_sum_sim
[i
] = ebin
->e_sim
[i
].esum
;
1460 mde_delta_h_coll_update_energyhistory(dhc_
, enerhist
);
1464 void EnergyOutput::restoreFromEnergyHistory(const energyhistory_t
& enerhist
)
1466 unsigned int nener
= static_cast<unsigned int>(ebin_
->nener
);
1468 if ((enerhist
.nsum
> 0 && nener
!= enerhist
.ener_sum
.size())
1469 || (enerhist
.nsum_sim
> 0 && nener
!= enerhist
.ener_sum_sim
.size()))
1472 "Mismatch between number of energies in run input (%u) and checkpoint file (%zu "
1474 nener
, enerhist
.ener_sum
.size(), enerhist
.ener_sum_sim
.size());
1477 ebin_
->nsteps
= enerhist
.nsteps
;
1478 ebin_
->nsum
= enerhist
.nsum
;
1479 ebin_
->nsteps_sim
= enerhist
.nsteps_sim
;
1480 ebin_
->nsum_sim
= enerhist
.nsum_sim
;
1482 for (int i
= 0; i
< ebin_
->nener
; i
++)
1484 ebin_
->e
[i
].eav
= (enerhist
.nsum
> 0 ? enerhist
.ener_ave
[i
] : 0);
1485 ebin_
->e
[i
].esum
= (enerhist
.nsum
> 0 ? enerhist
.ener_sum
[i
] : 0);
1486 ebin_
->e_sim
[i
].esum
= (enerhist
.nsum_sim
> 0 ? enerhist
.ener_sum_sim
[i
] : 0);
1490 mde_delta_h_coll_restore_energyhistory(dhc_
, enerhist
.deltaHForeignLambdas
.get());
1494 int EnergyOutput::numEnergyTerms() const
1496 return ebin_
->nener
;