mirror of
https://github.com/rdkit/rdkit.git
synced 2026-06-04 21:54:27 +08:00
* stub for YAeHMOP integration * switch to using catch for the test * first steps towards calling from rdkit * switch to using some smart pointers * runs, but has unpredictable seg faults. Needs valgrinding * cleanup * allow hacky linux builds too * start using unique_ptrs for memory management * don't find princ axes by default Support probably isn't compiled in. * save computed properties * additional cleanup * start working on a smarter way to get the code * start making the cmake externalproject support work * seems to work * does not work yet * this actually seems to build and work on linux * theoretically more robust * handle the parameter file * first wrapper * try to get mac builds working too * initial pass at including reduced charge and overlap population matrices * update docs * backup commit * switch to using cmake to handle the C++ spec * switch to using a results object for the eHT calculations * support fermi and total E
136 lines
4.3 KiB
C++
136 lines
4.3 KiB
C++
//
|
|
// Copyright (C) 2018 Greg Landrum
|
|
//
|
|
#include "EHTTools.h"
|
|
#include <GraphMol/RDKitBase.h>
|
|
#include <GraphMol/MolOps.h>
|
|
#include <mutex>
|
|
#include <fstream>
|
|
|
|
extern "C" {
|
|
#include <yaehmop/bind.h>
|
|
}
|
|
|
|
namespace RDKit {
|
|
namespace EHTTools {
|
|
const std::string _EHTCharge = "_EHTCharge";
|
|
const std::string _EHTMullikenOP = "_EHTMullikenOP";
|
|
const std::string _EHTChargeMatrix = "_EHTChargeMatrix";
|
|
|
|
// we should only call into the C code, which uses tons of globals, from one
|
|
// thread at a time. This mutex enforces that.
|
|
std::mutex yaehmop_mutex;
|
|
|
|
bool runMol(const ROMol &mol, EHTResults &results, int confId) {
|
|
std::lock_guard<std::mutex> lock(yaehmop_mutex);
|
|
|
|
// -----------------------------
|
|
// ----- BOILERPLATE -----------
|
|
// -----------------------------
|
|
FILE *nullfile = fopen("nul", "w");
|
|
status_file = nullfile;
|
|
output_file = nullfile;
|
|
FILE *dest = fopen("run.out", "w+");
|
|
|
|
unit_cell = (cell_type *)calloc(1, sizeof(cell_type));
|
|
details = (detail_type *)calloc(1, sizeof(detail_type));
|
|
|
|
set_details_defaults(details);
|
|
set_cell_defaults(unit_cell);
|
|
|
|
safe_strcpy(details->title, (char *)"RDKit job");
|
|
|
|
// molecular calculation
|
|
details->Execution_Mode = MOLECULAR;
|
|
details->num_KPOINTS = 1;
|
|
details->K_POINTS = (k_point_type *)calloc(1, sizeof(k_point_type));
|
|
details->K_POINTS[0].weight = 1.0;
|
|
details->avg_props = 1;
|
|
details->use_symmetry = 1;
|
|
details->find_princ_axes = 0;
|
|
details->net_chg_PRT = 1;
|
|
details->ROP_mat_PRT = 1;
|
|
details->Rchg_mat_PRT = 1;
|
|
|
|
unit_cell->using_Zmat = 0;
|
|
unit_cell->using_xtal_coords = 0;
|
|
// ---------------------------------
|
|
// ----- END BOILERPLATE -----------
|
|
// ---------------------------------
|
|
|
|
unit_cell->num_atoms = mol.getNumAtoms();
|
|
unit_cell->num_raw_atoms = unit_cell->num_atoms;
|
|
|
|
unit_cell->atoms =
|
|
(atom_type *)calloc(unit_cell->num_atoms, sizeof(atom_type));
|
|
|
|
const Conformer &conf = mol.getConformer(confId);
|
|
for (unsigned int i = 0; i < mol.getNumAtoms(); ++i) {
|
|
safe_strcpy(unit_cell->atoms[i].symb,
|
|
(char *)mol.getAtomWithIdx(i)->getSymbol().c_str());
|
|
auto p = conf.getAtomPos(i);
|
|
unit_cell->atoms[i].loc.x = p.x;
|
|
unit_cell->atoms[i].loc.y = p.y;
|
|
unit_cell->atoms[i].loc.z = p.z;
|
|
}
|
|
|
|
unit_cell->charge = MolOps::getFormalCharge(mol);
|
|
|
|
// -----------------------------
|
|
// ----- BOILERPLATE -----------
|
|
// -----------------------------
|
|
const char *parmFilePtr = nullptr;
|
|
std::string pfName = "";
|
|
if (std::getenv("BIND_PARM_FILE") == nullptr) {
|
|
auto rdbase = std::getenv("RDBASE");
|
|
if (rdbase != nullptr) {
|
|
pfName += rdbase;
|
|
pfName += "/Data/eht_parms.dat";
|
|
std::ifstream f(pfName.c_str());
|
|
if (f.good()) {
|
|
parmFilePtr = pfName.c_str();
|
|
} else {
|
|
std::cerr << "file " << pfName << " doesn't seem to exist" << std::endl;
|
|
}
|
|
}
|
|
}
|
|
|
|
fill_atomic_parms(unit_cell->atoms, unit_cell->num_atoms, NULL,
|
|
const_cast<char *>(parmFilePtr));
|
|
unit_cell->num_raw_atoms = unit_cell->num_atoms;
|
|
charge_to_num_electrons(unit_cell);
|
|
build_orbital_lookup_table(unit_cell, &num_orbs, &orbital_lookup_table);
|
|
|
|
run_eht(dest);
|
|
// ---------------------------------
|
|
// ----- END BOILERPLATE -----------
|
|
// ---------------------------------
|
|
|
|
// pull properties
|
|
results.numAtoms = mol.getNumAtoms();
|
|
results.numOrbitals = num_orbs;
|
|
results.fermiEnergy = properties.Fermi_E;
|
|
results.totalEnergy = properties.total_E;
|
|
results.atomicCharges = std::make_unique<double[]>(mol.getNumAtoms());
|
|
std::memcpy(static_cast<void *>(results.atomicCharges.get()),
|
|
static_cast<void *>(properties.net_chgs),
|
|
mol.getNumAtoms() * sizeof(double));
|
|
size_t sz = mol.getNumAtoms() * mol.getNumAtoms();
|
|
results.reducedChargeMatrix = std::make_unique<double[]>(sz);
|
|
memcpy(static_cast<void *>(results.reducedChargeMatrix.get()),
|
|
static_cast<void *>(properties.Rchg_mat), sz * sizeof(double));
|
|
|
|
sz = mol.getNumAtoms() * (mol.getNumAtoms() + 1) / 2;
|
|
results.reducedOverlapPopulationMatrix = std::make_unique<double[]>(sz);
|
|
memcpy(static_cast<void *>(results.reducedOverlapPopulationMatrix.get()),
|
|
static_cast<void *>(properties.ROP_mat), sz * sizeof(double));
|
|
|
|
cleanup_memory();
|
|
|
|
fclose(nullfile);
|
|
fclose(dest);
|
|
return true;
|
|
}
|
|
|
|
} // end of namespace EHTTools
|
|
} // end of namespace RDKit
|