2 * This file is part of the GROMACS molecular simulation package.
4 * Copyright (c) 2009,2010,2011,2012,2013, by the GROMACS development team, led by
5 * David van der Spoel, Berk Hess, Erik Lindahl, and including many
6 * others, as listed in the AUTHORS file in the top-level source
7 * 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 functions in evaluate.h.
40 * One of the major bottlenecks for selection performance is that all the
41 * evaluation is carried out for atoms.
42 * There are several cases when the evaluation could be done for residues
43 * or molecules instead, including keywords that select by residue and
44 * cases where residue centers are used as reference positions.
45 * Implementing this would require a mechanism for recognizing whether
46 * something can be evaluated by residue/molecule instead by atom, and
47 * converting selections by residue/molecule into selections by atom
50 * \author Teemu Murtola <teemu.murtola@gmail.com>
51 * \ingroup module_selection
55 #include "gromacs/legacyheaders/maths.h"
56 #include "gromacs/legacyheaders/smalloc.h"
57 #include "gromacs/legacyheaders/vec.h"
59 #include "gromacs/selection/indexutil.h"
60 #include "gromacs/selection/poscalc.h"
61 #include "gromacs/selection/selection.h"
62 #include "gromacs/selection/selmethod.h"
63 #include "gromacs/utility/exceptions.h"
64 #include "gromacs/utility/gmxassert.h"
68 #include "selectioncollection-impl.h"
71 using gmx::SelectionTreeElement;
72 using gmx::SelectionTreeElementPointer;
78 * Reserves memory for a selection element from the evaluation memory pool.
80 * This class implements RAII semantics for allocating memory for selection
81 * element values from a selection evaluation memory pool.
83 * \ingroup module_selection
85 class MempoolSelelemReserver
88 //! Constructs a reserver without initial reservation.
89 MempoolSelelemReserver() {}
91 * Constructs a reserver with initial reservation.
93 * \param[in,out] sel Selection element for which to reserve.
94 * \param[in] count Number of values to reserve.
98 MempoolSelelemReserver(const SelectionTreeElementPointer &sel, int count)
102 //! Frees any memory allocated using this reserver.
103 ~MempoolSelelemReserver()
107 sel_->mempoolRelease();
112 * Reserves memory for selection element values using this reserver.
114 * \param[in,out] sel Selection element for which to reserve.
115 * \param[in] count Number of values to reserve.
117 * Allocates space to store \p count output values in \p sel from the
118 * memory pool associated with \p sel, or from the heap if there is no
119 * memory pool. Type of values to allocate is automatically determined
122 void reserve(const SelectionTreeElementPointer &sel, int count)
124 GMX_RELEASE_ASSERT(!sel_,
125 "Can only reserve one element with one instance");
126 sel->mempoolReserve(count);
131 SelectionTreeElementPointer sel_;
135 * Reserves memory for an index group from the evaluation memory pool.
137 * This class implements RAII semantics for allocating memory for an index
138 * group from a selection evaluation memory pool.
140 * \ingroup module_selection
142 class MempoolGroupReserver
146 * Creates a reserver associated with a given memory pool.
148 * \param mp Memory pool from which to reserve memory.
150 explicit MempoolGroupReserver(gmx_sel_mempool_t *mp)
154 //! Frees any memory allocated using this reserver.
155 ~MempoolGroupReserver()
159 _gmx_sel_mempool_free_group(mp_, g_);
164 * Reserves memory for an index group using this reserver.
166 * \param[in,out] g Index group to reserve.
167 * \param[in] count Number of atoms to reserve space for.
169 * Allocates memory from the memory pool to store \p count atoms in
172 void reserve(gmx_ana_index_t *g, int count)
174 GMX_RELEASE_ASSERT(g_ == NULL, "Can only reserve one element with one instance");
175 _gmx_sel_mempool_alloc_group(mp_, g, count);
180 gmx_sel_mempool_t *mp_;
185 * Assigns a temporary value for a selection element.
187 * This class implements RAII semantics for temporarily assigning the value
188 * pointer of a selection element to point to a different location.
190 * \ingroup module_selection
192 class SelelemTemporaryValueAssigner
195 //! Constructs an assigner without an initial assignment.
196 SelelemTemporaryValueAssigner()
197 : old_ptr_(NULL), old_nalloc_(0)
201 * Constructs an assigner with an initial assignment.
203 * \param[in,out] sel Selection element for which to assign.
204 * \param[in] vsource Element to which \p sel values will point to.
208 SelelemTemporaryValueAssigner(const SelectionTreeElementPointer &sel,
209 const SelectionTreeElement &vsource)
211 assign(sel, vsource);
213 //! Undoes any temporary assignment done using this assigner.
214 ~SelelemTemporaryValueAssigner()
218 _gmx_selvalue_setstore_alloc(&sel_->v, old_ptr_, old_nalloc_);
223 * Assigns a temporary value pointer.
225 * \param[in,out] sel Selection element for which to assign.
226 * \param[in] vsource Element to which \p sel values will point to.
228 * Assigns the value pointer in \p sel to point to the values in
229 * \p vsource, i.e., any access/modification to values in \p sel
230 * actually accesses values in \p vsource.
232 void assign(const SelectionTreeElementPointer &sel,
233 const SelectionTreeElement &vsource)
235 GMX_RELEASE_ASSERT(!sel_,
236 "Can only assign one element with one instance");
237 GMX_RELEASE_ASSERT(sel->v.type == vsource.v.type,
238 "Mismatching selection value types");
239 old_ptr_ = sel->v.u.ptr;
240 old_nalloc_ = sel->v.nalloc;
241 _gmx_selvalue_setstore(&sel->v, vsource.v.u.ptr);
246 SelectionTreeElementPointer sel_;
254 * \param[in] fp File handle to receive the output.
255 * \param[in] evalfunc Function pointer to print.
258 _gmx_sel_print_evalfunc_name(FILE *fp, gmx::sel_evalfunc evalfunc)
264 else if (evalfunc == &_gmx_sel_evaluate_root)
268 else if (evalfunc == &_gmx_sel_evaluate_static)
270 fprintf(fp, "static");
272 else if (evalfunc == &_gmx_sel_evaluate_subexpr_simple)
274 fprintf(fp, "subexpr_simple");
276 else if (evalfunc == &_gmx_sel_evaluate_subexpr_staticeval)
278 fprintf(fp, "subexpr_staticeval");
280 else if (evalfunc == &_gmx_sel_evaluate_subexpr)
282 fprintf(fp, "subexpr");
284 else if (evalfunc == &_gmx_sel_evaluate_subexprref_simple)
286 fprintf(fp, "ref_simple");
288 else if (evalfunc == &_gmx_sel_evaluate_subexprref)
292 else if (evalfunc == &_gmx_sel_evaluate_method)
294 fprintf(fp, "method");
296 else if (evalfunc == &_gmx_sel_evaluate_modifier)
300 else if (evalfunc == &_gmx_sel_evaluate_not)
304 else if (evalfunc == &_gmx_sel_evaluate_and)
308 else if (evalfunc == &_gmx_sel_evaluate_or)
312 else if (evalfunc == &_gmx_sel_evaluate_arithmetic)
314 fprintf(fp, "arithmetic");
318 fprintf(fp, "%p", (void*)(evalfunc));
323 * \param[out] data Evaluation data structure to initialize.
324 * \param[in] mp Memory pool for intermediate evaluation values.
325 * \param[in] gall Index group with all the atoms.
326 * \param[in] top Topology structure for evaluation.
327 * \param[in] fr New frame for evaluation.
328 * \param[in] pbc New PBC information for evaluation.
331 _gmx_sel_evaluate_init(gmx_sel_evaluate_t *data,
332 gmx_sel_mempool_t *mp, gmx_ana_index_t *gall,
333 t_topology *top, t_trxframe *fr, t_pbc *pbc)
343 * Recursively initializes the flags for evaluation.
345 * \param[in,out] sel Selection element to clear.
347 * The \ref SEL_INITFRAME flag is set for \ref SEL_EXPRESSION elements whose
348 * method defines the \p init_frame callback (see sel_framefunc()), and
349 * cleared for other elements.
351 * The \ref SEL_EVALFRAME flag is cleared for all elements.
354 init_frame_eval(SelectionTreeElementPointer sel)
358 sel->flags &= ~(SEL_INITFRAME | SEL_EVALFRAME);
359 if (sel->type == SEL_EXPRESSION)
361 if (sel->u.expr.method && sel->u.expr.method->init_frame)
363 sel->flags |= SEL_INITFRAME;
366 if (sel->child && sel->type != SEL_SUBEXPRREF)
368 init_frame_eval(sel->child);
377 SelectionEvaluator::SelectionEvaluator()
382 * \param[in,out] coll The selection collection to evaluate.
383 * \param[in] fr Frame for which the evaluation should be carried out.
384 * \param[in] pbc PBC data, or NULL if no PBC should be used.
385 * \returns 0 on successful evaluation, a non-zero error code on error.
387 * This functions sets the global variables for topology, frame and PBC,
388 * clears some information in the selection to initialize the evaluation
389 * for a new frame, and evaluates \p sel and all the selections pointed by
390 * the \p next pointers of \p sel.
392 * This is the only function that user code should call if they want to
393 * evaluate a selection for a new frame.
396 SelectionEvaluator::evaluate(SelectionCollection *coll,
397 t_trxframe *fr, t_pbc *pbc)
399 gmx_ana_selcollection_t *sc = &coll->impl_->sc_;
400 gmx_sel_evaluate_t data;
402 _gmx_sel_evaluate_init(&data, sc->mempool, &sc->gall, sc->top, fr, pbc);
403 init_frame_eval(sc->root);
404 SelectionTreeElementPointer sel = sc->root;
407 /* Clear the evaluation group of subexpressions */
408 if (sel->child && sel->child->type == SEL_SUBEXPR)
410 sel->child->u.cgrp.isize = 0;
411 /* Not strictly necessary, because the value will be overwritten
412 * during first evaluation of the subexpression anyways, but we
413 * clear the group for clarity. Note that this is _not_ done during
414 * compilation because of some additional complexities involved
415 * (see compiler.c), so it should not be relied upon in
416 * _gmx_sel_evaluate_subexpr(). */
417 if (sel->child->v.type == GROUP_VALUE)
419 sel->child->v.u.g->isize = 0;
424 sel->evaluate(&data, sel, NULL);
428 /* Update selection information */
429 SelectionDataList::const_iterator isel;
430 for (isel = sc->sel.begin(); isel != sc->sel.end(); ++isel)
432 internal::SelectionData &sel = **isel;
433 sel.refreshMassesAndCharges(sc->top);
434 sel.updateCoveredFractionForFrame();
439 * \param[in,out] coll The selection collection to evaluate.
440 * \param[in] nframes Total number of frames.
443 SelectionEvaluator::evaluateFinal(SelectionCollection *coll, int nframes)
445 gmx_ana_selcollection_t *sc = &coll->impl_->sc_;
447 SelectionDataList::const_iterator isel;
448 for (isel = sc->sel.begin(); isel != sc->sel.end(); ++isel)
450 internal::SelectionData &sel = **isel;
451 sel.restoreOriginalPositions(sc->top);
452 sel.computeAverageCoveredFraction(nframes);
459 * \param[in] data Data for the current frame.
460 * \param[in] sel Selection element being evaluated.
461 * \param[in] g Group for which \p sel should be evaluated.
462 * \returns 0 on success, a non-zero error code on error.
464 * Evaluates each child of \p sel in \p g.
467 _gmx_sel_evaluate_children(gmx_sel_evaluate_t *data,
468 const SelectionTreeElementPointer &sel,
471 SelectionTreeElementPointer child = sel->child;
476 child->evaluate(data, child, g);
483 * \param[in] data Data for the current frame.
484 * \param[in] sel Selection element being evaluated.
485 * \param[in] g Group for which \p sel should be evaluated
486 * (not used, can be NULL).
487 * \returns 0 on success, a non-zero error code on error.
489 * Evaluates the first child element in the group defined by \p sel->u.cgrp.
490 * If \p sel->u.cgrp is empty, nothing is done.
491 * The value of \p sel is not touched (root elements do not evaluate to
494 * This function can be used as gmx::SelectionTreeElement::evaluate for
495 * \ref SEL_ROOT elements.
498 _gmx_sel_evaluate_root(gmx_sel_evaluate_t *data,
499 const SelectionTreeElementPointer &sel,
502 if (sel->u.cgrp.isize == 0 || !sel->child->evaluate)
507 sel->child->evaluate(data, sel->child,
508 sel->u.cgrp.isize < 0 ? NULL : &sel->u.cgrp);
512 * \param[in] data Data for the current frame.
513 * \param[in] sel Selection element being evaluated.
514 * \param[in] g Group for which \p sel should be evaluated.
515 * \returns 0 for success.
517 * Sets the value of \p sel to the intersection of \p g and \p sel->u.cgrp.
519 * This function can be used as gmx::SelectionTreeElement::evaluate for
520 * \ref SEL_CONST elements with value type \ref GROUP_VALUE.
523 _gmx_sel_evaluate_static(gmx_sel_evaluate_t *data,
524 const SelectionTreeElementPointer &sel,
527 gmx_ana_index_intersection(sel->v.u.g, &sel->u.cgrp, g);
531 /*********************************************************************
532 * SUBEXPRESSION EVALUATION
533 *********************************************************************/
536 * \param[in] data Data for the current frame.
537 * \param[in] sel Selection element being evaluated.
538 * \param[in] g Group for which \p sel should be evaluated.
539 * \returns 0 on success, a non-zero error code on error.
541 * Evaluates the child element (there should be exactly one) in \p g.
542 * The compiler has taken care that the child actually stores the evaluated
543 * value in the value pointer of this element.
545 * This function is used as gmx::SelectionTreeElement::evaluate for
546 * \ref SEL_SUBEXPR elements that are used only once, and hence do not need
547 * full subexpression handling.
550 _gmx_sel_evaluate_subexpr_simple(gmx_sel_evaluate_t *data,
551 const SelectionTreeElementPointer &sel,
554 if (sel->child->evaluate)
556 sel->child->evaluate(data, sel->child, g);
558 sel->v.nr = sel->child->v.nr;
562 * \param[in] data Data for the current frame.
563 * \param[in] sel Selection element being evaluated.
564 * \param[in] g Group for which \p sel should be evaluated.
565 * \returns 0 on success, a non-zero error code on error.
567 * If this is the first call for this frame, evaluates the child element
568 * there should be exactly one in \p g.
569 * The compiler has taken care that the child actually stores the evaluated
570 * value in the value pointer of this element.
571 * Assumes that \p g is persistent for the duration of the whole evaluation.
573 * This function is used as gmx::SelectionTreeElement::evaluate for
574 * \ref SEL_SUBEXPR elements that have a static evaluation group, and hence do
575 * not need full subexpression handling.
578 _gmx_sel_evaluate_subexpr_staticeval(gmx_sel_evaluate_t *data,
579 const SelectionTreeElementPointer &sel,
582 if (sel->u.cgrp.isize == 0)
584 sel->child->evaluate(data, sel->child, g);
585 sel->v.nr = sel->child->v.nr;
588 sel->u.cgrp.isize = -1;
592 gmx_ana_index_set(&sel->u.cgrp, g->isize, g->index, sel->u.cgrp.name, 0);
598 * \param[in] data Data for the current frame.
599 * \param[in] sel Selection element being evaluated.
600 * \param[in] g Group for which \p sel should be evaluated.
601 * \returns 0 on success, a non-zero error code on error.
603 * Finds the part of \p g for which the subexpression
604 * has not yet been evaluated by comparing \p g to \p sel->u.cgrp.
605 * If the part is not empty, the child expression is evaluated for this
606 * part, and the results merged to the old values of the child.
607 * The value of \p sel itself is undefined after the call.
610 * The call to gmx_ana_index_difference() can take quite a lot of unnecessary
611 * time if the subexpression is evaluated either several times for the same
612 * group or for completely distinct groups.
613 * However, in the majority of cases, these situations occur when
614 * _gmx_sel_evaluate_subexpr_staticeval() can be used, so this should not be a
618 _gmx_sel_evaluate_subexpr(gmx_sel_evaluate_t *data,
619 const SelectionTreeElementPointer &sel,
622 gmx_ana_index_t gmiss;
624 MempoolGroupReserver gmissreserver(data->mp);
625 if (sel->u.cgrp.isize == 0)
628 SelelemTemporaryValueAssigner assigner(sel->child, *sel);
629 sel->child->evaluate(data, sel->child, g);
631 /* We need to keep the name for the cgrp across the copy to avoid
632 * problems if g has a name set. */
633 char *name = sel->u.cgrp.name;
634 gmx_ana_index_copy(&sel->u.cgrp, g, false);
635 sel->u.cgrp.name = name;
640 gmissreserver.reserve(&gmiss, g->isize);
641 gmx_ana_index_difference(&gmiss, g, &sel->u.cgrp);
646 MempoolSelelemReserver reserver(sel->child, gmiss.isize);
647 /* Evaluate the missing values for the child */
648 sel->child->evaluate(data, sel->child, &gmiss);
649 /* Merge the missing values to the existing ones. */
650 if (sel->v.type == GROUP_VALUE)
652 gmx_ana_index_merge(sel->v.u.g, sel->child->v.u.g, sel->v.u.g);
658 i = sel->u.cgrp.isize - 1;
660 /* TODO: This switch is kind of ugly, but it may be difficult to
661 * do this portably without C++ templates. */
665 for (k = sel->u.cgrp.isize + gmiss.isize - 1; k >= 0; k--)
667 if (i < 0 || (j >= 0 && sel->u.cgrp.index[i] < gmiss.index[j]))
669 sel->v.u.i[k] = sel->child->v.u.i[j--];
673 sel->v.u.i[k] = sel->v.u.i[i--];
679 for (k = sel->u.cgrp.isize + gmiss.isize - 1; k >= 0; k--)
681 if (i < 0 || (j >= 0 && sel->u.cgrp.index[i] < gmiss.index[j]))
683 sel->v.u.r[k] = sel->child->v.u.r[j--];
687 sel->v.u.r[k] = sel->v.u.r[i--];
693 // Note: with the currently allowed syntax, this case is never
695 for (k = sel->u.cgrp.isize + gmiss.isize - 1; k >= 0; k--)
697 if (i < 0 || (j >= 0 && sel->u.cgrp.index[i] < gmiss.index[j]))
699 sel->v.u.s[k] = sel->child->v.u.s[j--];
703 sel->v.u.s[k] = sel->v.u.s[i--];
709 /* TODO: Implement this */
710 GMX_THROW(gmx::NotImplementedError("position subexpressions not implemented properly"));
714 GMX_THROW(gmx::InternalError("Invalid subexpression type"));
717 gmx_ana_index_merge(&sel->u.cgrp, &sel->u.cgrp, &gmiss);
722 * \param[in] data Data for the current frame.
723 * \param[in] sel Selection element being evaluated.
724 * \param[in] g Group for which \p sel should be evaluated.
725 * \returns 0 for success.
727 * Sets the value pointers of the child and its child to point to the same
728 * memory as the value pointer of this element to avoid copying, and then
729 * evaluates evaluates the child.
731 * This function is used as gmx::SelectionTreeElement:evaluate for
732 * \ref SEL_SUBEXPRREF elements for which the \ref SEL_SUBEXPR does not have
736 _gmx_sel_evaluate_subexprref_simple(gmx_sel_evaluate_t *data,
737 const SelectionTreeElementPointer &sel,
742 _gmx_selvalue_setstore(&sel->child->v, sel->v.u.ptr);
743 _gmx_selvalue_setstore_alloc(&sel->child->child->v, sel->v.u.ptr,
744 sel->child->child->v.nalloc);
745 sel->child->evaluate(data, sel->child, g);
747 sel->v.nr = sel->child->v.nr;
750 sel->u.param->val.nr = sel->v.nr;
751 if (sel->u.param->nvalptr)
753 *sel->u.param->nvalptr = sel->u.param->val.nr;
759 * \param[in] data Data for the current frame.
760 * \param[in] sel Selection element being evaluated.
761 * \param[in] g Group for which \p sel should be evaluated.
762 * \returns 0 on success, a non-zero error code on error.
764 * If the value type is \ref POS_VALUE, the value of the child is simply
765 * copied to set the value of \p sel (the child subexpression should
766 * already have been evaluated by its root).
767 * If the value type is something else, the child is evaluated for the
768 * group \p g, and the value of the child is then copied.
769 * There should be only one child element.
771 * This function is used as gmx::SelectionTreeElement::evaluate for
772 * \ref SEL_SUBEXPRREF elements.
775 _gmx_sel_evaluate_subexprref(gmx_sel_evaluate_t *data,
776 const SelectionTreeElementPointer &sel,
783 sel->child->evaluate(data, sel->child, g);
785 const SelectionTreeElementPointer &expr = sel->child;
791 sel->v.nr = expr->v.nr;
792 memcpy(sel->v.u.i, expr->v.u.i, sel->v.nr*sizeof(*sel->v.u.i));
796 sel->v.nr = g->isize;
797 /* Extract the values corresponding to g */
798 for (i = j = 0; i < g->isize; ++i, ++j)
800 while (sel->child->u.cgrp.index[j] < g->index[i])
804 sel->v.u.i[i] = expr->v.u.i[j];
812 sel->v.nr = expr->v.nr;
813 memcpy(sel->v.u.r, expr->v.u.r, sel->v.nr*sizeof(*sel->v.u.r));
817 sel->v.nr = g->isize;
818 /* Extract the values corresponding to g */
819 for (i = j = 0; i < g->isize; ++i, ++j)
821 while (sel->child->u.cgrp.index[j] < g->index[i])
825 sel->v.u.r[i] = expr->v.u.r[j];
833 sel->v.nr = expr->v.nr;
834 memcpy(sel->v.u.s, expr->v.u.s, sel->v.nr*sizeof(*sel->v.u.s));
838 sel->v.nr = g->isize;
839 /* Extract the values corresponding to g */
840 for (i = j = 0; i < g->isize; ++i, ++j)
842 while (sel->child->u.cgrp.index[j] < g->index[i])
846 sel->v.u.s[i] = expr->v.u.s[j];
852 /* Currently, there is no need to do anything fancy here,
853 * but some future extensions may need a more flexible
855 gmx_ana_pos_copy(sel->v.u.p, expr->v.u.p, false);
861 gmx_ana_index_copy(sel->v.u.g, expr->v.u.g, false);
865 gmx_ana_index_intersection(sel->v.u.g, expr->v.u.g, g);
869 default: /* should not be reached */
870 GMX_THROW(gmx::InternalError("Invalid subexpression reference type"));
872 /* Store the number of values if needed */
875 sel->u.param->val.nr = sel->v.nr;
876 if (sel->u.param->nvalptr)
878 *sel->u.param->nvalptr = sel->u.param->val.nr;
883 /********************************************************************
884 * METHOD EXPRESSION EVALUATION
885 ********************************************************************/
888 * \param[in] data Data for the current frame.
889 * \param[in] sel Selection element being evaluated.
890 * \param[in] g Group for which \p sel should be evaluated.
891 * \returns 0 on success, a non-zero error code on error.
893 * Evaluates each child of a \ref SEL_EXPRESSION element.
894 * The value of \p sel is not touched.
896 * This function is not used as gmx::SelectionTreeElement::evaluate,
897 * but is used internally.
900 _gmx_sel_evaluate_method_params(gmx_sel_evaluate_t *data,
901 const SelectionTreeElementPointer &sel,
904 SelectionTreeElementPointer child = sel->child;
907 if (child->evaluate && !(child->flags & SEL_EVALFRAME))
909 if (child->flags & SEL_ATOMVAL)
911 child->evaluate(data, child, g);
915 child->flags |= SEL_EVALFRAME;
916 child->evaluate(data, child, NULL);
924 * \param[in] data Data for the current frame.
925 * \param[in] sel Selection element being evaluated.
926 * \param[in] g Group for which \p sel should be evaluated.
927 * \returns 0 on success, a non-zero error code on error.
929 * Evaluates all child selections (using _gmx_sel_evaluate_method_params())
930 * to evaluate any parameter values.
931 * If this is the first time this expression is evaluated for
932 * the frame, sel_framefunc() callback is called if one is provided.
933 * If a reference position calculation has been initialized for this element,
934 * the positions are also updated, and sel_updatefunc_pos() is used to
935 * evaluate the value. Otherwise, sel_updatefunc() is used.
937 * This function is used as gmx::SelectionTreeElement::evaluate for
938 * \ref SEL_EXPRESSION elements.
941 _gmx_sel_evaluate_method(gmx_sel_evaluate_t *data,
942 const SelectionTreeElementPointer &sel,
945 _gmx_sel_evaluate_method_params(data, sel, g);
946 if (sel->flags & SEL_INITFRAME)
948 sel->flags &= ~SEL_INITFRAME;
949 sel->u.expr.method->init_frame(data->top, data->fr, data->pbc,
954 gmx_ana_poscalc_update(sel->u.expr.pc, sel->u.expr.pos, g,
955 data->fr, data->pbc);
956 sel->u.expr.method->pupdate(data->top, data->fr, data->pbc,
957 sel->u.expr.pos, &sel->v,
962 sel->u.expr.method->update(data->top, data->fr, data->pbc, g,
963 &sel->v, sel->u.expr.mdata);
968 * \param[in] data Data for the current frame.
969 * \param[in] sel Selection element being evaluated.
970 * \param[in] g Group for which \p sel should be evaluated.
971 * \returns 0 on success, a non-zero error code on error.
973 * Evaluates all child selections (using _gmx_sel_evaluate_method_params())
974 * to evaluate any parameter values.
975 * If this is the first time this expression is evaluated for
976 * the frame, sel_framefunc() callback is called if one is provided.
977 * The modifier is then evaluated using sel_updatefunc_pos().
979 * This function is used as gmx::SelectionTreeElement::evaluate for
980 * \ref SEL_MODIFIER elements.
983 _gmx_sel_evaluate_modifier(gmx_sel_evaluate_t *data,
984 const SelectionTreeElementPointer &sel,
987 _gmx_sel_evaluate_method_params(data, sel, g);
988 if (sel->flags & SEL_INITFRAME)
990 sel->flags &= ~SEL_INITFRAME;
991 sel->u.expr.method->init_frame(data->top, data->fr, data->pbc,
994 GMX_RELEASE_ASSERT(sel->child != NULL,
995 "Modifier element with a value must have a child");
996 if (sel->child->v.type != POS_VALUE)
998 GMX_THROW(gmx::NotImplementedError("Non-position valued modifiers not implemented"));
1000 sel->u.expr.method->pupdate(data->top, data->fr, data->pbc,
1002 &sel->v, sel->u.expr.mdata);
1006 /********************************************************************
1007 * BOOLEAN EXPRESSION EVALUATION
1008 ********************************************************************/
1011 * \param[in] data Data for the current frame.
1012 * \param[in] sel Selection element being evaluated.
1013 * \param[in] g Group for which \p sel should be evaluated.
1014 * \returns 0 on success, a non-zero error code on error.
1016 * Evaluates the child element (there should be only one) in the group
1017 * \p g, and then sets the value of \p sel to the complement of the
1020 * This function is used as gmx::SelectionTreeElement::evaluate for
1021 * \ref SEL_BOOLEAN elements with \ref BOOL_NOT.
1024 _gmx_sel_evaluate_not(gmx_sel_evaluate_t *data,
1025 const SelectionTreeElementPointer &sel,
1028 MempoolSelelemReserver reserver(sel->child, g->isize);
1029 sel->child->evaluate(data, sel->child, g);
1030 gmx_ana_index_difference(sel->v.u.g, g, sel->child->v.u.g);
1034 * \param[in] data Data for the current frame.
1035 * \param[in] sel Selection element being evaluated.
1036 * \param[in] g Group for which \p sel should be evaluated.
1037 * \returns 0 on success, a non-zero error code on error.
1039 * Short-circuiting evaluation of logical AND expressions.
1041 * Starts by evaluating the first child element in the group \p g.
1042 * The each following child element is evaluated in the intersection
1043 * of all the previous values until all children have been evaluated
1044 * or the intersection becomes empty.
1045 * The value of \p sel is set to the intersection of all the (evaluated)
1048 * If the first child does not have an evaluation function, it is skipped
1049 * and the evaluation is started at the second child.
1050 * This happens if the first child is a constant expression and during
1051 * compilation it was detected that the evaluation group is always a subset
1052 * of the constant group
1053 * (currently, the compiler never detects this).
1055 * This function is used as gmx::SelectionTreeElement::evaluate for
1056 * \ref SEL_BOOLEAN elements with \ref BOOL_AND.
1059 _gmx_sel_evaluate_and(gmx_sel_evaluate_t *data,
1060 const SelectionTreeElementPointer &sel,
1063 SelectionTreeElementPointer child = sel->child;
1064 /* Skip the first child if it does not have an evaluation function. */
1065 if (!child->evaluate)
1067 child = child->next;
1069 /* Evaluate the first child */
1071 MempoolSelelemReserver reserver(child, g->isize);
1072 child->evaluate(data, child, g);
1073 gmx_ana_index_copy(sel->v.u.g, child->v.u.g, false);
1075 child = child->next;
1076 while (child && sel->v.u.g->isize > 0)
1078 MempoolSelelemReserver reserver(child, sel->v.u.g->isize);
1079 child->evaluate(data, child, sel->v.u.g);
1080 gmx_ana_index_intersection(sel->v.u.g, sel->v.u.g, child->v.u.g);
1081 child = child->next;
1086 * \param[in] data Data for the current frame.
1087 * \param[in] sel Selection element being evaluated.
1088 * \param[in] g Group for which \p sel should be evaluated.
1089 * \returns 0 on success, a non-zero error code on error.
1091 * Short-circuiting evaluation of logical OR expressions.
1093 * Starts by evaluating the first child element in the group \p g.
1094 * For each subsequent child, finds the part of \p g that is not
1095 * included the value of any previous child, and evaluates the child
1096 * in that group until the last child is evaluated or all of \p g
1097 * is included in some child value.
1098 * The value of \p sel is set to the union of all the (evaluated)
1101 * If the first child does not have an evaluation function, its value is
1102 * used without evaluation.
1103 * This happens if the first child is a constant expression, the selection
1104 * has been compiled, and the evaluation group is the same for each frame.
1105 * In this case, the compiler has taken care of that the child value is a
1106 * subset of \p g, making it unnecessary to evaluate it.
1108 * This function is used as gmx::SelectionTreeElement::evaluate for
1109 * \ref SEL_BOOLEAN elements with \ref BOOL_OR.
1112 _gmx_sel_evaluate_or(gmx_sel_evaluate_t *data,
1113 const SelectionTreeElementPointer &sel,
1116 gmx_ana_index_t tmp, tmp2;
1118 SelectionTreeElementPointer child = sel->child;
1119 if (child->evaluate)
1121 MempoolSelelemReserver reserver(child, g->isize);
1122 child->evaluate(data, child, g);
1123 gmx_ana_index_partition(sel->v.u.g, &tmp, g, child->v.u.g);
1127 gmx_ana_index_partition(sel->v.u.g, &tmp, g, child->v.u.g);
1129 child = child->next;
1130 while (child && tmp.isize > 0)
1134 MempoolSelelemReserver reserver(child, tmp.isize);
1135 child->evaluate(data, child, &tmp);
1136 gmx_ana_index_partition(&tmp, &tmp2, &tmp, child->v.u.g);
1138 sel->v.u.g->isize += tmp.isize;
1139 tmp.isize = tmp2.isize;
1140 tmp.index = tmp2.index;
1141 child = child->next;
1143 gmx_ana_index_sort(sel->v.u.g);
1147 /********************************************************************
1148 * ARITHMETIC EVALUATION
1149 ********************************************************************/
1152 * \param[in] data Data for the current frame.
1153 * \param[in] sel Selection element being evaluated.
1154 * \param[in] g Group for which \p sel should be evaluated.
1155 * \returns 0 on success, a non-zero error code on error.
1158 _gmx_sel_evaluate_arithmetic(gmx_sel_evaluate_t *data,
1159 const SelectionTreeElementPointer &sel,
1163 real lval, rval = 0., val = 0.;
1165 const SelectionTreeElementPointer &left = sel->child;
1166 const SelectionTreeElementPointer &right = left->next;
1168 SelelemTemporaryValueAssigner assigner;
1169 MempoolSelelemReserver reserver;
1172 assigner.assign(left, *sel);
1175 reserver.reserve(right, g->isize);
1178 else if (right && right->mempool)
1180 assigner.assign(right, *sel);
1182 _gmx_sel_evaluate_children(data, sel, g);
1184 n = (sel->flags & SEL_SINGLEVAL) ? 1 : g->isize;
1187 bool bArithNeg = (sel->u.arith.type == ARITH_NEG);
1188 GMX_ASSERT(right || bArithNeg,
1189 "Right operand cannot be null except for negations");
1190 for (i = i1 = i2 = 0; i < n; ++i)
1192 lval = left->v.u.r[i1];
1195 rval = right->v.u.r[i2];
1197 switch (sel->u.arith.type)
1199 case ARITH_PLUS: val = lval + rval; break;
1200 case ARITH_MINUS: val = lval - rval; break;
1201 case ARITH_NEG: val = -lval; break;
1202 case ARITH_MULT: val = lval * rval; break;
1203 case ARITH_DIV: val = lval / rval; break;
1204 case ARITH_EXP: val = pow(lval, rval); break;
1206 sel->v.u.r[i] = val;
1207 if (!(left->flags & SEL_SINGLEVAL))
1211 if (!bArithNeg && !(right->flags & SEL_SINGLEVAL))