Codebase list votca-xtp / debian/2021.1-1 src / libxtp / geometry_optimization.cc
debian/2021.1-1

Tree @debian/2021.1-1 (Download .tar.gz)

geometry_optimization.cc @debian/2021.1-1raw · history · blame

/*
 *            Copyright 2009-2020 The VOTCA Development Team
 *                       (http://www.votca.org)
 *
 *      Licensed under the Apache License, Version 2.0 (the "License")
 *
 * You may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 *              http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 *
 */

// Standard includes
#include "functional"

// Local VOTCA includes
#include "votca/xtp/bfgs_trm.h"
#include "votca/xtp/energy_costfunction.h"
#include "votca/xtp/forces.h"
#include "votca/xtp/geometry_optimization.h"
#include "votca/xtp/statetracker.h"

namespace votca {
namespace xtp {

void GeometryOptimization::Initialize(tools::Property& options) {

  std::string statestring = options.get(".state").as<std::string>();
  _opt_state.FromString(statestring);
  if (!_opt_state.Type().isExciton()) {
    throw std::runtime_error(
        "At the moment only excitonic states can be optimized");
  }
  // default convergence parameters from ORCA
  _conv.deltaE = options.get(".convergence.energy").as<double>();  // Hartree
  _conv.RMSForce =
      options.get(".convergence.RMSForce").as<double>();  // Hartree/Bohr
  _conv.MaxForce =
      options.get(".convergence.MaxForce").as<double>();  // Hartree/Bohr
  _conv.RMSStep = options.get(".convergence.RMSStep").as<double>();  // Bohr
  _conv.MaxStep = options.get(".convergence.MaxStep").as<double>();  // Bohr
  _trust_radius = options.get("optimizer.trust").as<double>();       // Angstrom
  _trust_radius *= tools::conv::ang2bohr;  // initial trust radius in a.u.

  _max_iteration = options.get(".maxiter").as<Index>();
  _trajfile = options.get(".trajectory_file").as<std::string>();
  _optimizer = options.get(".optimizer.method").as<std::string>();
  _force_options = options.get(".forces");
  _statetracker_options = options.get(".statetracker");
}

void GeometryOptimization::Evaluate() {
  XTP_LOG(Log::error, *_pLog)
      << "Requested geometry optimization of excited state "
      << _opt_state.ToString() << std::flush;

  StateTracker tracker;
  tracker.Initialize(_statetracker_options);
  tracker.setInitialState(_opt_state);
  tracker.setLogger(_pLog);
  tracker.PrintInfo();

  // get a force object
  Forces force_engine(_gwbse_engine, tracker);
  force_engine.Initialize(_force_options);
  force_engine.setLog(_pLog);
  XTP_LOG(Log::error, *_pLog)
      << (boost::format("Convergence of total energy: %1$8.6f Hartree ") %
          _conv.deltaE)
             .str()
      << std::flush;
  XTP_LOG(Log::error, *_pLog)
      << (boost::format("Convergence of RMS Force:    %1$8.6f Hartree/Bohr ") %
          _conv.RMSForce)
             .str()
      << std::flush;
  XTP_LOG(Log::error, *_pLog)
      << (boost::format("Convergence of Max Force:    %1$8.6f Hartree/Bohr ") %
          _conv.MaxForce)
             .str()
      << std::flush;
  XTP_LOG(Log::error, *_pLog)
      << (boost::format("Convergence of RMS Step:     %1$8.6f Bohr ") %
          _conv.RMSStep)
             .str()
      << std::flush;
  XTP_LOG(Log::error, *_pLog)
      << (boost::format("Convergence of Max Step:     %1$8.6f Bohr ") %
          _conv.MaxStep)
             .str()
      << std::flush;
  XTP_LOG(Log::error, *_pLog)
      << (boost::format("Initial trust radius:        %1$8.6f Bohr") %
          _trust_radius)
             .str()
      << std::flush;

  Energy_costfunction e_cost =
      Energy_costfunction(_gwbse_engine, tracker, _orbitals, force_engine);
  e_cost.setConvergenceParameters(_conv);
  e_cost.setLog(_pLog);
  // get the optimizer
  if (_optimizer == "BFGS-TRM") {
    BFGSTRM bfgstrm(e_cost);
    std::vector<std::function<void()> > callbacks;
    std::function<void()> reporting = std::bind(
        Report, std::cref(bfgstrm), std::cref(force_engine), std::ref(*_pLog));
    callbacks.push_back(reporting);
    std::function<void()> filewrite = std::bind(
        WriteTrajectory, _trajfile, _orbitals.QMAtoms(), std::cref(bfgstrm));
    callbacks.push_back(filewrite);
    bfgstrm.setCallbacks(callbacks);
    bfgstrm.setNumofIterations(_max_iteration);
    bfgstrm.setTrustRadius(_trust_radius);
    bfgstrm.setLog(_pLog);
    bfgstrm.Optimize(Energy_costfunction::QMAtoms2Vector(_orbitals.QMAtoms()));
  }
  return;
}

void GeometryOptimization::Report(const BFGSTRM& bfgstrm, const Forces& forces,
                                  Logger& pLog) {

  XTP_LOG(Log::error, pLog) << std::flush;
  XTP_LOG(Log::error, pLog)
      << (boost::format("=========== OPTIMIZATION SUMMARY "
                        "================================= "))
             .str()
      << std::flush;
  XTP_LOG(Log::error, pLog)
      << "At iteration  " << bfgstrm.getIteration() << std::flush;
  XTP_LOG(Log::error, pLog)
      << (boost::format(" ---- POSITIONS (Angstrom)   ")).str() << std::flush;
  XTP_LOG(Log::error, pLog)
      << (boost::format(" Atom\t x\t  y\t  z ")).str() << std::flush;
  const Eigen::VectorXd& atomvec = bfgstrm.getParameters();
  for (Index i = 0; i < atomvec.size(); i += 3) {
    XTP_LOG(Log::error, pLog)
        << (boost::format("%1$4d    %2$+1.4f  %3$+1.4f  %4$+1.4f") % (i / 3) %
            (atomvec(i) * votca::tools::conv::bohr2ang) %
            (atomvec(i + 1) * votca::tools::conv::bohr2ang) %
            (atomvec(i + 2) * votca::tools::conv::bohr2ang))
               .str()
        << std::flush;
  }
  XTP_LOG(Log::error, pLog)
      << (boost::format("   Total energy:     %1$12.8f Hartree ") %
          bfgstrm.getCost())
             .str()
      << std::flush;
  XTP_LOG(Log::error, pLog)
      << (boost::format("   Trust radius:     %1$12.8f Bohr     ") %
          bfgstrm.getTrustRadius())
             .str()
      << std::flush;
  forces.Report();
  return;
}

void GeometryOptimization::WriteTrajectory(const std::string& filename,
                                           QMMolecule& atoms,
                                           const BFGSTRM& bfgstrm) {
  std::ofstream ofs;
  if (bfgstrm.getIteration() == 0) {
    ofs.open(filename, std::ofstream::out);
  } else {
    ofs.open(filename, std::ofstream::app);
  }
  if (!ofs.is_open()) {
    throw std::runtime_error("Bad file handle: " + filename);
  }
  // write coordinates as xyz file
  ofs << atoms.size() << std::endl;
  ofs << "iteration " << bfgstrm.getIteration() << " energy "
      << bfgstrm.getCost() << " Hartree" << std::endl;
  Energy_costfunction::Vector2QMAtoms(bfgstrm.getParameters(), atoms);
  for (const QMAtom& atom : atoms) {
    const Eigen::Vector3d pos = atom.getPos() * tools::conv::bohr2ang;
    ofs << atom.getElement() << " " << pos.x() << " " << pos.y() << " "
        << pos.z() << std::endl;
  }
  ofs.close();
  return;
}

}  // namespace xtp
}  // namespace votca