File: BondStretch.cpp

package info (click to toggle)
rdkit 201203-3
  • links: PTS, VCS
  • area: main
  • in suites: wheezy
  • size: 37,840 kB
  • sloc: cpp: 93,902; python: 51,897; java: 5,192; ansic: 3,497; xml: 2,499; sql: 1,641; yacc: 1,518; lex: 1,076; makefile: 325; fortran: 183; sh: 153; cs: 51
file content (106 lines) | stat: -rw-r--r-- 3,458 bytes parent folder | download
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
// $Id: BondStretch.cpp 1528 2010-09-26 17:04:37Z glandrum $
//
//  Copyright (C) 2004-2006 Rational Discovery LLC
//
//   @@ All Rights Reserved @@
//  This file is part of the RDKit.
//  The contents are covered by the terms of the BSD license
//  which is included in the file license.txt, found at the root
//  of the RDKit source tree.
//

#include "BondStretch.h"
#include "Params.h"
#include <math.h>
#include <ForceField/ForceField.h>
#include <RDGeneral/Invariant.h>

namespace ForceFields {
  namespace UFF {
    namespace Utils {
      double calcBondRestLength(double bondOrder,
				const AtomicParams *end1Params,
				const AtomicParams *end2Params){
	PRECONDITION(bondOrder>0,"bad bond order");
	PRECONDITION(end1Params,"bad params pointer");
	PRECONDITION(end2Params,"bad params pointer");

	double ri=end1Params->r1,rj=end2Params->r1;

	// this is the pauling correction:
	double rBO = -Params::lambda * (ri + rj) * log(bondOrder);

	// O'Keefe and Breese electronegativity correction:
	double Xi=end1Params->GMP_Xi,Xj=end2Params->GMP_Xi;
	double rEN = ri*rj*(sqrt(Xi)-sqrt(Xj))*(sqrt(Xi)-sqrt(Xj)) /
	  (Xi*ri + Xj*rj);
    
	double res = ri + rj + rBO + rEN;
	return res;
      }
  
      double calcBondForceConstant(double restLength,
				   const AtomicParams *end1Params,
				   const AtomicParams *end2Params){
	double res = 2.0 * Params::G * end1Params->Z1 * end2Params->Z1 / pow(restLength,3.0);
	return res;
      }
    } // end of namespace Utils
  
    BondStretchContrib::BondStretchContrib(ForceField *owner,
					   unsigned int idx1,unsigned int idx2,
					   double bondOrder,
					   const AtomicParams *end1Params,
					   const AtomicParams *end2Params){
      PRECONDITION(owner,"bad owner");
      PRECONDITION(end1Params,"bad params pointer");
      PRECONDITION(end2Params,"bad params pointer");
      RANGE_CHECK(0,idx1,owner->positions().size()-1);
      RANGE_CHECK(0,idx2,owner->positions().size()-1);

      dp_forceField = owner;
      d_end1Idx = idx1;
      d_end2Idx = idx2;

      this->d_restLen = Utils::calcBondRestLength(bondOrder,
						  end1Params,end2Params);
      this->d_forceConstant = Utils::calcBondForceConstant(this->d_restLen,
							   end1Params,end2Params);
    }

    double BondStretchContrib::getEnergy(double *pos) const {
      PRECONDITION(dp_forceField,"no owner");
      PRECONDITION(pos,"bad vector");

      double distTerm=this->dp_forceField->distance(this->d_end1Idx,this->d_end2Idx,pos) -
	this->d_restLen;
      double res = 0.5*this->d_forceConstant*distTerm*distTerm;
      return res;
    }
    void BondStretchContrib::getGrad(double *pos,double *grad) const {
      PRECONDITION(dp_forceField,"no owner");
      PRECONDITION(pos,"bad vector");
      PRECONDITION(grad,"bad vector");


      double dist=this->dp_forceField->distance(this->d_end1Idx,this->d_end2Idx,pos);
      double preFactor = this->d_forceConstant*(dist-this->d_restLen);

      //std::cout << "\tDist("<<this->d_end1Idx<<","<<this->d_end2Idx<<") " << dist << std::endl;
      double *end1Coords = &(pos[3*this->d_end1Idx]);
      double *end2Coords = &(pos[3*this->d_end2Idx]);
      for(int i=0;i<3;i++){
	double dGrad;
	if(dist>0.0){
	  dGrad=preFactor * (end1Coords[i]-end2Coords[i])/dist;
	} else {
	  // move a small amount in an arbitrary direction
	  dGrad=this->d_forceConstant*.01;
	}
	grad[3*this->d_end1Idx+i] += dGrad;
	grad[3*this->d_end2Idx+i] -= dGrad;
      }    
    }
  
  }
}