source: src/Filling/Filler.cpp@ df855a

ForceAnnealing_goodresults ForceAnnealing_tocheck
Last change on this file since df855a was 9eb71b3, checked in by Frederik Heber <frederik.heber@…>, 8 years ago

Commented out MemDebug include and Memory::ignore.

  • MemDebug clashes with various allocation operators that use a specific placement in memory. It is so far not possible to wrap new/delete fully. Hence, we stop this effort which so far has forced us to put ever more includes (with clashes) into MemDebug and thereby bloat compilation time.
  • MemDebug does not add that much usefulness which is not also provided by valgrind.
  • Property mode set to 100644
File size: 5.7 KB
Line 
1/*
2 * Project: MoleCuilder
3 * Description: creates and alters molecular systems
4 * Copyright (C) 2012 University of Bonn. All rights reserved.
5 *
6 *
7 * This file is part of MoleCuilder.
8 *
9 * MoleCuilder is free software: you can redistribute it and/or modify
10 * it under the terms of the GNU General Public License as published by
11 * the Free Software Foundation, either version 2 of the License, or
12 * (at your option) any later version.
13 *
14 * MoleCuilder 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
17 * GNU General Public License for more details.
18 *
19 * You should have received a copy of the GNU General Public License
20 * along with MoleCuilder. If not, see <http://www.gnu.org/licenses/>.
21 */
22
23/*
24 * Filler.cpp
25 *
26 * Created on: Jan 16, 2012
27 * Author: heber
28 */
29
30
31// include config.h
32#ifdef HAVE_CONFIG_H
33#include <config.h>
34#endif
35
36//#include "CodePatterns/MemDebug.hpp"
37
38#include <algorithm>
39#include <boost/bind.hpp>
40#include <boost/lambda/lambda.hpp>
41#include <sstream>
42#include <vector>
43
44#include "Filler.hpp"
45
46#include "CodePatterns/Assert.hpp"
47#include "CodePatterns/Log.hpp"
48
49#include "Atom/atom.hpp"
50#include "Box.hpp"
51#include "ClusterInterface.hpp"
52#include "Descriptors/AtomIdDescriptor.hpp"
53#include "Inserter/Inserter.hpp"
54#include "LinearAlgebra/RealSpaceMatrix.hpp"
55#include "LinearAlgebra/Vector.hpp"
56#include "molecule.hpp"
57#include "NodeTypes.hpp"
58#include "Predicates/FillPredicate.hpp"
59#include "Predicates/Ops_FillPredicate.hpp"
60#include "World.hpp"
61
62
63Filler::Filler(const Mesh &_mesh, const FillPredicate &_predicate, const Inserter &_inserter) :
64 mesh(_mesh),
65 predicate(!_predicate),
66 inserter(_inserter)
67{}
68
69Filler::~Filler()
70{}
71
72bool Filler::operator()(
73 CopyAtomsInterface &copyMethod,
74 ClusterInterface::Cluster_impl cluster,
75 ClusterVector_t &ClonedClusters) const
76{
77 const NodeSet &nodes = mesh.getNodes();
78 std::stringstream output;
79 std::for_each( nodes.begin(), nodes.end(), output << boost::lambda::_1 << " ");
80 LOG(3, "DEBUG: Listing nodes to check: " << output.str());
81 if (nodes.size() == 0)
82 return false;
83 NodeSet FillNodes(nodes.size(), zeroVec);
84
85 // evaluate predicates at each FillNode
86 {
87 // move filler cluster's atoms out of domain such that it does not disturb the predicate.
88 // we only move the atoms as otherwise two translate ShapeOps will be on top of the Shape
89 // which is subsequently copied to all other cloned Clusters ...
90 Vector BoxDiagonal;
91 {
92 const RealSpaceMatrix &M = World::getInstance().getDomain().getM();
93 BoxDiagonal = (M * Vector(1.,1.,1.));
94 BoxDiagonal -= cluster->getShape().getCenter();
95 BoxDiagonal *= 1. + 2.*cluster->getShape().getRadius()/BoxDiagonal.Norm(); // extend it a little further
96 AtomIdSet atoms = cluster->getAtoms();
97 for (AtomIdSet::iterator iter = atoms.begin(); iter != atoms.end(); ++iter)
98 (*iter)->setPosition( (*iter)->getPosition() + BoxDiagonal );
99 LOG(1, "INFO: Translating original cluster's atoms by " << BoxDiagonal << ".");
100 }
101
102 // evaluate predicate and gather into new set
103 NodeSet::iterator transform_end =
104 std::remove_copy_if(nodes.begin(), nodes.end(), FillNodes.begin(), predicate );
105 FillNodes.erase(transform_end, FillNodes.end());
106
107 // shift cluster back to original place
108 {
109 AtomIdSet atoms = cluster->getAtoms();
110 for (AtomIdSet::iterator iter = atoms.begin(); iter != atoms.end(); ++iter)
111 (*iter)->setPosition( (*iter)->getPosition() - BoxDiagonal );
112 LOG(1, "INFO: Translating original cluster's atoms back.");
113 }
114 }
115
116 if (FillNodes.size() == 0) {
117 ELOG(2, "For none of the nodes did the predicate return true.");
118 return false;
119 } else {
120 LOG(1, "INFO: " << FillNodes.size() << " out of " << nodes.size() << " returned true from predicate.");
121 }
122
123 // clone clusters
124 ClonedClusters.resize(FillNodes.size());
125 {
126 std::vector<ClusterInterface::Cluster_impl>::iterator clusteriter = ClonedClusters.begin();
127 *clusteriter = cluster;
128 clusteriter++;
129 std::generate_n(clusteriter, FillNodes.size()-1,
130 boost::bind(&ClusterInterface::clone, boost::cref(cluster), boost::ref(copyMethod), zeroVec) );
131 }
132
133 // insert each cluster by abusing std::search a bit:
134 {
135 // we look for the subsequence of FillNodes inside clusters. If Inserter always
136 // returns true, we'll have the iterator pointing at first cluster
137 std::vector<ClusterInterface::Cluster_impl>::const_iterator inserteriter =
138 std::search(ClonedClusters.begin(), ClonedClusters.end(), FillNodes.begin(), FillNodes.end(),
139 boost::bind(&Inserter::operator(), boost::cref(inserter), _1, _2));
140 if( inserteriter != ClonedClusters.begin()) {
141 ELOG(1, "Not all cloned clusters could be successfully inserted.");
142 return false;
143 }
144 }
145
146 // create molecules for each cluster and fill in atoms
147 {
148 std::vector<molecule *> molecules(ClonedClusters.size()-1, NULL);
149 std::generate_n(molecules.begin(), FillNodes.size()-1,
150 boost::bind(&World::createMolecule, World::getPointer()) );
151 std::vector<ClusterInterface::Cluster_impl>::const_iterator clusteriter = ClonedClusters.begin();
152 ++clusteriter;
153 std::vector<molecule *>::iterator moliter = molecules.begin();
154 for (;moliter != molecules.end(); ++clusteriter, ++moliter) {
155 AtomIdSet atoms = (*clusteriter)->getAtoms();
156 for (AtomIdSet::iterator iter = atoms.begin(); iter != atoms.end(); ++iter)
157 (*moliter)->AddAtom(*iter);
158 }
159 }
160
161 // give final statment on whether at least \a single cluster has been placed
162 return ( FillNodes.size() != 0);
163}
164
Note: See TracBrowser for help on using the repository browser.