/* -------------------------------------------------------------------------- * * OpenMM * * -------------------------------------------------------------------------- * * This is part of the OpenMM molecular simulation toolkit originating from * * Simbios, the NIH National Center for Physics-Based Simulation of * * Biological Structures at Stanford, funded under the NIH Roadmap for * * Medical Research, grant U54 GM072970. See https://simtk.org. * * * * Portions copyright (c) 2010 Stanford University and the Authors. * * Authors: Peter Eastman * * Contributors: * * * * Permission is hereby granted, free of charge, to any person obtaining a * * copy of this software and associated documentation files (the "Software"), * * to deal in the Software without restriction, including without limitation * * the rights to use, copy, modify, merge, publish, distribute, sublicense, * * and/or sell copies of the Software, and to permit persons to whom the * * Software is furnished to do so, subject to the following conditions: * * * * The above copyright notice and this permission notice shall be included in * * all copies or substantial portions of the Software. * * * * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR * * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, * * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL * * THE AUTHORS, CONTRIBUTORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, * * DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR * * OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE * * USE OR OTHER DEALINGS IN THE SOFTWARE. * * -------------------------------------------------------------------------- */ #include #include "openmm/internal/SplineFitter.h" #include "openmm/OpenMMException.h" using namespace OpenMM; using namespace std; void SplineFitter::createNaturalSpline(const vector& x, const vector& y, vector& deriv) { int n = x.size(); if (y.size() != n) throw OpenMMException("createNaturalSpline: x and y vectors must have same length"); if (n < 3) throw OpenMMException("createNaturalSpline: the length of the input array must be at least 3"); deriv.resize(n); // Create the system of equations to solve. vector a(n), b(n), c(n), rhs(n); a[0] = 0.0; b[0] = 1.0; c[0] = 0.0; rhs[0] = 0.0; for (int i = 1; i < n-1; i++) { a[i] = x[i]-x[i-1]; b[i] = 2.0*(x[i+1]-x[i-1]); c[i] = x[i+1]-x[i]; rhs[i] = 6.0*((y[i+1]-y[i])/(x[i+1]-x[i]) - (y[i]-y[i-1])/(x[i]-x[i-1])); } a[n-1] = 0.0; b[n-1] = 1.0; c[n-1] = 0.0; rhs[n-1] = 0.0; // Solve them. solveTridiagonalMatrix(a, b, c, rhs, deriv); } void SplineFitter::createPeriodicSpline(const vector& x, const vector& y, vector& deriv) { int n = x.size(); if (y.size() != n) throw OpenMMException("createPeriodicSpline: x and y vectors must have same length"); if (n < 3) throw OpenMMException("createPeriodicSpline: the length of the input array must be at least 3"); if (y[0] != y[n-1]) throw OpenMMException("createPeriodicSpline: the first and last points must have the same value"); deriv.resize(n); // Create the system of equations to solve. vector a(n), b(n), c(n), rhs(n); a[0] = 0.0; b[0] = 2.0*(x[1]-x[0]+x[n-1]-x[n-2]); c[0] = x[1]-x[0]; rhs[0] = 6.0*((y[1]-y[0])/(x[1]-x[0]) - (y[n-1]-y[n-2])/(x[n-1]-x[n-2])); for (int i = 1; i < n-1; i++) { a[i] = x[i]-x[i-1]; b[i] = 2.0*(x[i+1]-x[i-1]); c[i] = x[i+1]-x[i]; rhs[i] = 6.0*((y[i+1]-y[i])/(x[i+1]-x[i]) - (y[i]-y[i-1])/(x[i]-x[i-1])); } a[n-1] = 0.0; b[n-1] = 1.0; c[n-1] = 0.0; rhs[n-1] = 0.0; double beta = x[n-1]-x[n-2]; double alpha = -1.0; double gamma = -b[0]; // This is a cyclic tridiagonal matrix. We solve it using the Sherman-Morrison method, // which involves solving two tridiagonal systems. b[0] -= gamma; b[n-1] -= alpha*beta/gamma; solveTridiagonalMatrix(a, b, c, rhs, deriv); vector u(n, 0.0), z(n); u[0] = gamma; u[n-1] = alpha; solveTridiagonalMatrix(a, b, c, u, z); double scale = (deriv[0]+beta*deriv[n-1]/gamma)/(1.0+z[0]+beta*z[n-1]/gamma); for (int i = 0; i < n; i++) deriv[i] -= scale*z[i]; } double SplineFitter::evaluateSpline(const vector& x, const vector& y, const vector& deriv, double t) { int n = x.size(); if (t < x[0] || t > x[n-1]) throw OpenMMException("evaluateSpline: specified point is outside the range defined by the spline"); // Perform a binary search to identify the interval containing the point to evaluate. int lower = 0; int upper = n-1; while (upper-lower > 1) { int middle = (upper+lower)/2; if (x[middle] > t) upper = middle; else lower = middle; } // Evaluate the spline. double dx = x[upper]-x[lower]; double a = (x[upper]-t)/dx; double b = 1.0-a; return a*y[lower]+b*y[upper]+((a*a*a-a)*deriv[lower] + (b*b*b-b)*deriv[upper])*dx*dx/6.0; } double SplineFitter::evaluateSplineDerivative(const vector& x, const vector& y, const vector& deriv, double t) { int n = x.size(); if (t < x[0] || t > x[n-1]) throw OpenMMException("evaluateSplineDerivative: specified point is outside the range defined by the spline"); // Perform a binary search to identify the interval containing the point to evaluate. int lower = 0; int upper = n-1; while (upper-lower > 1) { int middle = (upper+lower)/2; if (x[middle] > t) upper = middle; else lower = middle; } // Evaluate the spline. double dx = x[upper]-x[lower]; double a = (x[upper]-t)/dx; double b = 1.0-a; double dadx = -1.0/dx; return dadx*y[lower]-dadx*y[upper]+((1.0-3.0*a*a)*deriv[lower] + (3.0*b*b-1.0)*deriv[upper])*dx/6.0; } void SplineFitter::solveTridiagonalMatrix(const vector& a, const vector& b, const vector& c, const vector& rhs, vector& sol) { int n = a.size(); vector gamma(n); // Decompose the matrix. sol[0] = rhs[0]/b[0]; double beta = b[0]; for (int i = 1; i < n; i++) { gamma[i] = c[i-1]/beta; beta = b[i]-a[i]*gamma[i]; sol[i] = (rhs[i]-a[i]*sol[i-1])/beta; } // Perform backsubstitation. for (int i = n-2; i >= 0; i--) sol[i] -= gamma[i+1]*sol[i+1]; }