- Timestamp:
- May 9, 2008, 2:20:08 PM (17 years ago)
- Branches:
- 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, 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
- Children:
- 0b05147
- Parents:
- de293ac
- Location:
- src
- Files:
-
- 3 edited
Legend:
- Unmodified
- Added
- Removed
-
src/builder.cpp
rde293ac rfc850d 720 720 int Z; 721 721 int j, axis, count, faktor; 722 int MinimumRingSize = -1;722 int *MinimumRingSize = NULL; 723 723 enum ConfigStatus config_present = absent; 724 724 MoleculeLeafClass *Subgraphs = NULL; … … 1140 1140 } 1141 1141 delete(Subgraphs); // we don't need the list here, so free everything 1142 delete[](MinimumRingSize); 1142 1143 Subgraphs = NULL; 1143 1144 end = clock(); -
src/molecules.cpp
rde293ac rfc850d 1179 1179 { 1180 1180 int No = 0; 1181 int MinimumRingSize;1181 int *MinimumRingSize = NULL; 1182 1182 MoleculeLeafClass *Subgraphs = NULL; 1183 1183 bond *Binder = first; … … 1190 1190 } 1191 1191 delete(Subgraphs); 1192 delete[](MinimumRingSize); 1192 1193 } 1193 1194 while(Binder->next != last) { … … 1437 1438 * \param *out output stream for debugging 1438 1439 * \param ReturnStack true - return pointer to atom stack of separable components, false - return NULL 1439 * \param MinimumRingSize contains smallest ring size in molecular structure on return or -1 if no rings were found1440 * \param *&MinimumRingSize contains smallest ring size in molecular structure on return or -1 if no rings were found 1440 1441 * \return list of each disconnected subgraph as an individual molecule class structure 1441 1442 */ 1442 MoleculeLeafClass * molecule::DepthFirstSearchAnalysis(ofstream *out, bool ReturnStack, int &MinimumRingSize)1443 MoleculeLeafClass * molecule::DepthFirstSearchAnalysis(ofstream *out, bool ReturnStack, int *&MinimumRingSize) 1443 1444 { 1444 1445 class StackClass<atom *> *AtomStack; … … 1630 1631 * \param *out output stream for debugging 1631 1632 * \param *BackEdgeStack stack with all back edges found during DFS scan 1632 * \param MinimumRingSize contains smallest ring size in molecular structure on return or -1 if no rings were found, if set is maximum search distance1633 * \param *&MinimumRingSize contains smallest ring size in molecular structure on return or -1 if no rings were found, if set is maximum search distance 1633 1634 * \todo BFS from the not-same-LP to find back to starting point of tributary cycle over more than one bond 1634 1635 */ 1635 void molecule::CyclicStructureAnalysis(ofstream *out, class StackClass<bond *> * BackEdgeStack, int &MinimumRingSize)1636 void molecule::CyclicStructureAnalysis(ofstream *out, class StackClass<bond *> * BackEdgeStack, int *&MinimumRingSize) 1636 1637 { 1637 1638 atom **PredecessorList = (atom **) Malloc(sizeof(atom *)*AtomCount, "molecule::CyclicStructureAnalysis: **PredecessorList"); … … 1642 1643 atom *Walker = NULL, *OtherAtom = NULL, *Root = NULL; 1643 1644 bond *Binder = NULL, *BackEdge = NULL; 1644 int RingSize, NumCycles ;1645 int RingSize, NumCycles, MinRingSize = -1; 1645 1646 1646 1647 // initialise each vertex as white with no predecessor, empty queue, color Root lightgray … … 1650 1651 ColorList[i] = white; 1651 1652 } 1653 MinimumRingSize = new int[AtomCount]; 1654 for(int i=0;i<AtomCount;i++) 1655 MinimumRingSize[i] = AtomCount; 1656 1652 1657 1653 1658 *out << Verbose(1) << "Back edge list - "; … … 1655 1660 1656 1661 *out << Verbose(1) << "Analysing cycles ... " << endl; 1657 if ((MinimumRingSize <= 2) || (MinimumRingSize > AtomCount))1658 MinimumRingSize = AtomCount;1659 1662 NumCycles = 0; 1660 1663 while (!BackEdgeStack->IsEmpty()) { … … 1670 1673 //*out << Verbose(1) << "---------------------------------------------------------------------------------------------------------" << endl; 1671 1674 OtherAtom = NULL; 1672 while ((Walker != Root) && ((OtherAtom == NULL) || (ShortestPathList[OtherAtom->nr] < MinimumRingSize ))) { // look for Root1675 while ((Walker != Root) && ((OtherAtom == NULL) || (ShortestPathList[OtherAtom->nr] < MinimumRingSize[Walker->GetTrueFather()->nr]))) { // look for Root 1673 1676 Walker = BFSStack->PopFirst(); 1674 1677 //*out << Verbose(2) << "Current Walker is " << *Walker << ", we look for SP to Root " << *Root << "." << endl; … … 1684 1687 ShortestPathList[OtherAtom->nr] = ShortestPathList[Walker->nr]+1; 1685 1688 //*out << Verbose(2) << "Coloring OtherAtom " << OtherAtom->Name << " lightgray, its predecessor is " << Walker->Name << " and its Shortest Path is " << ShortestPathList[OtherAtom->nr] << " egde(s) long." << endl; 1686 if (ShortestPathList[OtherAtom->nr] < MinimumRingSize ) { // Check for maximum distance1689 if (ShortestPathList[OtherAtom->nr] < MinimumRingSize[Walker->GetTrueFather()->nr]) { // Check for maximum distance 1687 1690 //*out << Verbose(3) << "Putting OtherAtom into queue." << endl; 1688 1691 BFSStack->Push(OtherAtom); … … 1711 1714 } 1712 1715 *out << Walker->Name << " with a length of " << RingSize << "." << endl << endl; 1713 if (RingSize < MinimumRingSize) 1714 MinimumRingSize = RingSize; 1716 // walk through all and set MinimumRingSize 1717 Walker = Root; 1718 while (Walker != BackEdge->rightatom) { 1719 Walker = PredecessorList[Walker->nr]; 1720 if (RingSize < MinimumRingSize[Walker->GetTrueFather()->nr]) 1721 MinimumRingSize[Walker->GetTrueFather()->nr] = RingSize; 1722 } 1723 if ((RingSize < MinRingSize) || (MinRingSize == -1)) 1724 MinRingSize = RingSize; 1715 1725 } else { 1716 *out << Verbose(1) << "No ring with length equal to or smaller than " << MinimumRingSize<< " found." << endl;1726 *out << Verbose(1) << "No ring containing " << *Root << " with length equal to or smaller than " << MinimumRingSize[Walker->GetTrueFather()->nr] << " found." << endl; 1717 1727 } 1718 1728 … … 1725 1735 } 1726 1736 } 1737 // go over all atoms 1738 Root = start; 1739 while(Root->next != end) { 1740 Root = Root->next; 1741 1742 if (MinimumRingSize[Root->GetTrueFather()->nr] == AtomCount) { // check whether MinimumRingSize is set, if not BFS to next where it is 1743 ShortestPathList[Walker->nr] = 0; 1744 BFSStack->ClearStack(); // start with empty BFS stack 1745 BFSStack->Push(Walker); 1746 TouchedStack->Push(Walker); 1747 //*out << Verbose(1) << "---------------------------------------------------------------------------------------------------------" << endl; 1748 OtherAtom = Walker; 1749 while ((Walker != Root) && (OtherAtom != NULL)) { // look for Root 1750 Walker = BFSStack->PopFirst(); 1751 //*out << Verbose(2) << "Current Walker is " << *Walker << ", we look for SP to Root " << *Root << "." << endl; 1752 for(int i=0;i<NumberOfBondsPerAtom[Walker->nr];i++) { 1753 Binder = ListOfBondsPerAtom[Walker->nr][i]; 1754 if (Binder != BackEdge) { // only walk along DFS spanning tree (otherwise we always find SP of one being backedge Binder) 1755 OtherAtom = Binder->GetOtherAtom(Walker); 1756 //*out << Verbose(2) << "Current OtherAtom is: " << OtherAtom->Name << " for bond " << *Binder << "." << endl; 1757 if (ColorList[OtherAtom->nr] == white) { 1758 TouchedStack->Push(OtherAtom); 1759 ColorList[OtherAtom->nr] = lightgray; 1760 PredecessorList[OtherAtom->nr] = Walker; // Walker is the predecessor 1761 ShortestPathList[OtherAtom->nr] = ShortestPathList[Walker->nr]+1; 1762 //*out << Verbose(2) << "Coloring OtherAtom " << OtherAtom->Name << " lightgray, its predecessor is " << Walker->Name << " and its Shortest Path is " << ShortestPathList[OtherAtom->nr] << " egde(s) long." << endl; 1763 if (MinimumRingSize[OtherAtom->GetTrueFather()->nr] != AtomCount) { // if the other atom is connected to a ring 1764 MinimumRingSize[Root->GetTrueFather()->nr] = ShortestPathList[OtherAtom->nr]+MinimumRingSize[OtherAtom->GetTrueFather()->nr]; 1765 OtherAtom = NULL; //break; 1766 break; 1767 } else 1768 BFSStack->Push(OtherAtom); 1769 } else { 1770 //*out << Verbose(3) << "Not Adding, has already been visited." << endl; 1771 } 1772 } else { 1773 //*out << Verbose(3) << "Not Visiting, is a back edge." << endl; 1774 } 1775 } 1776 ColorList[Walker->nr] = black; 1777 //*out << Verbose(1) << "Coloring Walker " << Walker->Name << " black." << endl; 1778 } 1779 1780 // now clean the lists 1781 while (!TouchedStack->IsEmpty()){ 1782 Walker = TouchedStack->PopFirst(); 1783 PredecessorList[Walker->nr] = NULL; 1784 ShortestPathList[Walker->nr] = -1; 1785 ColorList[Walker->nr] = white; 1786 } 1787 } 1788 *out << Verbose(1) << "Minimum ring size of " << *Root << " is " << MinimumRingSize[Root->GetTrueFather()->nr] << "." << endl; 1789 } 1790 1727 1791 1728 1729 if (MinimumRingSize != -1) 1730 *out << Verbose(1) << "Minimum ring size is " << MinimumRingSize << ", over " << NumCycles << " cycles total." << endl; 1792 if (MinRingSize != -1) 1793 *out << Verbose(1) << "Minimum ring size is " << MinRingSize << ", over " << NumCycles << " cycles total." << endl; 1731 1794 else 1732 1795 *out << Verbose(1) << "No rings were detected in the molecular structure." << endl; … … 2148 2211 bool status = false; 2149 2212 ifstream InputFile; 2150 stringstream line; 2213 2214 // initialize mask list 2215 for(int i=0;i<AtomCount;i++) 2216 AtomMask[i] = false; 2151 2217 2152 2218 if (Order < 0) { // adaptive increase of BondOrder per site 2153 for(int i=0;i<AtomCount;i++)2154 AtomMask[i] =false;2219 if (AtomMask[AtomCount] == true) // break after one step 2220 return false; 2155 2221 // parse the EnergyPerFragment file 2156 line.str(path); 2157 line << "/" << FRAGMENTPREFIX << ENERGYPERFRAGMENT; 2158 InputFile.open(line.str().c_str(), ios::in); 2159 if (InputFile != NULL) { 2160 char *buffer = (char *) Malloc(sizeof(char)*MAXSTRINGSIZE, "molecule::CheckOrderAtSite: *buffer"); 2222 char *buffer = (char *) Malloc(sizeof(char)*MAXSTRINGSIZE, "molecule::CheckOrderAtSite: *buffer"); 2223 sprintf(buffer, "%s/%s%s.dat", path, FRAGMENTPREFIX, ENERGYPERFRAGMENT); 2224 InputFile.open(buffer, ios::in); 2225 if ((InputFile != NULL) && (GlobalKeySetList != NULL)) { 2226 // transmorph graph keyset list into indexed KeySetList 2227 map<int,KeySet> IndexKeySetList; 2228 for(Graph::iterator runner = GlobalKeySetList->begin(); runner != GlobalKeySetList->end(); runner++) { 2229 IndexKeySetList.insert( pair<int,KeySet>(runner->second.first,runner->first) ); 2230 } 2161 2231 int lines = 0; 2162 2232 // count the number of lines, i.e. the number of fragments 2233 InputFile.getline(buffer, MAXSTRINGSIZE); // skip comment lines 2234 InputFile.getline(buffer, MAXSTRINGSIZE); 2163 2235 while(!InputFile.eof()) { 2164 2236 InputFile.getline(buffer, MAXSTRINGSIZE); 2165 2237 lines++; 2166 2238 } 2239 *out << Verbose(2) << "Scanned " << lines-1 << " lines." << endl; // one endline too much 2167 2240 InputFile.clear(); 2168 2241 InputFile.seekg(ios::beg); 2169 map< double, int> FragmentEnergies;2170 int No ;2171 intValue;2242 map<int, pair<double,int> > AdaptiveCriteriaList; // (Root No., (Value, Order)) ! 2243 int No, FragOrder; 2244 double Value; 2172 2245 // each line represents a fragment root (Atom::nr) id and its energy contribution 2246 InputFile.getline(buffer, MAXSTRINGSIZE); // skip comment lines 2247 InputFile.getline(buffer, MAXSTRINGSIZE); 2173 2248 while(!InputFile.eof()) { 2174 2249 InputFile.getline(buffer, MAXSTRINGSIZE); 2175 InputFile >> No; 2176 InputFile >> Value; 2177 FragmentEnergies.insert( make_pair(Value, No) ); 2178 } 2179 for(map<double, int>::iterator runner = FragmentEnergies.lower_bound(pow(1.,Order)); runner != FragmentEnergies.begin(); runner++) { 2180 // as the smallest number in each set has always been the root (we use global id to keep the doubles away), seek smallest and insert into AtomMask 2181 AtomMask[(*runner).second] = true; 2250 if (strlen(buffer) > 2) { 2251 //*out << Verbose(2) << "Scanning: " << buffer; 2252 stringstream line(buffer); 2253 line >> FragOrder; 2254 line >> ws >> No; 2255 line >> ws >> Value; // skip time entry 2256 line >> ws >> Value; 2257 No -= 1; // indices start at 1 in file, not 0 2258 //*out << Verbose(2) << " - yields (" << No << "," << Value << ")" << endl; 2259 2260 // clean the list of those entries that have been superceded by higher order terms already 2261 map<int,KeySet>::iterator marker = IndexKeySetList.find(No); // find keyset to Frag No. 2262 if (marker != IndexKeySetList.end()) { // if found 2263 // as the smallest number in each set has always been the root (we use global id to keep the doubles away), seek smallest and insert into AtomMask 2264 pair <map<int, pair<double,int> >::iterator, bool> InsertedElement = AdaptiveCriteriaList.insert( make_pair(*((*marker).second.begin()), pair<double,int>( Value, Order) )); 2265 map<int, pair<double,int> >::iterator PresentItem = InsertedElement.first; 2266 if (!InsertedElement.second) { // this root is already present 2267 if ((*PresentItem).second.second < FragOrder) // if order there is lower, update entry with higher-order term 2268 //if ((*PresentItem).second.first < (*runner).first) // as higher-order terms are not always better, we skip this part (which would always include this site into adaptive increase) 2269 { // if value is smaller, update value and order 2270 (*PresentItem).second.first = Value; 2271 (*PresentItem).second.second = FragOrder; 2272 *out << Verbose(2) << "Updated element (" << (*PresentItem).first << ",[" << (*PresentItem).second.first << "," << (*PresentItem).second.second << "])." << endl; 2273 } 2274 } else { 2275 *out << Verbose(2) << "Inserted element (" << (*PresentItem).first << ",[" << (*PresentItem).second.first << "," << (*PresentItem).second.second << "])." << endl; 2276 } 2277 } else { 2278 *out << Verbose(1) << "No Fragment under No. " << No << "found." << endl; 2279 } 2280 } 2281 } 2282 // then map back onto (Value, (Root Nr., Order)) (i.e. sorted by value to pick the highest ones) 2283 map<double, pair<int,int> > FinalRootCandidates; 2284 *out << Verbose(1) << "Root candidate list is: " << endl; 2285 for(map<int, pair<double,int> >::iterator runner = AdaptiveCriteriaList.begin(); runner != AdaptiveCriteriaList.end(); runner++) { 2286 Walker = FindAtom((*runner).first); 2287 if (Walker != NULL) { 2288 if ((*runner).second.second >= Walker->AdaptiveOrder) { // only insert if this is an "active" root site for the current order 2289 *out << Verbose(2) << "(" << (*runner).first << ",[" << (*runner).second.first << "," << (*runner).second.second << "])" << endl; 2290 FinalRootCandidates.insert( make_pair( (*runner).second.first, pair<int,int>((*runner).first, (*runner).second.second) ) ); 2291 } 2292 } else { 2293 cerr << "Atom No. " << (*runner).second.first << " was not found in this molecule." << endl; 2294 } 2295 } 2296 // pick the ones still below threshold and mark as to be adaptively updated 2297 for(map<double, pair<int,int> >::iterator runner = FinalRootCandidates.upper_bound(pow(10.,Order)); runner != FinalRootCandidates.end(); runner++) { 2298 No = (*runner).second.first; 2299 *out << Verbose(2) << "Root " << No << " is still above threshold (10^{" << Order <<"}: " << runner->first << ", setting entry " << No << " of Atom mask to true." << endl; 2300 AtomMask[No] = true; 2301 status = true; 2182 2302 } 2183 2303 // close and done 2184 2304 InputFile.close(); 2185 2305 InputFile.clear(); 2186 Free((void **)&buffer, "molecule::CheckOrderAtSite: *buffer");2187 2306 } else { 2188 cerr << "Unable to parse " << FRAGMENTPREFIX << ENERGYPERFRAGMENT << " file." << endl; 2189 return false; 2190 } 2307 cerr << "Unable to parse " << buffer << " file, incrementing all." << endl; 2308 while (Walker->next != end) { 2309 Walker = Walker->next; 2310 #ifdef ADDHYDROGEN 2311 if (Walker->type->Z != 1) // skip hydrogen 2312 #endif 2313 { 2314 AtomMask[Walker->nr] = true; // include all (non-hydrogen) atoms 2315 status = true; 2316 } 2317 } 2318 } 2319 Free((void **)&buffer, "molecule::CheckOrderAtSite: *buffer"); 2191 2320 // pick a given number of highest values and set AtomMask 2192 } else if (Order > 0){ // global increase of Bond Order2321 } else { // global increase of Bond Order 2193 2322 while (Walker->next != end) { 2194 2323 Walker = Walker->next; … … 2198 2327 { 2199 2328 AtomMask[Walker->nr] = true; // include all (non-hydrogen) atoms 2200 if ( Walker->AdaptiveOrder < Order)2329 if ((Order != 0) && (Walker->AdaptiveOrder < Order)) 2201 2330 status = true; 2202 2331 } 2203 2332 } 2333 if ((Order == 0) && (AtomMask[AtomCount] == true)) // single stepping, just check 2334 status = false; 2335 2204 2336 if (!status) 2205 2337 *out << Verbose(1) << "Order at every site is already equal or above desired order " << Order << "." << endl; 2206 } else { // single stepping, just check 2207 if (AtomMask[AtomCount] != true) // if true, we would've done a step already 2208 for(int i=0;i<=AtomCount;i++) 2209 AtomMask[i] = true; 2210 else 2211 status = false; 2212 } 2338 } 2339 2340 // print atom mask for debugging 2341 *out << " "; 2342 for(int i=0;i<AtomCount;i++) 2343 *out << (i % 10); 2344 *out << endl << "Atom mask is: "; 2345 for(int i=0;i<AtomCount;i++) 2346 *out << (AtomMask[i] ? "t" : "f"); 2347 *out << endl; 2213 2348 2214 2349 return status; … … 2268 2403 MoleculeListClass *BondFragments = NULL; 2269 2404 int *SortIndex = NULL; 2270 int MinimumRingSize;2405 int *MinimumRingSize = NULL; 2271 2406 int FragmentCounter; 2272 2407 MoleculeLeafClass *MolecularWalker = NULL; … … 2315 2450 2316 2451 // =================================== Begin of FRAGMENTATION =============================== 2317 // check cyclic lengths 2318 if ((MinimumRingSize != -1) && (Order >= MinimumRingSize)) { 2319 *out << Verbose(0) << "Bond order " << Order << " greater than or equal to Minimum Ring size of " << MinimumRingSize << " found is not allowed." << endl; 2320 } else { 2321 // ===== 6a. assign each keyset to its respective subgraph ===== 2322 Subgraphs->next->AssignKeySetsToFragment(out, this, ParsedFragmentList, ListOfLocalAtoms, FragmentList, (FragmentCounter = 0), false); 2323 2324 KeyStack *RootStack = new KeyStack[Subgraphs->next->Count()]; 2325 AtomMask = new bool[AtomCount+1]; 2326 for (int i=0;i<AtomCount;i++) // need to set to false to recognize in single-stepping (one global step set them all to true) 2327 AtomMask[i] = false; 2328 while (CheckOrderAtSite(out, AtomMask, ParsedFragmentList, Order)) { 2329 AtomMask[AtomCount] = true; // last plus one entry is used as marker that we have been through this loop once already in CheckOrderAtSite() 2330 // ===== 6b. fill RootStack for each subgraph (second adaptivity check) ===== 2331 Subgraphs->next->FillRootStackForSubgraphs(out, RootStack, AtomMask, Order, (FragmentCounter = 0)); 2332 2333 // ===== 7. fill the bond fragment list ===== 2334 FragmentCounter = 0; 2335 MolecularWalker = Subgraphs; 2336 while (MolecularWalker->next != NULL) { 2337 MolecularWalker = MolecularWalker->next; 2338 *out << Verbose(1) << "Fragmenting subgraph " << MolecularWalker << "." << endl; 2339 // output ListOfBondsPerAtom for debugging 2340 MolecularWalker->Leaf->OutputListOfBonds(out); 2341 if (MolecularWalker->Leaf->first->next != MolecularWalker->Leaf->last) { 2342 2343 // call BOSSANOVA method 2344 *out << Verbose(0) << endl << " ========== BOND ENERGY of subgraph " << FragmentCounter << " ========================= " << endl; 2345 MolecularWalker->Leaf->FragmentBOSSANOVA(out, FragmentList[FragmentCounter], RootStack[FragmentCounter]); 2346 } else { 2347 cerr << "Subgraph " << MolecularWalker << " has no atoms!" << endl; 2348 } 2349 FragmentCounter++; // next fragment list 2350 } 2351 } 2352 delete[](RootStack); 2353 delete[](AtomMask); 2354 } 2452 // ===== 6a. assign each keyset to its respective subgraph ===== 2453 Subgraphs->next->AssignKeySetsToFragment(out, this, ParsedFragmentList, ListOfLocalAtoms, FragmentList, (FragmentCounter = 0), false); 2454 2455 KeyStack *RootStack = new KeyStack[Subgraphs->next->Count()]; 2456 AtomMask = new bool[AtomCount+1]; 2457 while (CheckOrderAtSite(out, AtomMask, ParsedFragmentList, Order, configuration->configpath)) { 2458 AtomMask[AtomCount] = true; // last plus one entry is used as marker that we have been through this loop once already in CheckOrderAtSite() 2459 // ===== 6b. fill RootStack for each subgraph (second adaptivity check) ===== 2460 Subgraphs->next->FillRootStackForSubgraphs(out, RootStack, AtomMask, (FragmentCounter = 0)); 2461 2462 // ===== 7. fill the bond fragment list ===== 2463 FragmentCounter = 0; 2464 MolecularWalker = Subgraphs; 2465 while (MolecularWalker->next != NULL) { 2466 MolecularWalker = MolecularWalker->next; 2467 *out << Verbose(1) << "Fragmenting subgraph " << MolecularWalker << "." << endl; 2468 // output ListOfBondsPerAtom for debugging 2469 MolecularWalker->Leaf->OutputListOfBonds(out); 2470 if (MolecularWalker->Leaf->first->next != MolecularWalker->Leaf->last) { 2471 2472 // call BOSSANOVA method 2473 *out << Verbose(0) << endl << " ========== BOND ENERGY of subgraph " << FragmentCounter << " ========================= " << endl; 2474 MolecularWalker->Leaf->FragmentBOSSANOVA(out, FragmentList[FragmentCounter], RootStack[FragmentCounter], MinimumRingSize); 2475 } else { 2476 cerr << "Subgraph " << MolecularWalker << " has no atoms!" << endl; 2477 } 2478 FragmentCounter++; // next fragment list 2479 } 2480 } 2481 delete[](RootStack); 2482 delete[](AtomMask); 2355 2483 delete(ParsedFragmentList); 2484 delete[](MinimumRingSize); 2356 2485 2357 2486 // free the index lookup list … … 3627 3756 * \param Fragment&*List list of already present keystacks (adaptive scheme) or empty list 3628 3757 * \param &RootStack stack with all root candidates (unequal to each atom in complete molecule if adaptive scheme is applied) 3758 * \param *MinimumRingSize minimum ring size for each atom (molecule::Atomcount) 3629 3759 * \return pointer to Graph list 3630 3760 */ 3631 void molecule::FragmentBOSSANOVA(ofstream *out, Graph *&FragmentList, KeyStack &RootStack )3761 void molecule::FragmentBOSSANOVA(ofstream *out, Graph *&FragmentList, KeyStack &RootStack, int *MinimumRingSize) 3632 3762 { 3633 3763 Graph ***FragmentLowerOrdersList = NULL; … … 3675 3805 RootKeyNr = RootStack.front(); 3676 3806 RootStack.pop_front(); 3677 // increase adaptive order by one3678 3807 Walker = FindAtom(RootKeyNr); 3679 Walker->GetTrueFather()->AdaptiveOrder++; 3680 Order = Walker->AdaptiveOrder = Walker->GetTrueFather()->AdaptiveOrder; 3681 3682 // initialise Order-dependent entries of UniqueFragments structure 3683 FragmentSearch.BondsPerSPList = (bond **) Malloc(sizeof(bond *)*Order*2, "molecule::PowerSetGenerator: ***BondsPerSPList"); 3684 FragmentSearch.BondsPerSPCount = (int *) Malloc(sizeof(int)*Order, "molecule::PowerSetGenerator: *BondsPerSPCount"); 3685 for (int i=0;i<Order;i++) { 3686 FragmentSearch.BondsPerSPList[2*i] = new bond(); // start node 3687 FragmentSearch.BondsPerSPList[2*i+1] = new bond(); // end node 3688 FragmentSearch.BondsPerSPList[2*i]->next = FragmentSearch.BondsPerSPList[2*i+1]; // intertwine these two 3689 FragmentSearch.BondsPerSPList[2*i+1]->previous = FragmentSearch.BondsPerSPList[2*i]; 3690 FragmentSearch.BondsPerSPCount[i] = 0; 3691 } 3692 3693 // allocate memory for all lower level orders in this 1D-array of ptrs 3694 NumLevels = 1 << (Order-1); // (int)pow(2,Order); 3695 FragmentLowerOrdersList[RootNr] = (Graph **) Malloc(sizeof(Graph *)*NumLevels, "molecule::FragmentBOSSANOVA: **FragmentLowerOrdersList[]"); 3696 3697 // create top order where nothing is reduced 3698 *out << Verbose(0) << "==============================================================================================================" << endl; 3699 *out << Verbose(0) << "Creating KeySets of Bond Order " << Order << " for " << *Walker << ", NumLevels is " << NumLevels << ", " << (RootStack.size()-RootNr-1) << " Roots remaining." << endl; 3700 3701 // Create list of Graphs of current Bond Order (i.e. F_{ij}) 3702 FragmentLowerOrdersList[RootNr][0] = new Graph; 3703 FragmentSearch.TEFactor = 1.; 3704 FragmentSearch.Leaflet = FragmentLowerOrdersList[RootNr][0]; // set to insertion graph 3705 FragmentSearch.Root = Walker; 3706 NumMoleculesOfOrder[RootNr] = PowerSetGenerator(out, Walker->AdaptiveOrder, FragmentSearch, CompleteMolecule); 3707 *out << Verbose(1) << "Number of resulting KeySets is: " << NumMoleculesOfOrder[RootNr] << "." << endl; 3708 NumMolecules = 0; 3808 // check cyclic lengths 3809 if ((MinimumRingSize[Walker->GetTrueFather()->nr] != -1) && (Walker->GetTrueFather()->AdaptiveOrder+1 >= MinimumRingSize[Walker->GetTrueFather()->nr])) { 3810 *out << Verbose(0) << "Bond order " << Walker->GetTrueFather()->AdaptiveOrder << " of Root " << *Walker << " greater than or equal to Minimum Ring size of " << MinimumRingSize << " found is not allowed." << endl; 3811 } else { 3812 // increase adaptive order by one 3813 Walker->GetTrueFather()->AdaptiveOrder++; 3814 Order = Walker->AdaptiveOrder = Walker->GetTrueFather()->AdaptiveOrder; 3815 3816 // initialise Order-dependent entries of UniqueFragments structure 3817 FragmentSearch.BondsPerSPList = (bond **) Malloc(sizeof(bond *)*Order*2, "molecule::PowerSetGenerator: ***BondsPerSPList"); 3818 FragmentSearch.BondsPerSPCount = (int *) Malloc(sizeof(int)*Order, "molecule::PowerSetGenerator: *BondsPerSPCount"); 3819 for (int i=0;i<Order;i++) { 3820 FragmentSearch.BondsPerSPList[2*i] = new bond(); // start node 3821 FragmentSearch.BondsPerSPList[2*i+1] = new bond(); // end node 3822 FragmentSearch.BondsPerSPList[2*i]->next = FragmentSearch.BondsPerSPList[2*i+1]; // intertwine these two 3823 FragmentSearch.BondsPerSPList[2*i+1]->previous = FragmentSearch.BondsPerSPList[2*i]; 3824 FragmentSearch.BondsPerSPCount[i] = 0; 3825 } 3826 3827 // allocate memory for all lower level orders in this 1D-array of ptrs 3828 NumLevels = 1 << (Order-1); // (int)pow(2,Order); 3829 FragmentLowerOrdersList[RootNr] = (Graph **) Malloc(sizeof(Graph *)*NumLevels, "molecule::FragmentBOSSANOVA: **FragmentLowerOrdersList[]"); 3709 3830 3710 if ((NumLevels >> 1) > 0) { 3711 // create lower order fragments 3712 *out << Verbose(0) << "Creating list of unique fragments of lower Bond Order terms to be subtracted." << endl; 3713 Order = Walker->AdaptiveOrder; 3714 for (int source=0;source<(NumLevels >> 1);source++) { // 1-terms don't need any more splitting, that's why only half is gone through (shift again) 3715 // step down to next order at (virtual) boundary of powers of 2 in array 3716 while (source >= (1 << (Walker->AdaptiveOrder-Order))) // (int)pow(2,Walker->AdaptiveOrder-Order)) 3717 Order--; 3718 *out << Verbose(0) << "Current Order is: " << Order << "." << endl; 3719 for (int SubOrder=Order-1;SubOrder>0;SubOrder--) { 3720 int dest = source + (1 << (Walker->AdaptiveOrder-(SubOrder+1))); 3721 *out << Verbose(0) << "--------------------------------------------------------------------------------------------------------------" << endl; 3722 *out << Verbose(0) << "Current SubOrder is: " << SubOrder << " with source " << source << " to destination " << dest << "." << endl; 3723 3724 // every molecule is split into a list of again (Order - 1) molecules, while counting all molecules 3725 //*out << Verbose(1) << "Splitting the " << (*FragmentLowerOrdersList[RootNr][source]).size() << " molecules of the " << source << "th cell in the array." << endl; 3726 //NumMolecules = 0; 3727 FragmentLowerOrdersList[RootNr][dest] = new Graph; 3728 for(Graph::iterator runner = (*FragmentLowerOrdersList[RootNr][source]).begin();runner != (*FragmentLowerOrdersList[RootNr][source]).end(); runner++) { 3729 for (KeySet::iterator sprinter = (*runner).first.begin();sprinter != (*runner).first.end(); sprinter++) { 3730 Graph TempFragmentList; 3731 FragmentSearch.TEFactor = -(*runner).second.second; 3732 FragmentSearch.Leaflet = &TempFragmentList; // set to insertion graph 3733 FragmentSearch.Root = FindAtom(*sprinter); 3734 NumMoleculesOfOrder[RootNr] += PowerSetGenerator(out, SubOrder, FragmentSearch, (*runner).first); 3735 // insert new keysets FragmentList into FragmentLowerOrdersList[Walker->AdaptiveOrder-1][dest] 3736 *out << Verbose(1) << "Merging resulting key sets with those present in destination " << dest << "." << endl; 3737 InsertGraphIntoGraph(out, *FragmentLowerOrdersList[RootNr][dest], TempFragmentList, &NumMolecules); 3831 // create top order where nothing is reduced 3832 *out << Verbose(0) << "==============================================================================================================" << endl; 3833 *out << Verbose(0) << "Creating KeySets of Bond Order " << Order << " for " << *Walker << ", NumLevels is " << NumLevels << ", " << (RootStack.size()-RootNr-1) << " Roots remaining." << endl; 3834 3835 // Create list of Graphs of current Bond Order (i.e. F_{ij}) 3836 FragmentLowerOrdersList[RootNr][0] = new Graph; 3837 FragmentSearch.TEFactor = 1.; 3838 FragmentSearch.Leaflet = FragmentLowerOrdersList[RootNr][0]; // set to insertion graph 3839 FragmentSearch.Root = Walker; 3840 NumMoleculesOfOrder[RootNr] = PowerSetGenerator(out, Walker->AdaptiveOrder, FragmentSearch, CompleteMolecule); 3841 *out << Verbose(1) << "Number of resulting KeySets is: " << NumMoleculesOfOrder[RootNr] << "." << endl; 3842 NumMolecules = 0; 3843 3844 if ((NumLevels >> 1) > 0) { 3845 // create lower order fragments 3846 *out << Verbose(0) << "Creating list of unique fragments of lower Bond Order terms to be subtracted." << endl; 3847 Order = Walker->AdaptiveOrder; 3848 for (int source=0;source<(NumLevels >> 1);source++) { // 1-terms don't need any more splitting, that's why only half is gone through (shift again) 3849 // step down to next order at (virtual) boundary of powers of 2 in array 3850 while (source >= (1 << (Walker->AdaptiveOrder-Order))) // (int)pow(2,Walker->AdaptiveOrder-Order)) 3851 Order--; 3852 *out << Verbose(0) << "Current Order is: " << Order << "." << endl; 3853 for (int SubOrder=Order-1;SubOrder>0;SubOrder--) { 3854 int dest = source + (1 << (Walker->AdaptiveOrder-(SubOrder+1))); 3855 *out << Verbose(0) << "--------------------------------------------------------------------------------------------------------------" << endl; 3856 *out << Verbose(0) << "Current SubOrder is: " << SubOrder << " with source " << source << " to destination " << dest << "." << endl; 3857 3858 // every molecule is split into a list of again (Order - 1) molecules, while counting all molecules 3859 //*out << Verbose(1) << "Splitting the " << (*FragmentLowerOrdersList[RootNr][source]).size() << " molecules of the " << source << "th cell in the array." << endl; 3860 //NumMolecules = 0; 3861 FragmentLowerOrdersList[RootNr][dest] = new Graph; 3862 for(Graph::iterator runner = (*FragmentLowerOrdersList[RootNr][source]).begin();runner != (*FragmentLowerOrdersList[RootNr][source]).end(); runner++) { 3863 for (KeySet::iterator sprinter = (*runner).first.begin();sprinter != (*runner).first.end(); sprinter++) { 3864 Graph TempFragmentList; 3865 FragmentSearch.TEFactor = -(*runner).second.second; 3866 FragmentSearch.Leaflet = &TempFragmentList; // set to insertion graph 3867 FragmentSearch.Root = FindAtom(*sprinter); 3868 NumMoleculesOfOrder[RootNr] += PowerSetGenerator(out, SubOrder, FragmentSearch, (*runner).first); 3869 // insert new keysets FragmentList into FragmentLowerOrdersList[Walker->AdaptiveOrder-1][dest] 3870 *out << Verbose(1) << "Merging resulting key sets with those present in destination " << dest << "." << endl; 3871 InsertGraphIntoGraph(out, *FragmentLowerOrdersList[RootNr][dest], TempFragmentList, &NumMolecules); 3872 } 3738 3873 } 3874 *out << Verbose(1) << "Number of resulting molecules for SubOrder " << SubOrder << " is: " << NumMolecules << "." << endl; 3739 3875 } 3740 *out << Verbose(1) << "Number of resulting molecules for SubOrder " << SubOrder << " is: " << NumMolecules << "." << endl;3741 3876 } 3742 3877 } 3743 }3744 // now, we have completely filled each cell of FragmentLowerOrdersList[] for the current Walker->AdaptiveOrder3745 //NumMoleculesOfOrder[Walker->AdaptiveOrder-1] = NumMolecules;3746 TotalNumMolecules += NumMoleculesOfOrder[RootNr];3747 *out << Verbose(1) << "Number of resulting molecules for Order " << (int)Walker->GetTrueFather()->AdaptiveOrder << " is: " << NumMoleculesOfOrder[RootNr] << "." << endl;3748 RootStack.push_back(RootKeyNr); // put back on stack3749 RootNr++;3750 3751 // free Order-dependent entries of UniqueFragments structure for next loop cycle3752 Free((void **)&FragmentSearch.BondsPerSPCount, "molecule::PowerSetGenerator: *BondsPerSPCount");3753 for (int i=0;i<Order;i++) {3754 delete(FragmentSearch.BondsPerSPList[2*i]);3755 delete(FragmentSearch.BondsPerSPList[2*i+1]);3756 }3757 Free((void **)&FragmentSearch.BondsPerSPList, "molecule::PowerSetGenerator: ***BondsPerSPList");3878 // now, we have completely filled each cell of FragmentLowerOrdersList[] for the current Walker->AdaptiveOrder 3879 //NumMoleculesOfOrder[Walker->AdaptiveOrder-1] = NumMolecules; 3880 TotalNumMolecules += NumMoleculesOfOrder[RootNr]; 3881 *out << Verbose(1) << "Number of resulting molecules for Order " << (int)Walker->GetTrueFather()->AdaptiveOrder << " is: " << NumMoleculesOfOrder[RootNr] << "." << endl; 3882 RootStack.push_back(RootKeyNr); // put back on stack 3883 RootNr++; 3884 3885 // free Order-dependent entries of UniqueFragments structure for next loop cycle 3886 Free((void **)&FragmentSearch.BondsPerSPCount, "molecule::PowerSetGenerator: *BondsPerSPCount"); 3887 for (int i=0;i<Order;i++) { 3888 delete(FragmentSearch.BondsPerSPList[2*i]); 3889 delete(FragmentSearch.BondsPerSPList[2*i+1]); 3890 } 3891 Free((void **)&FragmentSearch.BondsPerSPList, "molecule::PowerSetGenerator: ***BondsPerSPList"); 3892 } 3758 3893 } 3759 3894 *out << Verbose(0) << "==============================================================================================================" << endl; -
src/molecules.hpp
rde293ac rfc850d 326 326 327 327 // Graph analysis 328 MoleculeLeafClass * DepthFirstSearchAnalysis(ofstream *out, bool ReturnStack, int &MinimumRingSize);329 void CyclicStructureAnalysis(ofstream *out, class StackClass<bond *> *BackEdgeStack, int &MinimumRingSize);328 MoleculeLeafClass * DepthFirstSearchAnalysis(ofstream *out, bool ReturnStack, int *&MinimumRingSize); 329 void CyclicStructureAnalysis(ofstream *out, class StackClass<bond *> *BackEdgeStack, int *&MinimumRingSize); 330 330 bond * FindNextUnused(atom *vertex); 331 331 void SetNextComponentNumber(atom *vertex, int nr); … … 353 353 void BreadthFirstSearchAdd(ofstream *out, molecule *Mol, atom **&AddedAtomList, bond **&AddedBondList, atom *Root, bond *Bond, int BondOrder, bool IsAngstroem); 354 354 /// -# BOSSANOVA 355 void FragmentBOSSANOVA(ofstream *out, Graph *&FragmentList, KeyStack &RootStack );355 void FragmentBOSSANOVA(ofstream *out, Graph *&FragmentList, KeyStack &RootStack, int *MinimumRingSize); 356 356 int PowerSetGenerator(ofstream *out, int Order, struct UniqueFragments &FragmentSearch, KeySet RestrictedKeySet); 357 357 bool BuildInducedSubgraph(ofstream *out, const molecule *Father); … … 413 413 bool AddLeaf(molecule *ptr, MoleculeLeafClass *Previous); 414 414 bool FillBondStructureFromReference(ofstream *out, molecule *reference, int &FragmentCounter, atom ***&ListOfLocalAtoms, bool FreeList = false); 415 bool FillRootStackForSubgraphs(ofstream *out, KeyStack *&RootStack, bool *AtomMask, int Order, int&FragmentCounter);415 bool FillRootStackForSubgraphs(ofstream *out, KeyStack *&RootStack, bool *AtomMask, int &FragmentCounter); 416 416 bool AssignKeySetsToFragment(ofstream *out, molecule *reference, Graph *KeySetList, atom ***&ListOfLocalAtoms, Graph **&FragmentList, int &FragmentCounter, bool FreeList = false); 417 417 bool FillListOfLocalAtoms(ofstream *out, atom ***&ListOfLocalAtoms, int &FragmentCounter, int GlobalAtomCount, bool &FreeList);
Note:
See TracChangeset
for help on using the changeset viewer.