source: src/Fragmentation/Fragmentation.cpp@ 3d9a8d

Action_Thermostats Add_AtomRandomPerturbation Add_FitFragmentPartialChargesAction Add_RotateAroundBondAction Add_SelectAtomByNameAction Added_ParseSaveFragmentResults AddingActions_SaveParseParticleParameters Adding_Graph_to_ChangeBondActions Adding_MD_integration_tests Adding_ParticleName_to_Atom Adding_StructOpt_integration_tests AtomFragments Automaking_mpqc_open AutomationFragmentation_failures Candidate_v1.5.4 Candidate_v1.6.0 Candidate_v1.6.1 Candidate_v1.7.0 ChangeBugEmailaddress ChangingTestPorts ChemicalSpaceEvaluator CombiningParticlePotentialParsing Combining_Subpackages Debian_Package_split Debian_package_split_molecuildergui_only Disabling_MemDebug Docu_Python_wait EmpiricalPotential_contain_HomologyGraph EmpiricalPotential_contain_HomologyGraph_documentation Enable_parallel_make_install Enhance_userguide Enhanced_StructuralOptimization Enhanced_StructuralOptimization_continued Example_ManyWaysToTranslateAtom Exclude_Hydrogens_annealWithBondGraph FitPartialCharges_GlobalError Fix_BoundInBox_CenterInBox_MoleculeActions Fix_ChargeSampling_PBC Fix_ChronosMutex Fix_FitPartialCharges Fix_FitPotential_needs_atomicnumbers Fix_ForceAnnealing Fix_IndependentFragmentGrids Fix_ParseParticles Fix_ParseParticles_split_forward_backward_Actions Fix_PopActions Fix_QtFragmentList_sorted_selection Fix_Restrictedkeyset_FragmentMolecule Fix_StatusMsg Fix_StepWorldTime_single_argument Fix_Verbose_Codepatterns Fix_fitting_potentials Fixes ForceAnnealing_goodresults ForceAnnealing_oldresults ForceAnnealing_tocheck ForceAnnealing_with_BondGraph ForceAnnealing_with_BondGraph_continued ForceAnnealing_with_BondGraph_continued_betteresults ForceAnnealing_with_BondGraph_contraction-expansion FragmentAction_writes_AtomFragments FragmentMolecule_checks_bonddegrees GeometryObjects Gui_Fixes Gui_displays_atomic_force_velocity ImplicitCharges IndependentFragmentGrids IndependentFragmentGrids_IndividualZeroInstances IndependentFragmentGrids_IntegrationTest IndependentFragmentGrids_Sole_NN_Calculation JobMarket_RobustOnKillsSegFaults JobMarket_StableWorkerPool JobMarket_unresolvable_hostname_fix MoreRobust_FragmentAutomation ODR_violation_mpqc_open PartialCharges_OrthogonalSummation PdbParser_setsAtomName PythonUI_with_named_parameters QtGui_reactivate_TimeChanged_changes Recreated_GuiChecks Rewrite_FitPartialCharges RotateToPrincipalAxisSystem_UndoRedo SaturateAtoms_findBestMatching SaturateAtoms_singleDegree StoppableMakroAction Subpackage_CodePatterns Subpackage_JobMarket Subpackage_LinearAlgebra Subpackage_levmar Subpackage_mpqc_open Subpackage_vmg Switchable_LogView ThirdParty_MPQC_rebuilt_buildsystem TrajectoryDependenant_MaxOrder TremoloParser_IncreasedPrecision TremoloParser_MultipleTimesteps TremoloParser_setsAtomName Ubuntu_1604_changes stable
Last change on this file since 3d9a8d was bfbd4a, checked in by Frederik Heber <heber@…>, 13 years ago

Storing KeySet file with global ids now.

  • We now parse in key set file in FragmentationAction::performCall().
  • new function Graph::getLocalGraph() which sorts out all keysets local to a given molecule.
  • this local graph is passed on to Fragmentation::FragmentMolecule().
  • Fragmentation::AssignKeySetsToFragment() now has const Graph ref as param.
  • Property mode set to 100644
File size: 23.2 KB
RevLine 
[bcf653]1/*
2 * Project: MoleCuilder
3 * Description: creates and alters molecular systems
[0aa122]4 * Copyright (C) 2010-2012 University of Bonn. All rights reserved.
[94d5ac6]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/>.
[bcf653]21 */
22
[cee0b57]23/*
[246e13]24 * Fragmentation.cpp
[cee0b57]25 *
[246e13]26 * Created on: Oct 18, 2011
[cee0b57]27 * Author: heber
28 */
29
[bf3817]30#ifdef HAVE_CONFIG_H
31#include <config.h>
32#endif
33
[ad011c]34#include "CodePatterns/MemDebug.hpp"
[112b09]35
[246e13]36#include "Fragmentation.hpp"
37
38#include "CodePatterns/Assert.hpp"
[47d041]39#include "CodePatterns/Info.hpp"
[246e13]40#include "CodePatterns/Log.hpp"
[49e1ae]41
[6f0841]42#include "Atom/atom.hpp"
[129204]43#include "Bond/bond.hpp"
[8e1f901]44#include "Descriptors/MoleculeDescriptor.hpp"
[3bdb6d]45#include "Element/element.hpp"
[ba1823]46#include "Element/periodentafel.hpp"
[730d7a]47#include "Fragmentation/AdaptivityMap.hpp"
[f96874]48#include "Fragmentation/AtomMask.hpp"
[ba1823]49#include "Fragmentation/fragmentation_helpers.hpp"
[dadc74]50#include "Fragmentation/Graph.hpp"
[06f41f3]51#include "Fragmentation/helpers.hpp"
[f0674a]52#include "Fragmentation/KeySet.hpp"
[f67817f]53#include "Fragmentation/PowerSetGenerator.hpp"
[a03d25]54#include "Fragmentation/UniqueFragments.hpp"
[129204]55#include "Graph/BondGraph.hpp"
[0fad93]56#include "Graph/AdjacencyList.hpp"
[6d551c]57#include "Graph/ListOfLocalAtoms.hpp"
[cee0b57]58#include "molecule.hpp"
[b34306]59#include "World.hpp"
[cee0b57]60
61
[246e13]62/** Constructor of class Fragmentation.
63 *
[07a47e]64 * \param _mol molecule for internal use (looking up atoms)
[9511c7]65 * \param _FileChecker instance contains adjacency parsed from elsewhere
[07a47e]66 * \param _saturation whether to treat hydrogen special and saturate dangling bonds or not
[cee0b57]67 */
[0fad93]68Fragmentation::Fragmentation(molecule *_mol, AdjacencyList &_FileChecker, const enum HydrogenSaturation _saturation) :
[07a47e]69 mol(_mol),
[9511c7]70 saturation(_saturation),
71 FileChecker(_FileChecker)
[246e13]72{}
[9879f6]73
[246e13]74/** Destructor of class Fragmentation.
75 *
[9879f6]76 */
[246e13]77Fragmentation::~Fragmentation()
78{}
[9879f6]79
[13a953]80
[cee0b57]81/** Performs a many-body bond order analysis for a given bond order.
82 * -# parses adjacency, keysets and orderatsite files
83 * -# RootStack is created for every subgraph (here, later we implement the "update 10 sites with highest energ
84y contribution", and that's why this consciously not done in the following loop)
85 * -# in a loop over all subgraphs
86 * -# calls FragmentBOSSANOVA with this RootStack and within the subgraph molecule structure
87 * -# creates molecule (fragment)s from the returned keysets (StoreFragmentFromKeySet)
88 * -# combines the generated molecule lists from all subgraphs
89 * -# saves to disk: fragment configs, adjacency, orderatsite, keyset files
90 * Note that as we split "this" molecule up into a list of subgraphs, i.e. a MoleculeListClass, we have two sets
91 * of vertex indices: Global always means the index in "this" molecule, whereas local refers to the molecule or
92 * subgraph in the MoleculeListClass.
[91207d]93 * \param atomids atomic ids (local to Fragmentation::mol) to fragment, used in AtomMask
[cee0b57]94 * \param Order up to how many neighbouring bonds a fragment contains in BondOrderScheme::BottumUp scheme
[99b0dc]95 * \param prefix prefix string for every fragment file name (may include path)
[49c059]96 * \param &DFS contains bond structure analysis data
[cee0b57]97 * \return 1 - continue, 2 - stop (no fragmentation occured)
98 */
[e71325]99int Fragmentation::FragmentMolecule(
100 const std::vector<atomId_t> &atomids,
101 int Order,
102 const std::string &prefix,
[bfbd4a]103 DepthFirstSearchAnalysis &DFS,
104 const Graph &ParsedFragmentList)
[cee0b57]105{
[339910]106 std::fstream File;
[cee0b57]107 bool CheckOrder = false;
108 int TotalNumberOfKeySets = 0;
109
[339910]110 LOG(0, std::endl);
[07a47e]111 switch (saturation) {
112 case DoSaturate:
[47d041]113 LOG(0, "I will treat hydrogen special and saturate dangling bonds with it.");
[07a47e]114 break;
115 case DontSaturate:
[47d041]116 LOG(0, "Hydrogen is treated just like the rest of the lot.");
[07a47e]117 break;
118 default:
119 ASSERT(0, "Fragmentation::FragmentMolecule() - there is a saturation setting which I have no idea about.");
120 break;
121 }
[cee0b57]122
123 // ++++++++++++++++++++++++++++ INITIAL STUFF: Bond structure analysis, file parsing, ... ++++++++++++++++++++++++++++++++++++++++++
[3aa8a5]124 bool FragmentationToDo = true;
[cee0b57]125
126 // ===== 1. Check whether bond structure is same as stored in files ====
127
128 // === compare it with adjacency file ===
[13a953]129 {
[06f41f3]130 const std::vector<atomId_t> globalids = getGlobalIdsFromLocalIds(*mol, atomids);
[3aa8a5]131 AdjacencyList WorldAdjacency(globalids);
132 FragmentationToDo = FragmentationToDo && (FileChecker > WorldAdjacency);
[13a953]133 }
[cee0b57]134
[91207d]135 // ===== 2. create AtomMask that takes Saturation condition into account
136 AtomMask_t AtomMask(atomids);
137 for (molecule::const_iterator iter = mol->begin(); iter != mol->end(); ++iter) {
138 // remove in hydrogen and we do saturate
139 if ((saturation == DoSaturate) && ((*iter)->getType()->getAtomicNumber() == 1)) // skip hydrogen
140 AtomMask.setFalse((*iter)->getNr());
141 }
142
[cee0b57]143 // ===== 4. check globally whether there's something to do actually (first adaptivity check)
[2a0eb0]144 FragmentationToDo = FragmentationToDo && ParseOrderAtSiteFromFile(atomids, prefix);
[cee0b57]145
146 // =================================== Begin of FRAGMENTATION ===============================
147 // ===== 6a. assign each keyset to its respective subgraph =====
[339910]148 ListOfLocalAtoms_t ListOfLocalAtoms;
149 Graph FragmentList;
150 AssignKeySetsToFragment(ParsedFragmentList, ListOfLocalAtoms, FragmentList, true);
[cee0b57]151
152 // ===== 6b. prepare and go into the adaptive (Order<0), single-step (Order==0) or incremental (Order>0) cycle
[339910]153 KeyStack RootStack;
[cee0b57]154 FragmentationToDo = false; // if CheckOrderAtSite just ones recommends fragmentation, we will save fragments afterwards
[f96874]155 bool LoopDoneAlready = false;
[339910]156 while ((CheckOrder = CheckOrderAtSite(AtomMask, ParsedFragmentList, Order, prefix, LoopDoneAlready))) {
[cee0b57]157 FragmentationToDo = FragmentationToDo || CheckOrder;
[f96874]158 LoopDoneAlready = true; // last plus one entry is used as marker that we have been through this loop once already in CheckOrderAtSite()
[cee0b57]159 // ===== 6b. fill RootStack for each subgraph (second adaptivity check) =====
[339910]160 FillRootStackForSubgraphs(RootStack, AtomMask);
161
162 // call BOSSANOVA method
163 FragmentBOSSANOVA(mol, FragmentList, RootStack);
[cee0b57]164 }
[47d041]165 LOG(2, "CheckOrder is " << CheckOrder << ".");
[cee0b57]166
167 // ==================================== End of FRAGMENTATION ============================================
168
169 // ===== 8a. translate list into global numbers (i.e. ones that are valid in "this" molecule, not in MolecularWalker->Leaf)
[339910]170 TranslateIndicesToGlobalIDs(FragmentList, TotalNumberOfKeySets, TotalGraph);
[00b59d5]171
[ca8bea]172 LOG(1, "STATUS: We have created " << TotalGraph.size() << " fragments.");
[cee0b57]173
[b4f72c]174
[ca8bea]175 // store adaptive orders into file
176 StoreOrderAtSiteFile(prefix);
[cee0b57]177
[47d041]178 LOG(0, "End of bond fragmentation.");
[cee0b57]179 return ((int)(!FragmentationToDo)+1); // 1 - continue, 2 - stop (no fragmentation occured)
180};
181
182
[246e13]183/** Performs BOSSANOVA decomposition at selected sites, increasing the cutoff by one at these sites.
184 * -# constructs a complete keyset of the molecule
185 * -# In a loop over all possible roots from the given rootstack
186 * -# increases order of root site
187 * -# calls PowerSetGenerator with this order, the complete keyset and the rootkeynr
188 * -# for all consecutive lower levels PowerSetGenerator is called with the suborder, the higher order keyset
189as the restricted one and each site in the set as the root)
190 * -# these are merged into a fragment list of keysets
191 * -# All fragment lists (for all orders, i.e. from all destination fields) are merged into one list for return
192 * Important only is that we create all fragments, it is not important if we create them more than once
193 * as these copies are filtered out via use of the hash table (KeySet).
194 * \param *out output stream for debugging
195 * \param Fragment&*List list of already present keystacks (adaptive scheme) or empty list
196 * \param &RootStack stack with all root candidates (unequal to each atom in complete molecule if adaptive scheme is applied)
197 * \return pointer to Graph list
[cee0b57]198 */
[339910]199void Fragmentation::FragmentBOSSANOVA(molecule *mol, Graph &FragmentList, KeyStack &RootStack)
[cee0b57]200{
[339910]201 Info FunctionInfo(__func__);
[246e13]202 Graph ***FragmentLowerOrdersList = NULL;
203 int NumLevels = 0;
204 int NumMolecules = 0;
205 int TotalNumMolecules = 0;
206 int *NumMoleculesOfOrder = NULL;
207 int Order = 0;
208 int UpgradeCount = RootStack.size();
209 KeyStack FragmentRootStack;
210 int RootKeyNr = 0;
211 int RootNr = 0;
212 UniqueFragments FragmentSearch;
[cee0b57]213
[246e13]214 // FragmentLowerOrdersList is a 2D-array of pointer to MoleculeListClass objects, one dimension represents the ANOVA expansion of a single order (i.e. 5)
215 // with all needed lower orders that are subtracted, the other dimension is the BondOrder (i.e. from 1 to 5)
216 NumMoleculesOfOrder = new int[UpgradeCount];
217 FragmentLowerOrdersList = new Graph**[UpgradeCount];
[cee0b57]218
[246e13]219 for(int i=0;i<UpgradeCount;i++) {
220 NumMoleculesOfOrder[i] = 0;
221 FragmentLowerOrdersList[i] = NULL;
[920c70]222 }
223
[246e13]224 FragmentSearch.Init(mol->FindAtom(RootKeyNr));
[5034e1]225
[246e13]226 // Construct the complete KeySet which we need for topmost level only (but for all Roots)
227 KeySet CompleteMolecule;
228 for (molecule::const_iterator iter = mol->begin(); iter != mol->end(); ++iter) {
229 CompleteMolecule.insert((*iter)->GetTrueFather()->getNr());
[cee0b57]230 }
231
[246e13]232 // this can easily be seen: if Order is 5, then the number of levels for each lower order is the total sum of the number of levels above, as
233 // each has to be split up. E.g. for the second level we have one from 5th, one from 4th, two from 3th (which in turn is one from 5th, one from 4th),
234 // hence we have overall four 2th order levels for splitting. This also allows for putting all into a single array (FragmentLowerOrdersList[])
235 // with the order along the cells as this: 5433222211111111 for BondOrder 5 needing 16=pow(2,5-1) cells (only we use bit-shifting which is faster)
236 RootNr = 0; // counts through the roots in RootStack
237 while ((RootNr < UpgradeCount) && (!RootStack.empty())) {
238 RootKeyNr = RootStack.front();
239 RootStack.pop_front();
240 atom *Walker = mol->FindAtom(RootKeyNr);
241 // check cyclic lengths
242 //if ((MinimumRingSize[Walker->GetTrueFather()->getNr()] != -1) && (Walker->GetTrueFather()->AdaptiveOrder+1 > MinimumRingSize[Walker->GetTrueFather()->getNr()])) {
[47d041]243 // LOG(0, "Bond order " << Walker->GetTrueFather()->AdaptiveOrder << " of Root " << *Walker << " greater than or equal to Minimum Ring size of " << MinimumRingSize << " found is not allowed.");
[246e13]244 //} else
245 {
246 // increase adaptive order by one
247 Walker->GetTrueFather()->AdaptiveOrder++;
248 Order = Walker->AdaptiveOrder = Walker->GetTrueFather()->AdaptiveOrder;
[cee0b57]249
[246e13]250 // initialise Order-dependent entries of UniqueFragments structure
251 class PowerSetGenerator PSG(&FragmentSearch, Walker->AdaptiveOrder);
[cee0b57]252
[246e13]253 // allocate memory for all lower level orders in this 1D-array of ptrs
254 NumLevels = 1 << (Order-1); // (int)pow(2,Order);
255 FragmentLowerOrdersList[RootNr] = new Graph*[NumLevels];
256 for (int i=0;i<NumLevels;i++)
257 FragmentLowerOrdersList[RootNr][i] = NULL;
258
259 // create top order where nothing is reduced
[47d041]260 LOG(0, "==============================================================================================================");
261 LOG(0, "Creating KeySets of Bond Order " << Order << " for " << *Walker << ", " << (RootStack.size()-RootNr) << " Roots remaining."); // , NumLevels is " << NumLevels << "
[246e13]262
263 // Create list of Graphs of current Bond Order (i.e. F_{ij})
264 FragmentLowerOrdersList[RootNr][0] = new Graph;
265 FragmentSearch.PrepareForPowersetGeneration(1., FragmentLowerOrdersList[RootNr][0], Walker);
[07a47e]266 NumMoleculesOfOrder[RootNr] = PSG(CompleteMolecule, saturation);
[246e13]267
268 // output resulting number
[47d041]269 LOG(1, "Number of resulting KeySets is: " << NumMoleculesOfOrder[RootNr] << ".");
[246e13]270 if (NumMoleculesOfOrder[RootNr] != 0) {
271 NumMolecules = 0;
272 } else {
273 Walker->GetTrueFather()->MaxOrder = true;
274 }
275 // now, we have completely filled each cell of FragmentLowerOrdersList[] for the current Walker->AdaptiveOrder
276 //NumMoleculesOfOrder[Walker->AdaptiveOrder-1] = NumMolecules;
277 TotalNumMolecules += NumMoleculesOfOrder[RootNr];
[47d041]278// LOG(1, "Number of resulting molecules for Order " << (int)Walker->GetTrueFather()->AdaptiveOrder << " is: " << NumMoleculesOfOrder[RootNr] << ".");
[246e13]279 RootStack.push_back(RootKeyNr); // put back on stack
280 RootNr++;
281 }
282 }
[47d041]283 LOG(0, "==============================================================================================================");
284 LOG(1, "Total number of resulting molecules is: " << TotalNumMolecules << ".");
285 LOG(0, "==============================================================================================================");
[246e13]286
287 // cleanup FragmentSearch structure
288 FragmentSearch.Cleanup();
289
290 // now, FragmentLowerOrdersList is complete, it looks - for BondOrder 5 - as this (number is the ANOVA Order of the terms therein)
291 // 5433222211111111
292 // 43221111
293 // 3211
294 // 21
295 // 1
296
297 // Subsequently, we combine all into a single list (FragmentList)
298 CombineAllOrderListIntoOne(FragmentList, FragmentLowerOrdersList, RootStack, mol);
299 FreeAllOrdersList(FragmentLowerOrdersList, RootStack, mol);
300 delete[](NumMoleculesOfOrder);
301};
302
303/** Estimates by educated guessing (using upper limit) the expected number of fragments.
304 * The upper limit is
305 * \f[
306 * n = N \cdot C^k
307 * \f]
308 * where \f$C=2^c\f$ and c is the maximum bond degree over N number of atoms.
309 * \param *out output stream for debugging
310 * \param order bond order k
311 * \return number n of fragments
312 */
313int Fragmentation::GuesstimateFragmentCount(int order)
314{
315 size_t c = 0;
316 int FragmentCount;
317 // get maximum bond degree
318 for (molecule::const_iterator iter = mol->begin(); iter != mol->end(); ++iter) {
319 const BondList& ListOfBonds = (*iter)->getListOfBonds();
320 c = (ListOfBonds.size() > c) ? ListOfBonds.size() : c;
321 }
[34e74fd]322 FragmentCount = (saturation == DoSaturate ? mol->getNoNonHydrogen() : mol->getAtomCount()) *(1 << (c*order));
[e791dc]323 LOG(1, "Upper limit for this subgraph is " << FragmentCount << " for "
324 << mol->getNoNonHydrogen() << " non-H atoms with maximum bond degree of " << c << ".");
[246e13]325 return FragmentCount;
326};
327
328
329/** Checks whether the OrderAtSite is still below \a Order at some site.
[f96874]330 * \param AtomMask defines true/false per global Atom::Nr to mask in/out each nuclear site, used to activate given number of site to increment order adaptively
[246e13]331 * \param *GlobalKeySetList list of keysets with global ids (valid in "this" molecule) needed for adaptive increase
332 * \param Order desired Order if positive, desired exponent in threshold criteria if negative (0 is single-step)
333 * \param path path to ENERGYPERFRAGMENT file (may be NULL if Order is non-negative)
[f96874]334 * \param LoopDoneAlready indicate whether we have done a fragmentation loop already
[246e13]335 * \return true - needs further fragmentation, false - does not need fragmentation
336 */
[e71325]337bool Fragmentation::CheckOrderAtSite(AtomMask_t &AtomMask, const Graph &GlobalKeySetList, int Order, const std::string &path, bool LoopDoneAlready)
[246e13]338{
339 bool status = false;
340
341 if (Order < 0) { // adaptive increase of BondOrder per site
[f96874]342 if (LoopDoneAlready) // break after one step
[246e13]343 return false;
344
345 // transmorph graph keyset list into indexed KeySetList
[339910]346 AdaptivityMap * IndexKeySetList = GlobalKeySetList.GraphToAdaptivityMap();
[246e13]347
348 // parse the EnergyPerFragment file
[851be8]349 IndexKeySetList->ScanAdaptiveFileIntoMap(path); // (Root No., (Value, Order)) !
350 // then map back onto (Value, (Root Nr., Order)) (i.e. sorted by value to pick the highest ones)
351 IndexKeySetList->ReMapAdaptiveCriteriaListToValue(mol);
352
353 // pick the ones still below threshold and mark as to be adaptively updated
354 if (IndexKeySetList->IsAdaptiveCriteriaListEmpty()) {
[47d041]355 ELOG(2, "Unable to parse file, incrementing all.");
[91207d]356 status = true;
[851be8]357 } else {
[91207d]358 // mark as false all sites that are below threshold already
359 status = IndexKeySetList->MarkUpdateCandidates(AtomMask, Order, mol);
[246e13]360 }
361
362 delete[](IndexKeySetList);
363 } else { // global increase of Bond Order
364 for(molecule::const_iterator iter = mol->begin(); iter != mol->end(); ++iter) {
[91207d]365 if (AtomMask.isTrue((*iter)->getNr())) { // skip masked out
366 // remove all that have reached desired order
367 if ((Order != 0) && ((*iter)->AdaptiveOrder >= Order)) // && ((*iter)->AdaptiveOrder < MinimumRingSize[(*iter)->getNr()]))
368 AtomMask.setFalse((*iter)->getNr());
369 else
[246e13]370 status = true;
371 }
372 }
[f96874]373 if ((!Order) && (!LoopDoneAlready)) // single stepping, just check
[246e13]374 status = true;
375
376 if (!status) {
377 if (Order == 0)
[47d041]378 LOG(1, "Single stepping done.");
[246e13]379 else
[47d041]380 LOG(1, "Order at every site is already equal or above desired order " << Order << ".");
[246e13]381 }
382 }
383
384 PrintAtomMask(AtomMask, mol->getAtomCount()); // for debugging
385
386 return status;
387};
388
389/** Stores pairs (Atom::Nr, Atom::AdaptiveOrder) into file.
390 * Atoms not present in the file get "-1".
391 * \param &path path to file ORDERATSITEFILE
392 * \return true - file writable, false - not writable
393 */
[e71325]394bool Fragmentation::StoreOrderAtSiteFile(const std::string &path)
[246e13]395{
396 string line;
397 ofstream file;
398
399 line = path + ORDERATSITEFILE;
400 file.open(line.c_str());
[47d041]401 LOG(1, "Writing OrderAtSite " << ORDERATSITEFILE << " ... ");
[246e13]402 if (file.good()) {
403 for_each(mol->begin(),mol->end(),bind2nd(mem_fun(&atom::OutputOrder), &file));
404 file.close();
[47d041]405 LOG(1, "done.");
[246e13]406 return true;
407 } else {
[47d041]408 LOG(1, "failed to open file " << line << ".");
[246e13]409 return false;
410 }
411};
412
413
414/** Parses pairs(Atom::Nr, Atom::AdaptiveOrder) from file and stores in molecule's Atom's.
415 * Atoms not present in the file get "0".
[2a0eb0]416 * \param atomids atoms to fragment, used in AtomMask
[246e13]417 * \param &path path to file ORDERATSITEFILEe
418 * \return true - file found and scanned, false - file not found
419 * \sa ParseKeySetFile() and CheckAdjacencyFileAgainstMolecule() as this is meant to be used in conjunction with the two
420 */
[e71325]421bool Fragmentation::ParseOrderAtSiteFromFile(const std::vector<atomId_t> &atomids, const std::string &path)
[246e13]422{
[339910]423 Info FunctionInfo(__func__);
[ed9da4]424 typedef unsigned char order_t;
425 typedef std::map<atomId_t, order_t> OrderArray_t;
426 OrderArray_t OrderArray;
[2a0eb0]427 AtomMask_t MaxArray(atomids);
[246e13]428 bool status;
429 int AtomNr, value;
430 string line;
431 ifstream file;
432
433 line = path + ORDERATSITEFILE;
434 file.open(line.c_str());
435 if (file.good()) {
436 while (!file.eof()) { // parse from file
437 AtomNr = -1;
438 file >> AtomNr;
439 if (AtomNr != -1) { // test whether we really parsed something (this is necessary, otherwise last atom is set twice and to 0 on second time)
440 file >> value;
441 OrderArray[AtomNr] = value;
442 file >> value;
[ed9da4]443 MaxArray.setValue(AtomNr, (bool)value);
[47d041]444 //LOG(2, "AtomNr " << AtomNr << " with order " << (int)OrderArray[AtomNr] << " and max order set to " << (int)MaxArray[AtomNr] << ".");
[246e13]445 }
446 }
447 file.close();
448
449 // set atom values
[59fff1]450 for(molecule::iterator iter=mol->begin();iter!=mol->end();++iter){
[246e13]451 (*iter)->AdaptiveOrder = OrderArray[(*iter)->getNr()];
[ed9da4]452 (*iter)->MaxOrder = MaxArray.isTrue((*iter)->getNr());
[246e13]453 }
454 //SetAtomValueToIndexedArray( OrderArray, &atom::getNr(), &atom::AdaptiveOrder );
455 //SetAtomValueToIndexedArray( MaxArray, &atom::getNr(), &atom::MaxOrder );
456
[47d041]457 LOG(1, "\t ... done.");
[246e13]458 status = true;
459 } else {
[47d041]460 LOG(1, "\t ... failed to open file " << line << ".");
[246e13]461 status = false;
462 }
463
464 return status;
465};
466
[339910]467/** Fills the root stack for sites to be used as root in fragmentation depending on order or adaptivity criteria
468 * Again, as in \sa FillBondStructureFromReference steps recursively through each Leaf in this chain list of molecule's.
469 * \param &RootStack stack to be filled
470 * \param AtomMask defines true/false per global Atom::Nr to mask in/out each nuclear site
471 * \return true - stack is non-empty, fragmentation necessary, false - stack is empty, no more sites to update
472 */
473void Fragmentation::FillRootStackForSubgraphs(KeyStack &RootStack, const AtomMask_t &AtomMask)
474{
475 for(molecule::const_iterator iter = mol->begin(); iter != mol->end(); ++iter) {
476 const atom * const Father = (*iter)->GetTrueFather();
477 if (AtomMask.isTrue(Father->getNr())) // apply mask
478 if ((saturation == DontSaturate) || ((*iter)->getType()->getAtomicNumber() != 1)) // skip hydrogen
479 RootStack.push_front((*iter)->getNr());
480 }
481}
482
483/** The indices per keyset are compared to the respective father's Atom::Nr in each subgraph and thus put into \a **&FragmentList.
484 * \param *KeySetList list with all keysets
485 * \param ListOfLocalAtoms Lookup table for each subgraph and index of each atom in global molecule, may be NULL on start, then it is filled
486 * \param **&FragmentList list to be allocated and returned
487 * \param FreeList true - ***ListOfLocalAtoms is free'd before return, false - it is not
488 * \retuen true - success, false - failure
489 */
[bfbd4a]490bool Fragmentation::AssignKeySetsToFragment(const Graph &KeySetList, ListOfLocalAtoms_t &ListOfLocalAtoms, Graph &FragmentList, bool FreeList)
[339910]491{
492 Info FunctionInfo(__func__);
493 bool status = true;
494 size_t KeySetCounter = 0;
495
496 // fill ListOfLocalAtoms if NULL was given
497 if (!mol->FillListOfLocalAtoms(ListOfLocalAtoms, mol->getAtomCount())) {
498 LOG(1, "Filling of ListOfLocalAtoms failed.");
499 return false;
500 }
501
502 if (KeySetList.size() != 0) { // if there are some scanned keysets at all
503 // assign scanned keysets
504 KeySet TempSet;
[bfbd4a]505 for (Graph::const_iterator runner = KeySetList.begin(); runner != KeySetList.end(); runner++) { // key sets contain global numbers!
[339910]506 if (ListOfLocalAtoms[mol->FindAtom(*((*runner).first.begin()))->getNr()] != NULL) {// as we may assume that that bond structure is unchanged, we only test the first key in each set
507 // translate keyset to local numbers
508 for (KeySet::iterator sprinter = (*runner).first.begin(); sprinter != (*runner).first.end(); sprinter++)
509 TempSet.insert(ListOfLocalAtoms[mol->FindAtom(*sprinter)->getNr()]->getNr());
510 // insert into FragmentList
511 FragmentList.insert(GraphPair(TempSet, pair<int, double> (KeySetCounter++, (*runner).second.second)));
512 }
513 TempSet.clear();
514 }
515 } else
516 LOG(1, "KeySetList is NULL or empty.");
517
518 if (FreeList) {
519 // free the index lookup list
520 ListOfLocalAtoms.clear();
521 }
522 return status;
523}
524
525/** Translate list into global numbers (i.e. ones that are valid in "this" molecule, not in MolecularWalker->Leaf)
526 * \param &FragmentList Graph with local numbers per fragment
527 * \param &TotalNumberOfKeySets global key set counter
528 * \param &TotalGraph Graph to be filled with global numbers
529 */
530void Fragmentation::TranslateIndicesToGlobalIDs(Graph &FragmentList, int &TotalNumberOfKeySets, Graph &TotalGraph)
531{
532 Info FunctionInfo(__func__);
533 for (Graph::iterator runner = FragmentList.begin(); runner != FragmentList.end(); runner++) {
534 KeySet TempSet;
535 for (KeySet::iterator sprinter = (*runner).first.begin(); sprinter != (*runner).first.end(); sprinter++)
[2ab6b6]536 TempSet.insert((mol->FindAtom(*sprinter))->GetTrueFather()->getId());
[339910]537 TotalGraph.insert(GraphPair(TempSet, pair<int, double> (TotalNumberOfKeySets++, (*runner).second.second)));
538 }
539}
540
Note: See TracBrowser for help on using the repository browser.