From 0fa2bc8b9ac68299949adc68767bee18444f8eb2 Mon Sep 17 00:00:00 2001 From: Jean-Luc Fattebert Date: Wed, 31 Jul 2024 15:50:15 -0400 Subject: [PATCH] Add driver based on external LBFGS implementation --- CMakeLists.txt | 10 +- drivers/CMakeLists.txt | 8 + drivers/lbfgspp.cc | 249 ++++++++++++++++++++++++++++++ scripts/build_ubuntu22_openmpi.sh | 2 + src/CMakeLists.txt | 2 +- src/MGmol.cc | 2 +- src/{LBFGS.cc => MGmolLBFGS.cc} | 28 ++-- src/{LBFGS.h => MGmolLBFGS.h} | 6 +- src/MGmol_NEB.cc | 6 +- src/lbfgsrlx.cc | 4 +- 10 files changed, 292 insertions(+), 25 deletions(-) create mode 100644 drivers/lbfgspp.cc rename src/{LBFGS.cc => MGmolLBFGS.cc} (86%) rename src/{LBFGS.h => MGmolLBFGS.h} (92%) diff --git a/CMakeLists.txt b/CMakeLists.txt index 753a455d..920c93da 100644 --- a/CMakeLists.txt +++ b/CMakeLists.txt @@ -4,7 +4,7 @@ cmake_policy(SET CMP0074 NEW) project (MGmol C CXX Fortran) -set (CMAKE_CXX_STANDARD 11) +set (CMAKE_CXX_STANDARD 14) # Specify the location of additional CMAKE modules SET(CMAKE_MODULE_PATH ${PROJECT_SOURCE_DIR}/cmake_modules) @@ -156,6 +156,14 @@ if(${MGMOL_WITH_TRICUBIC} OR DEFINED TRICUBIC_ROOT) endif(${TRICUBIC_FOUND}) endif(${MGMOL_WITH_TRICUBIC} OR DEFINED TRICUBIC_ROOT) +# LBFGS++ (optional) +if(DEFINED LBFGSPP_INCLUDE) + message(STATUS "LBFGSpp at ${LBFGSPP_INCLUDE}") + if(DEFINED EIGEN_INCLUDE) + message(STATUS "EIGEN at ${EIGEN_INCLUDE}") + endif(DEFINED EIGEN_INCLUDE) +endif(DEFINED LBFGSPP_INCLUDE) + # clang-format (optional) set(MGMOL_WITH_CLANG_FORMAT FALSE CACHE BOOL "Indent code with clang-format") if(${MGMOL_WITH_CLANG_FORMAT}) diff --git a/drivers/CMakeLists.txt b/drivers/CMakeLists.txt index 4eb1c741..e01a0366 100644 --- a/drivers/CMakeLists.txt +++ b/drivers/CMakeLists.txt @@ -6,6 +6,14 @@ add_executable(example1 example1.cc) target_include_directories(check_input PRIVATE ${Boost_INCLUDE_DIRS}) target_include_directories(example1 PRIVATE ${Boost_INCLUDE_DIRS}) +if(DEFINED LBFGSPP_INCLUDE) + add_executable(lbfgspp lbfgspp.cc) + + target_include_directories(lbfgspp PRIVATE ${Boost_INCLUDE_DIRS}) + target_include_directories(lbfgspp PRIVATE ${EIGEN_INCLUDE}) + target_include_directories(lbfgspp PRIVATE ${LBFGSPP_INCLUDE}) +endif(DEFINED LBFGSPP_INCLUDE) target_link_libraries(check_input mgmol_src) target_link_libraries(example1 mgmol_src) +target_link_libraries(lbfgspp mgmol_src) diff --git a/drivers/lbfgspp.cc b/drivers/lbfgspp.cc new file mode 100644 index 00000000..6ee8be54 --- /dev/null +++ b/drivers/lbfgspp.cc @@ -0,0 +1,249 @@ +// Copyright (c) 2017, Lawrence Livermore National Security, LLC and +// UT-Battelle, LLC. +// Produced at the Lawrence Livermore National Laboratory and the Oak Ridge +// National Laboratory. +// LLNL-CODE-743438 +// All rights reserved. +// This file is part of MGmol. For details, see https://github.com/llnl/mgmol. +// Please also read this link https://github.com/llnl/mgmol/LICENSE + +// This driver uses the header-only LBFGS++ library +// https://github.com/yixuan/LBFGSpp + +#include +#include +#include + +#include "Control.h" +#include "ExtendedGridOrbitals.h" +#include "LocGridOrbitals.h" +#include "MGmol.h" +#include "MGmol_MPI.h" +#include "MPIdata.h" +#include "mgmol_run.h" + +#include +#include +#include +#include + +#include +namespace po = boost::program_options; + +using Eigen::VectorXd; + +class MGmolEnergyAndForces +{ +private: + MGmolInterface* mgmol_; + int n_; + std::vector& anumbers_; + +public: + MGmolEnergyAndForces( + MGmolInterface* mgmol, const int n, std::vector& anumbers) + : mgmol_(mgmol), n_(n), anumbers_(anumbers) + { + } + + double operator()(const VectorXd& x, VectorXd& grad) + { + std::vector positions(n_); + for (int i = 0; i < n_; i++) + { + positions[i] = x[i]; + } + if (MPIdata::onpe0) + { + std::cout << "Positions:" << std::endl; + for (std::vector::iterator it = positions.begin(); + it != positions.end(); it += 3) + { + for (int i = 0; i < 3; i++) + std::cout << " " << *(it + i); + std::cout << std::endl; + } + } + + // compute energy and forces using all MPI tasks + // expect positions to be replicated on all MPI tasks + std::vector forces(n_); + double fx + = mgmol_->evaluateEnergyAndForces(positions, anumbers_, forces); + // print out results + if (MPIdata::onpe0) + { + std::cout << "Energy: " << fx << std::endl; + std::cout << "Forces:" << std::endl; + for (std::vector::iterator it = forces.begin(); + it != forces.end(); it += 3) + { + double norm = 0.; + for (int i = 0; i < 3; i++) + { + double val = *(it + i); + std::cout << " " << val; + norm += val * val; + } + std::cout << " norm: " << std::sqrt(norm); + std::cout << std::endl; + } + } + + // set gradient to negative forces + for (int i = 0; i < n_; i++) + { + grad[i] = -1. * forces[i]; + } + return fx; + } +}; + +int main(int argc, char** argv) +{ + int mpirc = MPI_Init(&argc, &argv); + if (mpirc != MPI_SUCCESS) + { + std::cerr << "MPI Initialization failed!!!" << std::endl; + MPI_Abort(MPI_COMM_WORLD, 0); + } + + MPI_Comm comm = MPI_COMM_WORLD; + + /* + * Initialize general things, like magma, openmp, IO, ... + */ + mgmol_init(comm); + + /* + * read runtime parameters + */ + std::string input_filename(""); + std::string lrs_filename; + std::string constraints_filename(""); + + float total_spin = 0.; + bool with_spin = false; + + po::variables_map vm; + + // read from PE0 only + if (MPIdata::onpe0) + { + read_config(argc, argv, vm, input_filename, lrs_filename, + constraints_filename, total_spin, with_spin); + } + + MGmol_MPI::setup(comm, std::cout, with_spin); + MGmol_MPI& mmpi = *(MGmol_MPI::instance()); + MPI_Comm global_comm = mmpi.commGlobal(); + + /* + * Setup control struct with run time parameters + */ + Control::setup(global_comm, with_spin, total_spin); + Control& ct = *(Control::instance()); + + ct.setOptions(vm); + + int ret = ct.checkOptions(); + if (ret < 0) return ret; + + mmpi.bcastGlobal(input_filename); + mmpi.bcastGlobal(lrs_filename); + + // Enter main scope + { + if (MPIdata::onpe0) + { + std::cout << "-------------------------" << std::endl; + std::cout << "Construct MGmol object..." << std::endl; + std::cout << "-------------------------" << std::endl; + } + + MGmolInterface* mgmol; + if (ct.isLocMode()) + mgmol = new MGmol(global_comm, *MPIdata::sout, + input_filename, lrs_filename, constraints_filename); + else + mgmol = new MGmol(global_comm, *MPIdata::sout, + input_filename, lrs_filename, constraints_filename); + + if (MPIdata::onpe0) + { + std::cout << "-------------------------" << std::endl; + std::cout << "MGmol setup..." << std::endl; + std::cout << "-------------------------" << std::endl; + } + mgmol->setup(); + + if (MPIdata::onpe0) + { + std::cout << "-------------------------" << std::endl; + std::cout << "Setup done..." << std::endl; + std::cout << "-------------------------" << std::endl; + } + + // here we just use the atomic positions read in and used + // to initialize MGmol + std::vector positions; + mgmol->getAtomicPositions(positions); + std::vector anumbers; + mgmol->getAtomicNumbers(anumbers); + if (MPIdata::onpe0) + { + std::cout << "Positions:" << std::endl; + std::vector::iterator ita = anumbers.begin(); + for (std::vector::iterator it = positions.begin(); + it != positions.end(); it += 3) + { + std::cout << *ita; + for (int i = 0; i < 3; i++) + std::cout << " " << *(it + i); + std::cout << std::endl; + ita++; + } + } + + // Set up parameters + LBFGSpp::LBFGSParam param; + param.epsilon = 4e-4; + param.max_iterations = 100; + + // Create solver and function object + LBFGSpp::LBFGSSolver solver(param); + const int n = positions.size(); + if (MPIdata::onpe0) std::cout << "n = " << n << std::endl; + MGmolEnergyAndForces fun(mgmol, n, anumbers); + + // initial guess + VectorXd x = VectorXd::Zero(n); + int i = 0; + for (auto& pos : positions) + { + x[i++] = pos; + } + + double eks; + int niter = solver.minimize(fun, x, eks); + + std::cout << niter << " iterations" << std::endl; + + delete mgmol; + + } // close main scope + + mgmol_finalize(); + + mpirc = MPI_Finalize(); + if (mpirc != MPI_SUCCESS) + { + std::cerr << "MPI Finalize failed!!!" << std::endl; + } + + time_t tt; + time(&tt); + if (onpe0) std::cout << " Run ended at " << ctime(&tt) << std::endl; + + return 0; +} diff --git a/scripts/build_ubuntu22_openmpi.sh b/scripts/build_ubuntu22_openmpi.sh index 23f510b6..8f4c6266 100755 --- a/scripts/build_ubuntu22_openmpi.sh +++ b/scripts/build_ubuntu22_openmpi.sh @@ -26,6 +26,8 @@ cmake -DCMAKE_INSTALL_PREFIX=${INSTALL_DIR} \ -DMGMOL_WITH_CLANG_FORMAT=ON \ -DCMAKE_PREFIX_PATH=${HOME}/bin \ -D CMAKE_CXX_FLAGS="-Wall -pedantic -Wextra" \ + -DLBFGSPP_INCLUDE=/home/q8j/GIT/LBFGSpp/include \ + -DEIGEN_INCLUDE=/home/q8j/GIT/eigen \ .. # call make install diff --git a/src/CMakeLists.txt b/src/CMakeLists.txt index 3e1188fa..0f1edb08 100644 --- a/src/CMakeLists.txt +++ b/src/CMakeLists.txt @@ -68,7 +68,7 @@ set(SOURCES GrassmanLineMinimization.cc GrassmanCG.cc GrassmanCGSparse.cc - LBFGS.cc + MGmolLBFGS.cc IonicStepper.cc Energy.cc GramMatrix.cc diff --git a/src/MGmol.cc b/src/MGmol.cc index d5daabe6..f400d6ed 100644 --- a/src/MGmol.cc +++ b/src/MGmol.cc @@ -37,12 +37,12 @@ #include "Hamiltonian.h" #include "Ions.h" #include "KBPsiMatrixSparse.h" -#include "LBFGS.h" #include "LocGridOrbitals.h" #include "LocalizationRegions.h" #include "MDfiles.h" #include "MGkernels.h" #include "MGmol.h" +#include "MGmolLBFGS.h" #include "MLWFTransform.h" #include "MPIdata.h" #include "MasksSet.h" diff --git a/src/LBFGS.cc b/src/MGmolLBFGS.cc similarity index 86% rename from src/LBFGS.cc rename to src/MGmolLBFGS.cc index 9154b7b3..3cbfedcc 100644 --- a/src/LBFGS.cc +++ b/src/MGmolLBFGS.cc @@ -7,7 +7,7 @@ // This file is part of MGmol. For details, see https://github.com/llnl/mgmol. // Please also read this link https://github.com/llnl/mgmol/LICENSE -#include "LBFGS.h" +#include "MGmolLBFGS.h" #include "Control.h" #include "Electrostatic.h" #include "MGmol.h" @@ -16,7 +16,7 @@ #include "ProjectedMatrices.h" template -LBFGS::LBFGS(OrbitalsType** orbitals, Ions& ions, +MGmolLBFGS::MGmolLBFGS(OrbitalsType** orbitals, Ions& ions, Rho& rho, ConstraintSet& constraints, std::shared_ptr lrs, ClusterOrbitals* local_cluster, MasksSet& masks, MasksSet& corrmasks, Electrostatic& electrostat, @@ -37,7 +37,7 @@ LBFGS::LBFGS(OrbitalsType** orbitals, Ions& ions, } template -void LBFGS::setup(const double dt) +void MGmolLBFGS::setup(const double dt) { if (lrs_) ref_lrs_ = std::shared_ptr( @@ -77,14 +77,14 @@ void LBFGS::setup(const double dt) } template -LBFGS::~LBFGS() +MGmolLBFGS::~MGmolLBFGS() { delete vh_init_; delete stepper_; } template -void LBFGS::reset(const double dt) +void MGmolLBFGS::reset(const double dt) { delete vh_init_; delete stepper_; @@ -93,7 +93,7 @@ void LBFGS::reset(const double dt) } template -int LBFGS::quenchElectrons(const int itmax, double& etot) +int MGmolLBFGS::quenchElectrons(const int itmax, double& etot) { etot_i_[0] = etot_i_[1]; etot_i_[1] = etot_i_[2]; @@ -104,7 +104,7 @@ int LBFGS::quenchElectrons(const int itmax, double& etot) } template -void LBFGS::updateRefs() +void MGmolLBFGS::updateRefs() { if (!stepper_->check_last_step_accepted()) { @@ -128,7 +128,7 @@ void LBFGS::updateRefs() } template <> -void LBFGS::updateRefMasks() +void MGmolLBFGS::updateRefMasks() { Control& ct = *(Control::instance()); @@ -147,12 +147,12 @@ void LBFGS::updateRefMasks() } template <> -void LBFGS::updateRefMasks() +void MGmolLBFGS::updateRefMasks() { } template -void LBFGS::setQuenchTol() const +void MGmolLBFGS::setQuenchTol() const { Control& ct = *(Control::instance()); ct.conv_tol = 0.1 * stepper_->etol(); @@ -163,7 +163,7 @@ void LBFGS::setQuenchTol() const } template -void LBFGS::updatePotAndMasks() +void MGmolLBFGS::updatePotAndMasks() { if (!stepper_->check_last_step_accepted()) electrostat_.setupInitialVh(*vh_init_); @@ -172,10 +172,10 @@ void LBFGS::updatePotAndMasks() } template -bool LBFGS::lbfgsLastStepNotAccepted() const +bool MGmolLBFGS::lbfgsLastStepNotAccepted() const { return !stepper_->check_last_step_accepted(); } -template class LBFGS; -template class LBFGS; +template class MGmolLBFGS; +template class MGmolLBFGS; diff --git a/src/LBFGS.h b/src/MGmolLBFGS.h similarity index 92% rename from src/LBFGS.h rename to src/MGmolLBFGS.h index a085af05..c8348d24 100644 --- a/src/LBFGS.h +++ b/src/MGmolLBFGS.h @@ -27,7 +27,7 @@ class MGmol; class KBPsiMatrixInterface; template -class LBFGS : public IonicAlgorithm +class MGmolLBFGS : public IonicAlgorithm { private: OrbitalsType** orbitals_; @@ -54,12 +54,12 @@ class LBFGS : public IonicAlgorithm void setup(const double dt); public: - LBFGS(OrbitalsType** orbitals, Ions& ions, Rho& rho, + MGmolLBFGS(OrbitalsType** orbitals, Ions& ions, Rho& rho, ConstraintSet& constraints, std::shared_ptr lrs, ClusterOrbitals* local_cluster, MasksSet& masks, MasksSet& corrmasks, Electrostatic& electrostat, const double dt, MGmol&); - ~LBFGS() override; + ~MGmolLBFGS() override; void reset(const double dt); diff --git a/src/MGmol_NEB.cc b/src/MGmol_NEB.cc index 5008ea30..b0f70e6f 100644 --- a/src/MGmol_NEB.cc +++ b/src/MGmol_NEB.cc @@ -9,8 +9,8 @@ #include "DFTsolver.h" #include "FIRE.h" -#include "LBFGS.h" #include "MGmol.h" +#include "MGmolLBFGS.h" #include "ProjectedMatricesInterface.h" using namespace std; @@ -34,8 +34,8 @@ void MGmol::geomOptimSetup() switch (ct.AtomsDynamic()) { case AtomsDynamicType::LBFGS: - geom_optimizer_ = new LBFGS(¤t_orbitals_, *ions_, *rho_, - *constraints_, *lrs_, local_cluster_, *currentMasks_, + geom_optimizer_ = new MGmolLBFGS(¤t_orbitals_, *ions_, + *rho_, *constraints_, *lrs_, local_cluster_, *currentMasks_, *corrMasks_, *electrostat_, ct.dt, *this); break; diff --git a/src/lbfgsrlx.cc b/src/lbfgsrlx.cc index 6698eef0..b66305a1 100644 --- a/src/lbfgsrlx.cc +++ b/src/lbfgsrlx.cc @@ -13,10 +13,10 @@ #include "Electrostatic.h" #include "Energy.h" #include "Ions.h" -#include "LBFGS.h" #include "LBFGS_IonicStepper.h" #include "LocalizationRegions.h" #include "MGmol.h" +#include "MGmolLBFGS.h" #include "MGmol_blas1.h" #include "MPIdata.h" #include "MasksSet.h" @@ -35,7 +35,7 @@ void MGmol::lbfgsrlx(OrbitalsType** orbitals, Ions& ions) { Control& ct = *(Control::instance()); - LBFGS lbfgs(orbitals, ions, *rho_, *constraints_, lrs_, + MGmolLBFGS lbfgs(orbitals, ions, *rho_, *constraints_, lrs_, local_cluster_.get(), *currentMasks_, *corrMasks_, *electrostat_, ct.dt, *this);