2 * This file is part of the GROMACS molecular simulation package.
4 * Copyright (c) 2013,2014,2015,2016,2018,2019, by the GROMACS development team, led by
5 * Mark Abraham, David van der Spoel, Berk Hess, and Erik Lindahl,
6 * and including many others, as listed in the AUTHORS file in the
7 * top-level source directory and at http://www.gromacs.org.
9 * GROMACS is free software; you can redistribute it and/or
10 * modify it under the terms of the GNU Lesser General Public License
11 * as published by the Free Software Foundation; either version 2.1
12 * of the License, or (at your option) any later version.
14 * GROMACS is distributed in the hope that it will be useful,
15 * but WITHOUT ANY WARRANTY; without even the implied warranty of
16 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
17 * Lesser General Public License for more details.
19 * You should have received a copy of the GNU Lesser General Public
20 * License along with GROMACS; if not, see
21 * http://www.gnu.org/licenses, or write to the Free Software Foundation,
22 * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
24 * If you want to redistribute modifications to GROMACS, please
25 * consider that scientific software is very special. Version
26 * control is crucial - bugs must be traceable. We will be happy to
27 * consider code for inclusion in the official distribution, but
28 * derived work must not be called official GROMACS. Details are found
29 * in the README & COPYING files - if they are missing, get the
30 * official version at http://www.gromacs.org.
32 * To help us fund GROMACS development, we humbly ask that you cite
33 * the research papers on the package. Check out http://www.gromacs.org.
37 * Implements gmx::AnalysisDataLifetimeModule.
39 * \author Teemu Murtola <teemu.murtola@gmail.com>
40 * \ingroup module_analysisdata
52 #include "gromacs/analysisdata/dataframe.h"
53 #include "gromacs/analysisdata/datastorage.h"
58 /********************************************************************
59 * AnalysisDataLifetimeModule
63 * Private implementation class for AnalysisDataLifetimeModule.
65 * \ingroup module_analysisdata
67 class AnalysisDataLifetimeModule::Impl
70 //! Container type for storing a histogram during the calculation.
71 typedef std::deque
<int> LifetimeHistogram
;
73 //! Initializes the implementation class with empty/default values.
74 Impl() : firstx_(0.0), lastx_(0.0), frameCount_(0), bCumulative_(false)
79 * Increments a lifetime histogram with a single lifetime.
81 * \param[in] dataSet Index of the histogram to increment.
82 * \param[in] lifetime Lifetime to add to the histogram.
84 void addLifetime(int dataSet
, int lifetime
)
88 LifetimeHistogram
&histogram
= lifetimeHistograms_
[dataSet
];
89 if (histogram
.size() < static_cast<unsigned>(lifetime
))
91 histogram
.resize(lifetime
, 0);
93 ++histogram
[lifetime
- 1];
97 //! X value of the first frame (used for determining output spacing).
99 //! X value of the last frame (used for determining output spacing).
101 //! Total number of frames (used for normalization and output spacing).
103 //! Whether to add subintervals of longer intervals explicitly.
106 * Length of current continuously present interval for each data column.
108 * While frame N has been processed, stores the length of an interval
109 * for each data column where that column has been continuously present
110 * up to and including frame N.
112 std::vector
<std::vector
<int> > currentLifetimes_
;
114 * Accumulated lifetime histograms for each data set.
116 std::vector
<LifetimeHistogram
> lifetimeHistograms_
;
119 AnalysisDataLifetimeModule::AnalysisDataLifetimeModule()
124 AnalysisDataLifetimeModule::~AnalysisDataLifetimeModule()
128 void AnalysisDataLifetimeModule::setCumulative(bool bCumulative
)
130 impl_
->bCumulative_
= bCumulative
;
133 int AnalysisDataLifetimeModule::flags() const
135 return efAllowMulticolumn
| efAllowMissing
| efAllowMultipleDataSets
;
139 AnalysisDataLifetimeModule::dataStarted(AbstractAnalysisData
*data
)
141 impl_
->currentLifetimes_
.reserve(data
->dataSetCount());
142 impl_
->lifetimeHistograms_
.reserve(data
->dataSetCount());
143 for (int i
= 0; i
< data
->dataSetCount(); ++i
)
145 impl_
->currentLifetimes_
.emplace_back(data
->columnCount(i
), 0);
146 impl_
->lifetimeHistograms_
.emplace_back();
151 AnalysisDataLifetimeModule::frameStarted(const AnalysisDataFrameHeader
&header
)
153 if (header
.index() == 0)
155 impl_
->firstx_
= header
.x();
157 impl_
->lastx_
= header
.x();
158 ++impl_
->frameCount_
;
159 // TODO: Check the input for even spacing.
163 AnalysisDataLifetimeModule::pointsAdded(const AnalysisDataPointSetRef
&points
)
165 const int dataSet
= points
.dataSetIndex();
166 // This assumption is strictly not necessary, but this is how the
167 // framework works currently, and makes the code below simpler.
168 GMX_ASSERT(points
.firstColumn() == 0
169 && points
.lastColumn() == ssize(impl_
->currentLifetimes_
[dataSet
]) - 1,
170 "Point set should cover all columns");
171 for (int i
= 0; i
< points
.columnCount(); ++i
)
173 // TODO: Perhaps add control over how this is determined?
174 const bool bPresent
= points
.present(i
) && points
.y(i
) > 0.0;
177 ++impl_
->currentLifetimes_
[dataSet
][i
];
179 else if (impl_
->currentLifetimes_
[dataSet
][i
] > 0)
181 impl_
->addLifetime(dataSet
, impl_
->currentLifetimes_
[dataSet
][i
]);
182 impl_
->currentLifetimes_
[dataSet
][i
] = 0;
188 AnalysisDataLifetimeModule::frameFinished(const AnalysisDataFrameHeader
& /*header*/)
193 AnalysisDataLifetimeModule::dataFinished()
195 // Need to process the elements present in the last frame explicitly.
196 for (size_t i
= 0; i
< impl_
->currentLifetimes_
.size(); ++i
)
198 for (size_t j
= 0; j
< impl_
->currentLifetimes_
[i
].size(); ++j
)
200 impl_
->addLifetime(i
, impl_
->currentLifetimes_
[i
][j
]);
203 impl_
->currentLifetimes_
.clear();
205 if (impl_
->bCumulative_
)
207 // Sum up subintervals of longer intervals into the histograms
208 // if explicitly requested.
209 std::vector
<Impl::LifetimeHistogram
>::iterator histogram
;
210 for (histogram
= impl_
->lifetimeHistograms_
.begin();
211 histogram
!= impl_
->lifetimeHistograms_
.end();
214 Impl::LifetimeHistogram::iterator shorter
, longer
;
215 for (shorter
= histogram
->begin(); shorter
!= histogram
->end(); ++shorter
)
217 int subIntervalCount
= 2;
218 for (longer
= shorter
+ 1; longer
!= histogram
->end();
219 ++longer
, ++subIntervalCount
)
221 // Interval of length shorter contains (longer - shorter + 1)
222 // continuous intervals of length longer.
223 *shorter
+= subIntervalCount
* (*longer
);
229 // X spacing is determined by averaging from the first and last frame
230 // instead of first two frames to avoid rounding issues.
232 (impl_
->frameCount_
> 1)
233 ? (impl_
->lastx_
- impl_
->firstx_
) / (impl_
->frameCount_
- 1)
235 setXAxis(0.0, spacing
);
237 // Determine output dimensionality to cover all the histograms.
238 setColumnCount(impl_
->lifetimeHistograms_
.size());
239 std::vector
<Impl::LifetimeHistogram
>::const_iterator histogram
;
240 size_t maxLifetime
= 1;
241 for (histogram
= impl_
->lifetimeHistograms_
.begin();
242 histogram
!= impl_
->lifetimeHistograms_
.end();
245 maxLifetime
= std::max(maxLifetime
, histogram
->size());
247 setRowCount(maxLifetime
);
249 // Fill up the output data from the histograms.
252 for (histogram
= impl_
->lifetimeHistograms_
.begin();
253 histogram
!= impl_
->lifetimeHistograms_
.end();
254 ++histogram
, ++column
)
257 Impl::LifetimeHistogram::const_iterator i
;
258 for (i
= histogram
->begin(); i
!= histogram
->end(); ++i
, ++row
)
260 // Normalize by the number of frames, taking into account the
261 // length of the interval (interval of length N cannot start in
262 // N-1 last frames). row is always smaller than frameCount_
263 // because of the histograms have at most frameCount_ entries.
264 const real normalized
= *i
/ static_cast<real
>(impl_
->frameCount_
- row
);
265 value(row
, column
).setValue(normalized
);
267 // Pad the rest of the histogram with zeros to match the longest
269 for (; row
< rowCount(); ++row
)
271 value(row
, column
).setValue(0.0);
274 impl_
->lifetimeHistograms_
.clear();