/*
* Project: MoleCuilder
* Description: creates and alters molecular systems
* Copyright (C) 2010-2012 University of Bonn. All rights reserved.
* Copyright (C) 2013 Frederik Heber. All rights reserved.
*
*
* This file is part of MoleCuilder.
*
* MoleCuilder is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 2 of the License, or
* (at your option) any later version.
*
* MoleCuilder is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with MoleCuilder. If not, see .
*/
/*
* bondgraph.cpp
*
* Created on: Oct 29, 2009
* Author: heber
*/
// include config.h
#ifdef HAVE_CONFIG_H
#include
#endif
// boost::graph uses placement new
#include
#include "CodePatterns/MemDebug.hpp"
#include
#include
#include
#include
#include
#include
#include
#include
#include "Atom/atom.hpp"
#include "Bond/bond.hpp"
#include "Graph/BondGraph.hpp"
#include "Box.hpp"
#include "Element/element.hpp"
#include "CodePatterns/Info.hpp"
#include "CodePatterns/Log.hpp"
#include "CodePatterns/Range.hpp"
#include "CodePatterns/Verbose.hpp"
#include "molecule.hpp"
#include "Element/periodentafel.hpp"
#include "Fragmentation/MatrixContainer.hpp"
#include "Graph/DepthFirstSearchAnalysis.hpp"
#include "LinearAlgebra/Vector.hpp"
#include "World.hpp"
#include "WorldTime.hpp"
const double BondGraph::BondThreshold = 0.4; //!< CSD threshold in bond check which is the width of the interval whose center is the sum of the covalent radii
BondGraph::BondGraph() :
BondLengthMatrix(NULL),
IsAngstroem(true)
{}
BondGraph::BondGraph(bool IsA) :
BondLengthMatrix(NULL),
IsAngstroem(IsA)
{}
BondGraph::~BondGraph()
{
CleanupBondLengthTable();
}
void BondGraph::CleanupBondLengthTable()
{
if (BondLengthMatrix != NULL) {
delete(BondLengthMatrix);
}
}
bool BondGraph::LoadBondLengthTable(
std::istream &input)
{
Info FunctionInfo(__func__);
bool status = true;
MatrixContainer *TempContainer = NULL;
// allocate MatrixContainer
if (BondLengthMatrix != NULL) {
LOG(1, "MatrixContainer for Bond length already present, removing.");
delete(BondLengthMatrix);
BondLengthMatrix = NULL;
}
TempContainer = new MatrixContainer;
// parse in matrix
if ((input.good()) && (status = TempContainer->ParseMatrix(input, 0, 1, 0))) {
LOG(1, "Parsing bond length matrix successful.");
} else {
ELOG(1, "Parsing bond length matrix failed.");
status = false;
}
if (status) // set to not NULL only if matrix was parsed
BondLengthMatrix = TempContainer;
else {
BondLengthMatrix = NULL;
delete(TempContainer);
}
return status;
}
double BondGraph::GetBondLength(
int firstZ,
int secondZ) const
{
double return_length;
if ((firstZ < 0) || (firstZ >= (int)BondLengthMatrix->Matrix[0].size()))
return -1.;
if ((secondZ < 0) || (secondZ >= (int)BondLengthMatrix->Matrix[0][firstZ].size()))
return -1.;
if (BondLengthMatrix == NULL) {
return_length = -1.;
} else {
return_length = BondLengthMatrix->Matrix[0][firstZ][secondZ];
}
LOG(4, "INFO: Request for length between " << firstZ << " and "
<< secondZ << ": " << return_length << ".");
return return_length;
}
range BondGraph::CovalentMinMaxDistance(
const element * const Walker,
const element * const OtherWalker) const
{
range MinMaxDistance(0.,0.);
MinMaxDistance.first = OtherWalker->getCovalentRadius() + Walker->getCovalentRadius();
MinMaxDistance.first *= (IsAngstroem) ? 1. : 1. / AtomicLengthToAngstroem;
MinMaxDistance.last = MinMaxDistance.first + BondThreshold;
MinMaxDistance.first -= BondThreshold;
return MinMaxDistance;
}
range BondGraph::BondLengthMatrixMinMaxDistance(
const element * const Walker,
const element * const OtherWalker) const
{
ASSERT(BondLengthMatrix, "BondGraph::BondLengthMatrixMinMaxDistance() called without NULL BondLengthMatrix.");
ASSERT(Walker, "BondGraph::BondLengthMatrixMinMaxDistance() - illegal element given.");
ASSERT(OtherWalker, "BondGraph::BondLengthMatrixMinMaxDistance() - illegal other element given.");
range MinMaxDistance(0.,0.);
MinMaxDistance.first = GetBondLength(Walker->getAtomicNumber()-1, OtherWalker->getAtomicNumber()-1);
MinMaxDistance.first *= (IsAngstroem) ? 1. : 1. / AtomicLengthToAngstroem;
MinMaxDistance.last = MinMaxDistance.first + BondThreshold;
MinMaxDistance.first -= BondThreshold;
return MinMaxDistance;
}
range BondGraph::getMinMaxDistance(
const element * const Walker,
const element * const OtherWalker) const
{
range MinMaxDistance(0.,0.);
if (BondLengthMatrix == NULL) {// safety measure if no matrix has been parsed yet
LOG(5, "DEBUG: Using Covalent radii criterion for [min,max) distances.");
MinMaxDistance = CovalentMinMaxDistance(Walker, OtherWalker);
} else {
LOG(5, "DEBUG: Using tabulated bond table criterion for [min,max) distances.");
MinMaxDistance = BondLengthMatrixMinMaxDistance(Walker, OtherWalker);
}
return MinMaxDistance;
}
range BondGraph::getMinMaxDistance(
const BondedParticle * const Walker,
const BondedParticle * const OtherWalker) const
{
return getMinMaxDistance(Walker->getType(), OtherWalker->getType());
}
range BondGraph::getMinMaxDistanceSquared(
const BondedParticle * const Walker,
const BondedParticle * const OtherWalker) const
{
// use non-squared version
range MinMaxDistance(getMinMaxDistance(Walker, OtherWalker));
// and square
MinMaxDistance.first *= MinMaxDistance.first;
MinMaxDistance.last *= MinMaxDistance.last;
return MinMaxDistance;
}
LinkedCell::LinkedCell_View BondGraph::getLinkedCell(const double max_distance) const
{
return World::getInstance().getLinkedCell(max_distance);
}
std::set< const element *> BondGraph::getElementSetFromNumbers(const std::set &Set) const
{
std::set< const element *> PresentElements;
for(std::set::const_iterator iter = Set.begin(); iter != Set.end(); ++iter)
PresentElements.insert( World::getInstance().getPeriode()->FindElement(*iter) );
return PresentElements;
}
Box &BondGraph::getDomain() const
{
return World::getInstance().getDomain();
}
unsigned int BondGraph::getTime() const
{
return WorldTime::getTime();
}
bool BondGraph::operator==(const BondGraph &other) const
{
if (IsAngstroem != other.IsAngstroem)
return false;
if (((BondLengthMatrix != NULL) && (other.BondLengthMatrix == NULL))
|| ((BondLengthMatrix == NULL) && (other.BondLengthMatrix != NULL)))
return false;
if ((BondLengthMatrix != NULL) && (other.BondLengthMatrix != NULL))
if (*BondLengthMatrix != *other.BondLengthMatrix)
return false;
return true;
}
/** Corrects the bond degree by one at most if necessary.
*
* We are constraint to bonds in \a egdes, if the candidate bond is in edges, we
* just don't increment its bond degree. We do not choose an alternative for this
* atom.
*
* \param edges only these edges can be updated
* \return number of corrections done
*/
int BondGraph::CorrectBondDegree(atom *_atom, const std::set& edges) const
{
int NoBonds = 0;
int OtherNoBonds = 0;
int FalseBondDegree = 0;
int CandidateDeltaNoBonds = -1;
atom *OtherWalker = NULL;
bond::ptr CandidateBond;
NoBonds = _atom->CountBonds();
LOG(3, "Walker " << *_atom << ": " << (int)_atom->getType()->getNoValenceOrbitals() << " > " << NoBonds << "?");
const int DeltaNoBonds = (int)(_atom->getType()->getNoValenceOrbitals()) - NoBonds;
if (DeltaNoBonds > 0) { // we have a mismatch, check all bonding partners for mismatch
const BondList& ListOfBonds = _atom->getListOfBonds();
for (BondList::const_iterator Runner = ListOfBonds.begin(); Runner != ListOfBonds.end(); (++Runner)) {
OtherWalker = (*Runner)->GetOtherAtom(_atom);
OtherNoBonds = OtherWalker->CountBonds();
const int OtherDeltaNoBonds = (int)(OtherWalker->getType()->getNoValenceOrbitals()) - OtherNoBonds;
LOG(3, "OtherWalker " << *OtherWalker << ": " << (int)OtherWalker->getType()->getNoValenceOrbitals() << " > " << OtherNoBonds << "?");
if (OtherDeltaNoBonds > 0) { // check if possible candidate
const BondList& OtherListOfBonds = OtherWalker->getListOfBonds();
if ((CandidateBond == NULL)
|| (ListOfBonds.size() > OtherListOfBonds.size()) // pick the one with fewer number of bonds first
|| ((CandidateDeltaNoBonds < 0) || (CandidateDeltaNoBonds > OtherDeltaNoBonds)) ) // pick the one closer to fulfilling its valence first
{
CandidateDeltaNoBonds = OtherDeltaNoBonds;
CandidateBond = (*Runner);
LOG(3, "New candidate is " << *CandidateBond << ".");
}
}
}
if ((CandidateBond != NULL)) {
if (edges.find(CandidateBond) != edges.end()) {
CandidateBond->setDegree(CandidateBond->getDegree()+1);
LOG(2, "Increased bond degree for bond " << *CandidateBond << ".");
} else
LOG(2, "Bond " << *CandidateBond << " is not in edges.");
} else {
ELOG(2, "Could not find correct degree for atom " << *_atom << ".");
FalseBondDegree++;
}
}
return FalseBondDegree;
}
/** Sets the weight of all connected bonds to one.
*/
void BondGraph::resetBondDegree(atom *_atom) const
{
const BondList &ListOfBonds = _atom->getListOfBonds();
for (BondList::const_iterator BondRunner = ListOfBonds.begin();
BondRunner != ListOfBonds.end();
++BondRunner)
(*BondRunner)->setDegree(1);
}
std::set BondGraph::getBackEdges() const
{
DepthFirstSearchAnalysis DFS;
DFS();
const std::deque& backedgestack = DFS.getBackEdgeStack();
std::set backedges(backedgestack.begin(), backedgestack.end());
return backedges;
}
std::set BondGraph::getTreeEdges() const
{
const std::set backedges = getBackEdges();
std::set alledges;
const World& world = World::getInstance();
for(World::AtomConstIterator iter = world.getAtomIter();
iter != world.atomEnd(); ++iter) {
const BondList &ListOfBonds = (*iter)->getListOfBonds();
alledges.insert(ListOfBonds.begin(), ListOfBonds.end());
}
std::set treeedges;
std::set_difference(
alledges.begin(), alledges.end(),
backedges.begin(), backedges.end(),
std::inserter(treeedges, treeedges.begin()));
return treeedges;
}
struct hasDelta {
bool operator()(atom *_atom) {
const double delta =
_atom->getType()->getNoValenceOrbitals() - _atom->CountBonds();
return (fabs(delta) > 0);
}
};
typedef std::set deltaatoms_t;
typedef std::set deltaedges_t;
static int gatherAllDeltaAtoms(
const deltaatoms_t &allatoms,
deltaatoms_t &deltaatoms)
{
static hasDelta delta;
deltaatoms.clear();
for (deltaatoms_t::const_iterator iter = allatoms.begin();
iter != allatoms.end(); ++iter) {
if (delta(*iter))
deltaatoms.insert(*iter);
}
return deltaatoms.size();
}
typedef boost::bimap AtomIndexLookup_t;
static AtomIndexLookup_t createAtomIndexLookup(
const deltaedges_t &edges)
{
AtomIndexLookup_t AtomIndexLookup;
size_t index = 0;
for (deltaedges_t::const_iterator edgeiter = edges.begin();
edgeiter != edges.end(); ++edgeiter) {
const bond::ptr & _bond = *edgeiter;
// insert left into lookup
std::pair< AtomIndexLookup_t::iterator, bool > inserter =
AtomIndexLookup.insert( AtomIndexLookup_t::value_type(index, _bond->leftatom) );
if (inserter.second)
++index;
// insert right into lookup
inserter = AtomIndexLookup.insert( AtomIndexLookup_t::value_type(index, _bond->rightatom) );
if (inserter.second)
++index;
}
return AtomIndexLookup;
}
typedef std::map< std::pair, bond::ptr> BondLookup_t;
static BondLookup_t createBondLookup(
const deltaedges_t &edges)
{
BondLookup_t BondLookup;
for (deltaedges_t::const_iterator edgeiter = edges.begin();
edgeiter != edges.end(); ++edgeiter) {
const bond::ptr & _bond = *edgeiter;
// insert bond into pair lookup
BondLookup.insert(
std::make_pair(
std::make_pair(_bond->leftatom, _bond->rightatom),
_bond)
);
}
return BondLookup;
}
typedef boost::adjacency_list graph_t;
typedef std::vector::vertex_descriptor> mate_t;
static void fillEdgesIntoMatching(
graph_t &g,
mate_t &mate,
const AtomIndexLookup_t &AtomIndexLookup,
const BondLookup_t &BondLookup,
deltaedges_t &matching
)
{
matching.clear();
boost::graph_traits::vertex_iterator vi, vi_end;
for(tie(vi,vi_end) = boost::vertices(g); vi != vi_end; ++vi)
if (mate[*vi] != boost::graph_traits::null_vertex() && *vi < mate[*vi]) {
atom * leftatom = AtomIndexLookup.left.at(*vi);
atom * rightatom = AtomIndexLookup.left.at(mate[*vi]);
std::pair AtomPair(leftatom, rightatom);
std::pair reverseAtomPair(rightatom, leftatom);
BondLookup_t::const_iterator iter = BondLookup.find(AtomPair);
if (iter != BondLookup.end()) {
matching.insert(iter->second);
} else {
iter = BondLookup.find(reverseAtomPair);
if (iter != BondLookup.end()) {
matching.insert(iter->second);
} else
ASSERT(0, "fillEdgesIntoMatching() - cannot find ("+toString(*vi)+","
+toString(mate[*vi])+") in BondLookup.");
}
}
}
static bool createMatching(
deltaedges_t &deltaedges,
deltaedges_t &matching
)
{
// gather all vertices for graph in a lookup structure
// to translate the graphs indices to atoms and also to associated bonds
AtomIndexLookup_t AtomIndexLookup = createAtomIndexLookup(deltaedges);
BondLookup_t BondLookup = createBondLookup(deltaedges);
const int n_vertices = AtomIndexLookup.size();
// construct graph
graph_t g(n_vertices);
// add edges to graph
for (deltaedges_t::const_iterator edgeiter = deltaedges.begin();
edgeiter != deltaedges.end(); ++edgeiter) {
const bond::ptr & _bond = *edgeiter;
boost::add_edge(
AtomIndexLookup.right.at(_bond->leftatom),
AtomIndexLookup.right.at(_bond->rightatom),
g);
}
// mate structure contains unique edge partner to every vertex in matching
mate_t mate(n_vertices);
// get maximum matching over given edges
bool success = boost::checked_edmonds_maximum_cardinality_matching(g, &mate[0]);
if (success) {
LOG(1, "STATUS: Found a matching of size " << matching_size(g, &mate[0]) << ".");
fillEdgesIntoMatching(g, mate, AtomIndexLookup, BondLookup, matching);
} else {
ELOG(2, "Failed to find maximum matching.");
}
return success;
}
bool BondGraph::checkBondDegree(const deltaatoms_t &allatoms) const
{
deltaatoms_t deltaatoms;
return (gatherAllDeltaAtoms(allatoms, deltaatoms) == 0);
}
int BondGraph::calculateBondDegree(const deltaatoms_t &allatoms) const
{
deltaatoms_t deltaatoms;
deltaedges_t deltaedges;
deltaedges_t matching;
LOG(1, "STATUS: Calculating bond degrees.");
if (DoLog(2)) {
std::stringstream output;
output << "INFO: All atoms are: ";
BOOST_FOREACH( atom *_atom, allatoms) {
output << *_atom << " ";
}
LOG(2, output.str());
}
size_t IterCount = 0;
// 1. gather all atoms with delta > 0
while ((gatherAllDeltaAtoms(allatoms, deltaatoms) != 0) && (IterCount < 100)) {
if (DoLog(3)) {
std::stringstream output;
output << "DEBUG: Current delta atoms are: ";
BOOST_FOREACH( atom *_atom, deltaatoms) {
output << *_atom << " ";
}
LOG(3, output.str());
}
// 2. gather all edges that connect atoms with both(!) having delta > 0
deltaedges.clear();
for (deltaatoms_t::const_iterator atomiter = deltaatoms.begin();
atomiter != deltaatoms.end(); ++atomiter) {
const atom * const _atom = *atomiter;
const BondList& ListOfBonds = (*atomiter)->getListOfBonds();
for (BondList::const_iterator bonditer = ListOfBonds.begin();
bonditer != ListOfBonds.end();++bonditer) {
const bond::ptr _bond = *bonditer;
if ((_bond->leftatom == _atom) && (deltaatoms.find(_bond->rightatom) != deltaatoms.end()))
deltaedges.insert(_bond);
else if ((_bond->rightatom == _atom) && (deltaatoms.find(_bond->leftatom) != deltaatoms.end()))
deltaedges.insert(_bond);
}
}
if (DoLog(3)) {
std::stringstream output;
output << "DEBUG: Current delta bonds are: ";
BOOST_FOREACH( bond::ptr _bond, deltaedges) {
output << *_bond << " ";
}
LOG(3, output.str());
}
if (deltaedges.empty())
break;
// 3. build matching over these edges
if (!createMatching(deltaedges, matching))
break;
if (DoLog(2)) {
std::stringstream output;
output << "DEBUG: Resulting matching is: ";
BOOST_FOREACH( bond::ptr _bond, matching) {
output << *_bond << " ";
}
LOG(2, output.str());
}
// 4. increase degree for each edge of the matching
LOG(2, "DEBUG: Increasing bond degrees by one.");
for (deltaedges_t::iterator edgeiter = matching.begin();
edgeiter != matching.end(); ++edgeiter) {
(*edgeiter)->setDegree( (*edgeiter)->getDegree()+1 );
}
// 6. stop after a given number of iterations or when all atoms are happy.
++IterCount;
};
// check whether we still have deltaatoms
{
hasDelta delta;
for (deltaatoms_t::const_iterator iter = allatoms.begin();
iter != allatoms.end(); ++iter)
if (delta(*iter))
ELOG(2, "Could not find satisfy charge neutrality for atom " << **iter << ".");
}
return deltaedges.size();
}