29#ifndef N2P2_NO_SF_CACHE
59 log <<
"*****************************************"
60 "**************************************\n";
62 log <<
"WELCOME TO n²p², A SOFTWARE PACKAGE FOR NEURAL NETWORK "
64 log <<
"-------------------------------------------------------"
69 log <<
"------------------------------------------------------------\n";
72 log <<
"Compile date/time : " __DATE__
" " __TIME__
"\n";
73 log <<
"------------------------------------------------------------\n";
75 log <<
"Features/Flags:\n";
76 log <<
"------------------------------------------------------------\n";
77 log <<
"Symmetry function groups : ";
78#ifndef N2P2_NO_SF_GROUPS
83 log <<
"Symmetry function cache : ";
84#ifndef N2P2_NO_SF_CACHE
89 log <<
"Timing function available : ";
95 log <<
"Asymmetric polynomial SFs : ";
96#ifndef N2P2_NO_ASYM_POLY
101 log <<
"SF low neighbor number check : ";
102#ifndef N2P2_NO_NEIGH_CHECK
107 log <<
"SF derivative memory layout : ";
108#ifndef N2P2_FULL_SFD_MEMORY
113 log <<
"MPI explicitly disabled : ";
120 log <<
strpr(
"OpenMP threads : %d\n", omp_get_max_threads());
122 log <<
"------------------------------------------------------------\n";
125 log <<
"Please cite the following papers when publishing results "
126 "obtained with n²p²:\n";
127 log <<
"-----------------------------------------"
128 "--------------------------------------\n";
129 log <<
" * General citation for n²p² and the LAMMPS interface:\n";
131 log <<
" Singraber, A.; Behler, J.; Dellago, C.\n";
132 log <<
" Library-Based LAMMPS Implementation of High-Dimensional\n";
133 log <<
" Neural Network Potentials.\n";
134 log <<
" J. Chem. Theory Comput. 2019 15 (3), 1827–1840.\n";
135 log <<
" https://doi.org/10.1021/acs.jctc.8b00770\n";
136 log <<
"-----------------------------------------"
137 "--------------------------------------\n";
138 log <<
" * Additionally, if you use the NNP training features of n²p²:\n";
140 log <<
" Singraber, A.; Morawietz, T.; Behler, J.; Dellago, C.\n";
141 log <<
" Parallel Multistream Training of High-Dimensional Neural\n";
142 log <<
" Network Potentials.\n";
143 log <<
" J. Chem. Theory Comput. 2019, 15 (5), 3075–3092.\n";
144 log <<
" https://doi.org/10.1021/acs.jctc.8b01092\n";
145 log <<
"-----------------------------------------"
146 "--------------------------------------\n";
147 log <<
" * Additionally, if polynomial symmetry functions are used:\n";
149 log <<
" Bircher, M. P.; Singraber, A.; Dellago, C.\n";
150 log <<
" Improved Description of Atomic Environments Using Low-Cost\n";
151 log <<
" Polynomial Functions with Compact Support.\n";
152 log <<
" arXiv:2010.14414 [cond-mat, physics:physics] 2020.\n";
153 log <<
" https://arxiv.org/abs/2010.14414\n";
156 log <<
"*****************************************"
157 "**************************************\n";
165 log <<
"*** SETUP: SETTINGS FILE ****************"
166 "**************************************\n";
169 size_t numCriticalProblems =
settings.loadFile(fileName);
171 if (numCriticalProblems > 0)
173 throw runtime_error(
strpr(
"ERROR: %zu critical problem(s) were found "
174 "in settings file.\n", numCriticalProblems));
177 if (
settings.keywordExists(
"nnp_type"))
179 string nnpTypeString =
settings[
"nnp_type"];
188 log <<
"This settings file defines a short-range NNP (2G-HDNNP).\n";
192 log <<
"This settings file defines a NNP with electrostatics and\n"
193 "non-local charge transfer (4G-HDNNP).\n";
197 log <<
"This settings file defines a short-range NNP similar to\n"
198 "4G-HDNNP with additional charge NN but with neither\n"
199 "electrostatics nor global charge equilibration\n"
200 "(method by M. Bircher).\n";
204 throw runtime_error(
"ERROR: Unknown NNP type.\n");
207 log <<
"*****************************************"
208 "**************************************\n";
215 bool initialHardness)
225#ifndef N2P2_FULL_SFD_MEMORY
228#ifndef N2P2_NO_SF_CACHE
231#ifndef N2P2_NO_SF_GROUPS
244 log <<
"*** SETUP: NORMALIZATION ****************"
245 "**************************************\n";
249 if (
settings.keywordExists(
"mean_energy") &&
250 settings.keywordExists(
"conv_energy") &&
251 settings.keywordExists(
"conv_length"))
258 log <<
"Data set normalization is used.\n";
262 if (
settings.keywordExists(
"conv_charge"))
265 log <<
strpr(
"Conversion factor charge : %24.16E\n",
268 if (
settings.keywordExists(
"atom_energy"))
271 log <<
"Atomic energy offsets are used in addition to"
272 " data set normalization.\n";
273 log <<
"Offsets will be subtracted from reference energies BEFORE"
274 " normalization is applied.\n";
277 else if ((!
settings.keywordExists(
"mean_energy")) &&
278 (!
settings.keywordExists(
"conv_energy")) &&
279 (!
settings.keywordExists(
"conv_length")) &&
280 (!
settings.keywordExists(
"conv_charge")))
283 log <<
"Data set normalization is not used.\n";
287 throw runtime_error(
"ERROR: Incorrect usage of normalization"
289 " Use all or none of \"mean_energy\", \"conv_energy\""
290 " and \"conv_length\".\n");
295 log <<
"*****************************************"
296 "**************************************\n";
305 log <<
"*** SETUP: ELEMENT MAP ******************"
306 "**************************************\n";
311 for (
size_t i = 0; i <
elementMap.size(); ++i)
317 log <<
"*****************************************"
318 "**************************************\n";
326 log <<
"*** SETUP: ELEMENTS *********************"
327 "**************************************\n";
333 throw runtime_error(
"ERROR: Inconsistent number of elements.\n");
342 if (
settings.keywordExists(
"atom_energy"))
345 for (settings::Settings::KeyMap::const_iterator it = r.first;
346 it != r.second; ++it)
348 vector<string> args =
split(
reduce(it->second.first));
351 setAtomicEnergyOffset(atof(args.at(1).c_str()));
354 log <<
"Atomic energy offsets per element:\n";
355 for (
size_t i = 0; i <
elementMap.size(); ++i)
357 log <<
strpr(
"Element %2zu: %16.8E\n",
358 i,
elements.at(i).getAtomicEnergyOffset());
360 log <<
"Energy offsets are automatically subtracted from reference "
363 log <<
"*****************************************"
364 "**************************************\n";
370 string directoryPrefix,
371 string fileNameFormat)
374 log <<
"*** SETUP: ELECTROSTATICS ***************"
375 "**************************************\n";
382 for (settings::Settings::KeyMap::const_iterator it = r.first;
383 it != r.second; ++it)
385 vector<string> args =
split(
reduce(it->second.first));
387 double hardness = atof(args.at(1).c_str());
389 elements.at(element).setHardness(hardness);
393 double hardness =
elements.at(i).getHardness();
395 log <<
strpr(
"Initial atomic hardness for element %2s: %16.8E\n",
402 string actualFileNameFormat = directoryPrefix + fileNameFormat;
403 log <<
strpr(
"Atomic hardness file name format: %s\n",
404 actualFileNameFormat.c_str());
407 string fileName =
strpr(actualFileNameFormat.c_str(),
409 log <<
strpr(
"Atomic hardness for element %2s from file %s: ",
414 if (data.size() != 1)
416 throw runtime_error(
"ERROR: Atomic hardness data is "
419 double hardness = data.at(0);
421 elements.at(i).setHardness(hardness);
429 double maxQsigma = 0.0;
431 for (settings::Settings::KeyMap::const_iterator it = r.first;
432 it != r.second; ++it)
434 vector<string> args =
split(
reduce(it->second.first));
436 double qsigma = atof(args.at(1).c_str());
438 elements.at(element).setQsigma(qsigma);
440 maxQsigma = max(qsigma, maxQsigma);
442 log <<
"Gaussian width of charge distribution per element:\n";
443 for (
size_t i = 0; i <
elementMap.size(); ++i)
445 double qsigma =
elements.at(i).getQsigma();
447 log <<
strpr(
"Element %2zu: %16.8E\n",
453 if (
settings.keywordExists(
"kspace_solver"))
455 string kspace_solver_string =
settings[
"kspace_solver"];
456 if (kspace_solver_string ==
"ewald")
460 else if (kspace_solver_string ==
"pppm")
464 else if (kspace_solver_string ==
"ewald_lammps")
472 if (
settings.keywordExists(
"ewald_truncation_error_method"))
474 string truncation_method_string =
475 settings[
"ewald_truncation_error_method"];
476 if (truncation_method_string ==
"0")
480 else if (truncation_method_string ==
"1")
488 if (
settings.keywordExists(
"ewald_prec"))
497 throw runtime_error(
"ERROR: ewald_truncation_error_method 1 requires "
500 log <<
strpr(
"Ewald truncation error method: %16d\n",
502 log <<
strpr(
"Ewald precision: %16.8E\n",
506 log <<
strpr(
"Ewald expected maximum charge: %16.8E\n",
512 if (
settings.keywordExists(
"four_pi_epsilon"))
518 if (
settings.keywordExists(
"screen_electrostatics"))
520 vector<string> args =
split(
settings[
"screen_electrostatics"]);
521 double inner = atof(args.at(0).c_str());
522 double outer = atof(args.at(1).c_str());
524 if (args.size() > 2) type = args.at(2);
535 log <<
"Screening function not used.\n";
538 log <<
"*****************************************"
539 "**************************************\n";
547 log <<
"*** SETUP: CUTOFF FUNCTIONS *************"
548 "**************************************\n";
559 throw invalid_argument(
"ERROR: 0 <= alpha < 1.0 is required.\n");
563 log <<
"Inner cutoff = Symmetry function cutoff * alpha\n";
565 log <<
"Equal cutoff function type for all symmetry functions:\n";
569 log <<
"x := (r - rc * alpha) / (rc - rc * alpha)\n";
570 log <<
"f(x) = 1/2 * (cos(pi*x) + 1)\n";
575 log <<
"f(r) = tanh^3(1 - r/rc)\n";
578 log <<
"WARNING: Inner cutoff parameter not used in combination"
579 " with this cutoff function.\n";
585 log <<
"f(r) = c * tanh^3(1 - r/rc), f(0) = 1\n";
588 log <<
"WARNING: Inner cutoff parameter not used in combination"
589 " with this cutoff function.\n";
595 log <<
"x := (r - rc * alpha) / (rc - rc * alpha)\n";
596 log <<
"f(x) = (2x - 3)x^2 + 1\n";
601 log <<
"x := (r - rc * alpha) / (rc - rc * alpha)\n";
602 log <<
"f(x) = ((15 - 6x)x - 10)x^3 + 1\n";
607 log <<
"x := (r - rc * alpha) / (rc - rc * alpha)\n";
608 log <<
"f(x) = (x(x(20x - 70) + 84) - 35)x^4 + 1\n";
613 log <<
"x := (r - rc * alpha) / (rc - rc * alpha)\n";
614 log <<
"f(x) = (x(x((315 - 70x)x - 540) + 420) - 126)x^5 + 1\n";
619 log <<
"x := (r - rc * alpha) / (rc - rc * alpha)\n";
620 log <<
"f(x) = exp(-1 / 1 - x^2)\n";
626 log <<
"WARNING: Hard cutoff used!\n";
630 throw invalid_argument(
"ERROR: Unknown cutoff type.\n");
633 log <<
"*****************************************"
634 "**************************************\n";
642 log <<
"*** SETUP: SYMMETRY FUNCTIONS ***********"
643 "**************************************\n";
647 for (settings::Settings::KeyMap::const_iterator it = r.first; it != r.second; ++it)
649 vector<string> args =
split(
reduce(it->second.first));
652 elements.at(element).addSymmetryFunction(it->second.first,
656 log <<
"Abbreviations:\n";
657 log <<
"--------------\n";
658 log <<
"ind .... Symmetry function index.\n";
659 log <<
"ec ..... Central atom element.\n";
660 log <<
"tp ..... Symmetry function type.\n";
661 log <<
"sbtp ... Symmetry function subtype (e.g. cutoff type).\n";
662 log <<
"e1 ..... Neighbor 1 element.\n";
663 log <<
"e2 ..... Neighbor 2 element.\n";
664 log <<
"eta .... Gaussian width eta.\n";
665 log <<
"rs/rl... Shift distance of Gaussian or left cutoff radius "
667 log <<
"angl.... Left cutoff angle for polynomial.\n";
668 log <<
"angr.... Right cutoff angle for polynomial.\n";
669 log <<
"la ..... Angle prefactor lambda.\n";
670 log <<
"zeta ... Angle term exponent zeta.\n";
671 log <<
"rc ..... Cutoff radius / right cutoff radius for polynomial.\n";
672 log <<
"a ...... Free parameter alpha (e.g. cutoff alpha).\n";
673 log <<
"ln ..... Line number in settings file.\n";
676 for (vector<Element>::iterator it =
elements.begin();
680 it->sortSymmetryFunctions();
683 log <<
strpr(
"Short range atomic symmetry functions element %2s :\n",
684 it->getSymbol().c_str());
685 log <<
"--------------------------------------------------"
686 "-----------------------------------------------\n";
687 log <<
" ind ec tp sbtp e1 e2 eta rs/rl "
688 "rc angl angr la zeta a ln\n";
689 log <<
"--------------------------------------------------"
690 "-----------------------------------------------\n";
691 log << it->infoSymmetryFunctionParameters();
692 log <<
"--------------------------------------------------"
693 "-----------------------------------------------\n";
703 log <<
strpr(
"Minimum cutoff radius for element %2s: %f\n",
707 log <<
strpr(
"Maximum cutoff radius (global) : %f\n",
710 log <<
"*****************************************"
711 "**************************************\n";
719 log <<
"*** SETUP: SYMMETRY FUNCTION SCALING ****"
720 "**************************************\n";
723 log <<
"No scaling for symmetry functions.\n";
724 for (vector<Element>::iterator it =
elements.begin();
727 it->setScalingNone();
730 log <<
"*****************************************"
731 "**************************************\n";
739 log <<
"*** SETUP: SYMMETRY FUNCTION SCALING ****"
740 "**************************************\n";
743 log <<
"Equal scaling type for all symmetry functions:\n";
744 if ( (
settings.keywordExists(
"scale_symmetry_functions" ))
745 && (!
settings.keywordExists(
"center_symmetry_functions")))
749 log <<
"Gs = Smin + (Smax - Smin) * (G - Gmin) / (Gmax - Gmin)\n";
751 else if ( (!
settings.keywordExists(
"scale_symmetry_functions" ))
752 && (
settings.keywordExists(
"center_symmetry_functions")))
756 log <<
"Gs = G - Gmean\n";
758 else if ( (
settings.keywordExists(
"scale_symmetry_functions" ))
759 && (
settings.keywordExists(
"center_symmetry_functions")))
763 log <<
"Gs = Smin + (Smax - Smin) * (G - Gmean) / (Gmax - Gmin)\n";
765 else if (
settings.keywordExists(
"scale_symmetry_functions_sigma"))
769 log <<
"Gs = Smin + (Smax - Smin) * (G - Gmean) / Gsigma\n";
776 log <<
"WARNING: No symmetry function scaling!\n";
785 if (
settings.keywordExists(
"scale_min_short"))
787 Smin = atof(
settings[
"scale_min_short"].c_str());
791 log <<
"WARNING: Keyword \"scale_min_short\" not found.\n";
792 log <<
" Default value for Smin = 0.0.\n";
796 if (
settings.keywordExists(
"scale_max_short"))
798 Smax = atof(
settings[
"scale_max_short"].c_str());
802 log <<
"WARNING: Keyword \"scale_max_short\" not found.\n";
803 log <<
" Default value for Smax = 1.0.\n";
811 log <<
strpr(
"Symmetry function scaling statistics from file: %s\n",
813 log <<
"-----------------------------------------"
814 "--------------------------------------\n";
816 file.open(fileName.c_str());
819 throw runtime_error(
"ERROR: Could not open file: \"" + fileName
823 vector<string> lines;
824 while (getline(file, line))
826 if (line.at(0) !=
'#') lines.push_back(line);
831 log <<
"Abbreviations:\n";
832 log <<
"--------------\n";
833 log <<
"ind ..... Symmetry function index.\n";
834 log <<
"min ..... Minimum symmetry function value.\n";
835 log <<
"max ..... Maximum symmetry function value.\n";
836 log <<
"mean .... Mean symmetry function value.\n";
837 log <<
"sigma ... Standard deviation of symmetry function values.\n";
838 log <<
"sf ...... Scaling factor for derivatives.\n";
839 log <<
"Smin .... Desired minimum scaled symmetry function value.\n";
840 log <<
"Smax .... Desired maximum scaled symmetry function value.\n";
841 log <<
"t ....... Scaling type.\n";
843 for (vector<Element>::iterator it =
elements.begin();
847 log <<
strpr(
"Scaling data for symmetry functions element %2s :\n",
848 it->getSymbol().c_str());
849 log <<
"-----------------------------------------"
850 "--------------------------------------\n";
851 log <<
" ind min max mean sigma sf Smin Smax t\n";
852 log <<
"-----------------------------------------"
853 "--------------------------------------\n";
854 log << it->infoSymmetryFunctionScaling();
855 log <<
"-----------------------------------------"
856 "--------------------------------------\n";
857 lines.erase(lines.begin(), lines.begin() + it->numSymmetryFunctions());
860 log <<
"*****************************************"
861 "**************************************\n";
869 log <<
"*** SETUP: SYMMETRY FUNCTION GROUPS *****"
870 "**************************************\n";
873 log <<
"Abbreviations:\n";
874 log <<
"--------------\n";
875 log <<
"ind .... Symmetry function index.\n";
876 log <<
"ec ..... Central atom element.\n";
877 log <<
"tp ..... Symmetry function type.\n";
878 log <<
"sbtp ... Symmetry function subtype (e.g. cutoff type).\n";
879 log <<
"e1 ..... Neighbor 1 element.\n";
880 log <<
"e2 ..... Neighbor 2 element.\n";
881 log <<
"eta .... Gaussian width eta.\n";
882 log <<
"rs/rl... Shift distance of Gaussian or left cutoff radius "
884 log <<
"angl.... Left cutoff angle for polynomial.\n";
885 log <<
"angr.... Right cutoff angle for polynomial.\n";
886 log <<
"la ..... Angle prefactor lambda.\n";
887 log <<
"zeta ... Angle term exponent zeta.\n";
888 log <<
"rc ..... Cutoff radius / right cutoff radius for polynomial.\n";
889 log <<
"a ...... Free parameter alpha (e.g. cutoff alpha).\n";
890 log <<
"ln ..... Line number in settings file.\n";
891 log <<
"mi ..... Member index.\n";
892 log <<
"sfi .... Symmetry function index.\n";
893 log <<
"e ...... Recalculate exponential term.\n";
895 for (vector<Element>::iterator it =
elements.begin();
898 it->setupSymmetryFunctionGroups();
899 log <<
strpr(
"Short range atomic symmetry function groups "
900 "element %2s :\n", it->getSymbol().c_str());
901 log <<
"------------------------------------------------------"
902 "----------------------------------------------------\n";
903 log <<
" ind ec tp sbtp e1 e2 eta rs/rl "
904 "rc angl angr la zeta a ln mi sfi e\n";
905 log <<
"------------------------------------------------------"
906 "----------------------------------------------------\n";
907 log << it->infoSymmetryFunctionGroups();
908 log <<
"------------------------------------------------------"
909 "----------------------------------------------------\n";
912 log <<
"*****************************************"
913 "**************************************\n";
921 log <<
"*** SETUP: SYMMETRY FUNCTION MEMORY *****"
922 "**************************************\n";
927 e.setupSymmetryFunctionMemory();
928 vector<size_t> symmetryFunctionNumTable
929 = e.getSymmetryFunctionNumTable();
930 vector<vector<size_t>> symmetryFunctionTable
931 = e.getSymmetryFunctionTable();
932 log <<
strpr(
"Symmetry function derivatives memory table "
933 "for element %2s :\n", e.getSymbol().c_str());
934 log <<
"-----------------------------------------"
935 "--------------------------------------\n";
936 log <<
"Relevant symmetry functions for neighbors with element:\n";
939 log <<
strpr(
"- %2s: %4zu of %4zu (%5.1f %)\n",
941 symmetryFunctionNumTable.at(i),
942 e.numSymmetryFunctions(),
943 (100.0 * symmetryFunctionNumTable.at(i))
944 / e.numSymmetryFunctions());
947 log <<
"-----------------------------------------"
948 "--------------------------------------\n";
949 for (
auto isf : symmetryFunctionTable.at(i))
951 SymFnc const& sf = e.getSymmetryFunction(isf);
954 log <<
"-----------------------------------------"
955 "--------------------------------------\n";
958 log <<
"-----------------------------------------"
959 "--------------------------------------\n";
965 log <<
strpr(
"%2s - symmetry function per-element index table:\n",
966 e.getSymbol().c_str());
967 log <<
"-----------------------------------------"
968 "--------------------------------------\n";
975 log <<
"-----------------------------------------"
976 "--------------------------------------\n";
977 for (
size_t i = 0; i < e.numSymmetryFunctions(); ++i)
979 SymFnc const& sf = e.getSymmetryFunction(i);
984 if (ipe == numeric_limits<size_t>::max())
995 log <<
"-----------------------------------------"
996 "--------------------------------------\n";
1001 log <<
"*****************************************"
1002 "**************************************\n";
1007#ifndef N2P2_NO_SF_CACHE
1011 log <<
"*** SETUP: SYMMETRY FUNCTION CACHE ******"
1012 "**************************************\n";
1018 vector<vector<SFCacheList>> cacheLists(
numElements);
1025 size_t ne = atoi(
split(identifier)[0].c_str());
1026 bool unknown =
true;
1027 for (
auto& c : cacheLists.at(ne))
1029 if (identifier == c.identifier)
1038 cacheLists.at(ne).push_back(SFCacheList());
1039 cacheLists.at(ne).back().element = ne;
1040 cacheLists.at(ne).back().identifier = identifier;
1041 cacheLists.at(ne).back().indices.push_back(s.
getIndex());
1047 log <<
strpr(
"Multiple cache identifiers for element %2s:\n\n",
1050 double cacheUsageMean = 0.0;
1051 size_t cacheCount = 0;
1058 vector<SFCacheList>& c = cacheLists.at(j);
1059 c.erase(remove_if(c.begin(),
1063 return l.indices.size() <= 1;
1065 cacheCount += c.size();
1066 for (
size_t k = 0; k < c.size(); ++k)
1068 cacheUsageMean += c.at(k).indices.size();
1071 log <<
strpr(
"Cache %zu, Identifier \"%s\", "
1072 "Symmetry functions",
1073 k, c.at(k).identifier.c_str());
1074 for (
auto si : c.at(k).indices)
1109 cacheUsageMean /= cacheCount;
1110 log <<
strpr(
"Element %2s: in total %zu caches, "
1111 "used %3.2f times on average.\n",
1112 e.
getSymbol().c_str(), cacheCount, cacheUsageMean);
1115 log <<
"-----------------------------------------"
1116 "--------------------------------------\n";
1120 log <<
"*****************************************"
1121 "**************************************\n";
1128 bool collectExtrapolationWarnings,
1129 bool writeExtrapolationWarnings,
1130 bool stopOnExtrapolationWarnings)
1133 log <<
"*** SETUP: SYMMETRY FUNCTION STATISTICS *"
1134 "**************************************\n";
1137 log <<
"Equal symmetry function statistics for all elements.\n";
1138 log <<
strpr(
"Collect min/max/mean/sigma : %d\n",
1139 (
int)collectStatistics);
1140 log <<
strpr(
"Collect extrapolation warnings : %d\n",
1141 (
int)collectExtrapolationWarnings);
1142 log <<
strpr(
"Write extrapolation warnings immediately to stderr: %d\n",
1143 (
int)writeExtrapolationWarnings);
1144 log <<
strpr(
"Halt on any extrapolation warning : %d\n",
1145 (
int)stopOnExtrapolationWarnings);
1146 for (vector<Element>::iterator it =
elements.begin();
1149 it->statistics.collectStatistics = collectStatistics;
1150 it->statistics.collectExtrapolationWarnings =
1151 collectExtrapolationWarnings;
1152 it->statistics.writeExtrapolationWarnings = writeExtrapolationWarnings;
1153 it->statistics.stopOnExtrapolationWarnings =
1154 stopOnExtrapolationWarnings;
1158 || collectExtrapolationWarnings
1159 || writeExtrapolationWarnings
1160 || stopOnExtrapolationWarnings;
1162 log <<
"*****************************************"
1163 "**************************************\n";
1173 e.getCutoffRadii(
cutoffs.at(e.getIndex()));
1174 if ( rCutScreen > 0 &&
1177 cutoffs.at(e.getIndex()).push_back(rCutScreen);
1185 log <<
"*** SETUP: NEURAL NETWORKS **************"
1186 "**************************************\n";
1197 nns.at(
id).name =
"electronegativity";
1198 nns.at(
id).weightFileFormat =
"weightse.%03zu.data";
1199 nns.at(
id).keywordSuffix =
"_electrostatic";
1200 nns.at(
id).keywordSuffix2 =
"_charge";
1207 nns.at(
id).name =
"charge";
1208 nns.at(
id).weightFileFormat =
"weightse.%03zu.data";
1209 nns.at(
id).keywordSuffix =
"_electrostatic";
1210 nns.at(
id).keywordSuffix2 =
"_charge";
1217 nns.at(
id).name =
"short range";
1218 nns.at(
id).weightFileFormat =
"weights.%03zu.data";
1219 nns.at(
id).keywordSuffix =
"_short";
1220 nns.at(
id).keywordSuffix2 =
"_short";
1226 size_t globalNumHiddenLayers = 0;
1233 if (
settings.keywordExists(keyword))
1235 globalNumHiddenLayers = atoi(
settings[keyword].c_str());
1238 t.numLayers = globalNumHiddenLayers + 2;
1243 if (
settings.keywordExists(keyword))
1246 for (settings::Settings::KeyMap::const_iterator it = r.first;
1247 it != r.second; ++it)
1249 vector<string> args =
split(
reduce(it->second.first));
1251 size_t const n = atoi(args.at(1).c_str());
1252 nn.
topology.at(e).numLayers = n + 2;
1258 if (t.numLayers == 0)
1260 throw runtime_error(
"ERROR: Number of neural network hidden "
1261 "layers unset for some elements.\n");
1267 t.numNeuronsPerLayer.resize(t.numLayers, 0);
1268 t.activationFunctionsPerLayer.resize(t.numLayers,
1273 vector<string> globalNumNeuronsPerHiddenLayer;
1275 if (
settings.keywordExists(keyword))
1278 if (globalNumHiddenLayers != globalNumNeuronsPerHiddenLayer.size())
1280 throw runtime_error(
strpr(
"ERROR: Inconsistent global NN "
1281 "topology keyword \"%s\".\n",
1285 vector<string> globalActivationFunctions;
1287 if (
settings.keywordExists(keyword))
1290 if (globalNumHiddenLayers != globalActivationFunctions.size() - 1)
1292 throw runtime_error(
strpr(
"ERROR: Inconsistent global NN "
1293 "topology keyword \"%s\".\n",
1298 bool globalNumNeurons = (globalNumNeuronsPerHiddenLayer.size() != 0);
1299 bool globalActivation = (globalActivationFunctions.size() != 0);
1303 size_t const nsf =
elements.at(i).numSymmetryFunctions();
1328 if ((
size_t)t.
numLayers != globalNumHiddenLayers + 2)
continue;
1331 if ((j == t.
numLayers - 1) && globalActivation)
1334 globalActivationFunctions.at(t.
numLayers - 2));
1338 if (globalNumNeurons)
1341 globalNumNeuronsPerHiddenLayer.at(j - 1).c_str());
1343 if (globalActivation)
1347 globalActivationFunctions.at(j - 1));
1354 if (
settings.keywordExists(keyword))
1357 for (settings::Settings::KeyMap::const_iterator it = r.first;
1358 it != r.second; ++it)
1360 vector<string> args =
split(
reduce(it->second.first));
1362 size_t n = args.size() - 1;
1366 throw runtime_error(
strpr(
"ERROR: Inconsistent per-element"
1367 " NN topology keyword \"%s\".\n",
1370 for (
int j = 1; j < t.
numLayers - 2; ++j)
1378 if (
settings.keywordExists(keyword))
1381 for (settings::Settings::KeyMap::const_iterator it = r.first;
1382 it != r.second; ++it)
1384 vector<string> args =
split(
reduce(it->second.first));
1386 size_t n = args.size() - 1;
1390 throw runtime_error(
strpr(
"ERROR: Inconsistent per-element"
1391 " NN topology keyword \"%s\".\n",
1394 for (
int j = 1; j < t.
numLayers - 1; ++j)
1410 throw runtime_error(
strpr(
1411 "ERROR: NN \"%s\", element %2s: number of "
1412 "neurons for layer %d unset.\n",
1414 elements.at(i).getSymbol().c_str(),
1420 throw runtime_error(
strpr(
1421 "ERROR: NN \"%s\", element %2s: activation "
1422 "functions for layer %d unset.\n",
1424 elements.at(i).getSymbol().c_str(),
1432 bool normalizeNeurons =
settings.keywordExists(
"normalize_nodes");
1433 log <<
strpr(
"Normalize neurons (all elements): %d\n",
1434 (
int)normalizeNeurons);
1435 log <<
"-----------------------------------------"
1436 "--------------------------------------\n";
1446 piecewise_construct,
1447 forward_as_tuple(k),
1455 nns.at(k).name.c_str(),
1458 log <<
"-----------------------------------------"
1459 "--------------------------------------\n";
1463 log <<
"*****************************************"
1464 "**************************************\n";
1476 map<string, string> fileNameFormats)
1479 log <<
"*** SETUP: NEURAL NETWORK WEIGHTS *******"
1480 "**************************************\n";
1485 string actualFileNameFormat;
1486 if (fileNameFormats.find(k) != fileNameFormats.end())
1488 actualFileNameFormat = fileNameFormats.at(k);
1490 else actualFileNameFormat =
nns.at(k).weightFileFormat;
1491 actualFileNameFormat = directoryPrefix + actualFileNameFormat;
1492 log <<
strpr(
"%s weight file name format: %s\n",
1493 cap(
nns.at(k).name).c_str(),
1494 actualFileNameFormat.c_str());
1498 log <<
"*****************************************"
1499 "**************************************\n";
1505 bool const derivatives)
1514 #pragma omp parallel for private (a, e)
1516 for (
size_t i = 0; i < structure.
atoms.size(); ++i)
1519 a = &(structure.
atoms.at(i));
1537#ifndef N2P2_NO_SF_CACHE
1541#ifndef N2P2_NO_NEIGH_CHECK
1547 log <<
strpr(
"WARNING: Structure %6zu Atom %6zu : %zu "
1570 for (
size_t i = 0; i < structure.
atoms.size(); ++i)
1572 a = &(structure.
atoms.at(i));
1586 bool const derivatives)
1595 #pragma omp parallel for private (a, e)
1597 for (
size_t i = 0; i < structure.
atoms.size(); ++i)
1600 a = &(structure.
atoms.at(i));
1618#ifndef N2P2_NO_SF_CACHE
1622#ifndef N2P2_NO_NEIGH_CHECK
1628 log <<
strpr(
"WARNING: Structure %6zu Atom %6zu : %zu "
1651 for (
size_t i = 0; i < structure.
atoms.size(); ++i)
1653 a = &(structure.
atoms.at(i));
1667 bool const derivatives,
1670 if (
id ==
"")
id =
nnk.front();
1674 for (vector<Atom>::iterator it = structure.
atoms.begin();
1675 it != structure.
atoms.end(); ++it)
1678 .neuralNetworks.at(
id);
1692 for (
auto& a : structure.
atoms)
1695 .neuralNetworks.at(
id);
1704 for (
auto& dChidGi : a.dChidG)
1714 else if (
id ==
"short")
1716 for (
auto& a : structure.
atoms)
1719 .neuralNetworks.at(
id);
1721 for (
size_t i = 0; i < a.G.size(); ++i)
1742 for (vector<Atom>::iterator it = structure.
atoms.begin();
1743 it != structure.
atoms.end(); ++it)
1747 .neuralNetworks.at(
"elec");
1748 nnCharge.
setInput(&((it->G).front()));
1758 .neuralNetworks.at(
"short");
1760 for (
size_t i = 0; i < it->G.size(); ++i)
1765 nnShort.
setInput(it->G.size(), it->charge);
1780 bool const derivativesElec)
1798 hardness(i) =
elements.at(i).getHardness();
1799 double const iSigma =
elements.at(i).getQsigma();
1800 sigmaSqrtPi(i) = sqrt(M_PI) * iSigma;
1803 double const jSigma =
elements.at(j).getQsigma();
1804 gammaSqrt2(i, j) = sqrt(2.0 * (iSigma * iSigma + jSigma * jSigma));
1818 if (derivativesElec)
1836 for (vector<Atom>::iterator it = structure.
atoms.begin();
1837 it != structure.
atoms.end(); ++it)
1862 for (vector<Atom>::iterator it = structure.
atoms.begin();
1863 it != structure.
atoms.end(); ++it)
1867 structure.
charge += it->charge;
1881 throw runtime_error(
"WARNING: Forces are not implemented yet.\n");
1887 #pragma omp parallel
1891 for (
size_t i = 0; i < structure.
atoms.size(); ++i) {
1912 for (vector<size_t>::const_iterator it = ai.
neighborsUnique.begin() + 1;
1917#ifndef N2P2_FULL_SFD_MEMORY
1918 vector<vector<size_t>>
const& tableFull
1924 size_t const numNeighbors =
1926 for (
size_t k = 0; k < numNeighbors; ++k) {
1931#ifndef N2P2_FULL_SFD_MEMORY
1952 for (
size_t i = 0; i < s.
numAtoms; ++i)
1954 auto &ai = s.
atoms[i];
1955 ai.f -= ai.pEelecpr;
1956 ai.fElec = -ai.pEelecpr;
1958 for (
size_t j = 0; j < s.
numAtoms; ++j)
1962#ifndef N2P2_FULL_SFD_MEMORY
1963 vector<vector<size_t> >
const &tableFull
1972 ai.f -= lambdaTotal(j) * (ai.dAdrQ[j] + dChidr);
1973 ai.fElec -= lambdaElec(j) * (ai.dAdrQ[j] + dChidr);
1986 useDEdG = (useForces || useDEdG);
1999#ifdef N2P2_NO_SF_GROUPS
2006 if (useDEdG && !useForces)
2009 for (
auto& a : structure.
atoms)
2013 a.dEdG.resize(a.numSymmetryFunctions + 1, 0.0);
2014 a.dChidG.resize(a.numSymmetryFunctions, 0.0);
2017 a.dEdG.resize(a.numSymmetryFunctions, 0.0);
2041 *
elements.at(i).getAtomicEnergyOffset();
2046 *
elements.at(i).getAtomicEnergyOffset();
2060 *
elements.at(i).getAtomicEnergyOffset();
2065 *
elements.at(i).getAtomicEnergyOffset();
2074 double result = 0.0;
2079 *
elements.at(i).getAtomicEnergyOffset();
2089 else result = structure.
energy;
2094 *
elements.at(i).getAtomicEnergyOffset();
2102 if (property ==
"energy")
return value *
convEnergy;
2104 else if (property ==
"charge")
return value *
convCharge;
2105 else if (property ==
"hardness")
2108 else throw runtime_error(
"ERROR: Unknown property to convert to "
2109 "normalized units.\n");
2128 if (property ==
"energy")
return value /
convEnergy;
2130 else if (property ==
"charge")
return value /
convCharge;
2131 else if (property ==
"hardness")
2134 else throw runtime_error(
"ERROR: Unknown property to convert to physical "
2173 for (vector<Element>::iterator it =
elements.begin();
2176 it->statistics.resetExtrapolationWarnings();
2184 size_t numExtrapolationWarnings = 0;
2186 for (vector<Element>::const_iterator it =
elements.begin();
2189 numExtrapolationWarnings +=
2190 it->statistics.countExtrapolationWarnings();
2193 return numExtrapolationWarnings;
2200 for (vector<Element>::const_iterator it =
elements.begin();
2203 v.push_back(it->numSymmetryFunctions());
2211 return settings.keywordExists(keyword);
2222 ofstream file(fileName.c_str());
2223 vector<string> settingsLines =
settings.getSettingsLines();
2224 for (
size_t i = 0; i < settingsLines.size(); ++i)
2226 if (find(prune.begin(), prune.end(), i) != prune.end())
2230 file << settingsLines.at(i) <<
'\n';
2246 vector<size_t> prune;
2249 for (vector<Element>::const_iterator it =
elements.begin();
2252 for (
size_t i = 0; i < it->numSymmetryFunctions(); ++i)
2254 SymFnc const& s = it->getSymmetryFunction(i);
2257 prune.push_back(it->getSymmetryFunction(i).getLineNumber());
2267 vector<vector<double> > sensitivity)
2269 vector<size_t> prune;
2273 for (
size_t j = 0; j <
elements.at(i).numSymmetryFunctions(); ++j)
2275 if (sensitivity.at(i).at(j) < threshold)
2278 elements.at(i).getSymmetryFunction(j).getLineNumber());
2287 string const& fileNameFormat)
2289 for (vector<Element>::iterator it =
elements.begin();
2292 string fileName =
strpr(fileNameFormat.c_str(),
2293 it->getAtomicNumber());
2294 log <<
strpr(
"Setting weights for element %2s from file: %s\n",
2295 it->getSymbol().c_str(),
2298 vector<size_t>(1, 0)
CutoffType
List of available cutoff function types.
Contains element-specific data.
void calculateSymmetryFunctions(Atom &atom, bool const derivatives) const
Calculate symmetry functions.
void calculateSymmetryFunctionGroups(Atom &atom, bool const derivatives) const
Calculate symmetry functions via groups.
std::vector< std::size_t > getCacheSizes() const
Get cache sizes for each neighbor atom element.
std::size_t getIndex() const
Get index.
void setCacheIndices(std::vector< std::vector< SFCacheList > > cacheLists)
Set cache indices for all symmetry functions of this element.
std::size_t updateSymmetryFunctionStatistics(Atom const &atom)
Update symmetry function statistics.
SymFnc const & getSymmetryFunction(std::size_t index) const
Get symmetry function instance.
std::string getSymbol() const
Get symbol.
std::map< std::string, NeuralNetwork > neuralNetworks
Neural networks for this element.
std::size_t numSymmetryFunctions() const
Get number of symmetry functions.
std::vector< std::size_t > const & getSymmetryFunctionNumTable() const
Get number of relevant symmetry functions per element.
double physicalEnergy(Structure const &structure, bool ref=true) const
Undo normalization for a given energy of structure.
bool checkExtrapolationWarnings
std::vector< std::vector< double > > cutoffs
Matrix storing all symmetry function cut-offs for all elements.
void setupNormalization(bool standalone=true)
Set up normalization.
ElementMap elementMap
Global element map, populated by setupElementMap().
std::vector< double > minCutoffRadius
void addEnergyOffset(Structure &structure, bool ref=true)
Add atomic energy offsets to reference energy.
void initialize()
Write welcome message with version information.
void logEwaldCutoffs()
Logs Ewald params whenever they change.
virtual void setupElementMap()
Set up the element map.
void readNeuralNetworkWeights(std::string const &id, std::string const &fileName)
Read in weights for a specific type of neural network.
virtual void setupNeuralNetwork()
Set up neural networks for all elements.
virtual void setupSymmetryFunctionCache(bool verbose=false)
Set up symmetry function cache.
void setupGeneric(std::string const &nnpDir="", bool skipNormalize=false, bool initialHardness=false)
Combine multiple setup routines and provide a basic NNP setup.
@ HDNNP_2G
Short range NNP (2G-HDNNP).
@ HDNNP_Q
Short range NNP with charge NN, no electrostatics/Qeq (M.
@ HDNNP_4G
NNP with electrostatics and non-local charge transfer (4G-HDNNP).
std::vector< std::string > nnk
virtual void setupElectrostatics(bool initialHardness=false, std::string directoryPrefix="", std::string fileNameFormat="hardness.%03zu.data")
Set up electrostatics related stuff (hardness, screening, ...).
std::vector< std::size_t > pruneSymmetryFunctionsRange(double threshold)
Prune symmetry functions according to their range and write settings file.
virtual void setupNeuralNetworkWeights(std::map< std::string, std::string > fileNameFormats=std::map< std::string, std::string >())
Set up neural network weights from files with given name format.
ScreeningFunction screeningFunction
SymFnc::ScalingType scalingType
void calculateAtomicNeuralNetworks(Structure &structure, bool const derivatives, std::string id="")
Calculate a single atomic neural network for a given atom and nn type.
double physical(std::string const &property, double value) const
Undo normalization for a given property.
void writePrunedSettingsFile(std::vector< std::size_t > prune, std::string fileName="output.nn") const
Copy settings file but comment out lines provided.
virtual void setupSymmetryFunctionGroups()
Set up symmetry function groups.
std::vector< Element > elements
double normalized(std::string const &property, double value) const
Apply normalization to given property.
void convertToNormalizedUnits(Structure &structure) const
Convert one structure to normalized units.
void calculateEnergy(Structure &structure) const
Calculate potential energy for a given structure.
void calculateSymmetryFunctionGroups(Structure &structure, bool const derivatives)
Calculate all symmetry function groups for all atoms in given structure.
settings::Settings settings
void setupSymmetryFunctionScalingNone()
Set up "empy" symmetry function scaling.
std::string settingsGetValue(std::string const &keyword) const
Get value for given keyword in Settings instance.
virtual void setupSymmetryFunctionScaling(std::string const &fileName="scaling.data")
Set up symmetry function scaling from file.
std::vector< std::size_t > getNumSymmetryFunctions() const
Get number of symmetry functions per element.
std::size_t getNumExtrapolationWarnings() const
Count total number of extrapolation warnings encountered for all elements and symmetry functions.
void removeEnergyOffset(Structure &structure, bool ref=true)
Remove atomic energy offsets from reference energy.
bool settingsKeywordExists(std::string const &keyword) const
Check if keyword was found in settings file.
void resetExtrapolationWarnings()
Erase all extrapolation warnings and reset counters.
double getEnergyWithOffset(Structure const &structure, bool ref=true) const
Add atomic energy offsets and return energy.
std::vector< std::size_t > minNeighbors
void convertToPhysicalUnits(Structure &structure) const
Convert one structure to physical units.
void calculateCharge(Structure &structure) const
Calculate total charge for a given structure.
virtual void setupSymmetryFunctions()
Set up all symmetry functions.
void chargeEquilibration(Structure &structure, bool const derivativesElec)
Perform global charge equilibration method.
void calculateSymmetryFunctions(Structure &structure, bool const derivatives)
Calculate all symmetry functions for all atoms in given structure.
void evaluateNNP(Structure &structure, bool useForces=true, bool useDEdG=true)
Evaluate neural network potential (includes total energy, optionally forces and in some cases charges...
void setupCutoffMatrix()
Setup matrix storing all symmetry function cut-offs for each element.
std::map< std::string, NNSetup > nns
void calculateForces(Structure &structure) const
Calculate forces for all atoms in given structure.
void setupSymmetryFunctionStatistics(bool collectStatistics, bool collectExtrapolationWarnings, bool writeExtrapolationWarnings, bool stopOnExtrapolationWarnings)
Set up symmetry function statistics collection.
void setupCutoff()
Set up cutoff function for all symmetry functions.
void loadSettingsFile(std::string const &fileName="input.nn")
Open settings file and load all keywords into memory.
std::vector< std::size_t > pruneSymmetryFunctionsSensitivity(double threshold, std::vector< std::vector< double > > sensitivity)
Prune symmetry functions with sensitivity analysis data.
double getEnergyOffset(Structure const &structure) const
Get atomic energy offset for given structure.
void setupSymmetryFunctionMemory(bool verbose=false)
Extract required memory dimensions for symmetry function derivatives.
void writeSettingsFile(std::ofstream *const &file) const
Write complete settings file.
double normalizedEnergy(Structure const &structure, bool ref=true) const
Apply normalization to given energy of structure.
CutoffFunction::CutoffType cutoffType
virtual void setupElements()
Set up all Element instances.
This class implements a feed-forward neural network.
@ AF_UNSET
Unset activation function.
void setInput(double const *const &input) const
Set neural network input layer node values.
void setConnections(double const *const &connections)
Set neural network weights and biases.
void propagate()
Propagate input information through all layers.
void calculateDEdG(double *dEdG) const
Calculate derivative of output neuron with respect to input neurons.
void getOutput(double *output) const
Get neural network output layer node values.
Symmetry function base class.
virtual std::string parameterLine() const =0
Give symmetry function parameters in one line.
double getGmin() const
Get private Gmin member variable.
double getGmax() const
Get private Gmax member variable.
std::vector< std::size_t > getIndexPerElement() const
Get private indexPerElement member variable.
virtual std::vector< std::string > getCacheIdentifiers() const
Get unique cache identifiers.
std::size_t getIndex() const
Get private index member variable.
std::pair< KeyMap::const_iterator, KeyMap::const_iterator > KeyRange
@ KOLAFA_PERRAM
Method 1: Optimized in n2p2 (DOI: 10.1080/08927029208049126).
@ JACKSON_CATLOW
Method 0: Used by RuNNer (DOI: 10.1080/08927028808080944).
string cap(string word)
Capitalize first letter of word.
string strpr(const char *format,...)
String version of printf function.
bool vectorContains(std::vector< T > const &stdVec, T value)
Test if vector contains specified value.
vector< string > split(string const &input, char delimiter)
Split string at each delimiter.
map< size_t, vector< double > > readColumnsFromFile(string fileName, vector< size_t > columns, char comment)
NeuralNetwork::ActivationFunction activationFromString(std::string c)
Convert string to activation function.
@ EWALD_SUM_LAMMPS
Solver 2: Ewald summation in LAMMPS.
@ EWALD_SUM
Solver 0: Ewald summation.
string reduce(string const &line, string const &whitespace, string const &fill)
Replace multiple whitespaces with fill.
Struct to store information on neighbor atoms.
std::size_t index
Index of neighbor atom.
Storage for a single atom.
std::vector< Neighbor > neighbors
Neighbor array (maximum number defined in macros.h.
std::size_t numSymmetryFunctions
Number of symmetry functions used to describe the atom environment.
Vec3D calculateDChidr(size_t const atomIndexOfR, double const maxCutoffRadius, std::vector< std::vector< size_t > > const *const tableFull=nullptr) const
Calculate dChi/dr of this atom's Chi with respect to the coordinates of the given atom.
Vec3D f
Force vector calculated by neural network.
Vec3D calculatePairForceShort(Neighbor const &neighbor, std::vector< std::vector< std::size_t > > const *const tableFull=nullptr) const
Calculate force resulting from gradient of this atom's (short-ranged) energy contribution with respec...
bool hasSymmetryFunctionDerivatives
If symmetry function derivatives are saved for this atom.
std::size_t index
Index number of this atom.
Vec3D calculateSelfForceShort() const
Calculate force resulting from gradient of this atom's (short-ranged) energy contribution with respec...
std::size_t getStoredMinNumNeighbors(double const cutoffRadius) const
Return needed number of neighbors for a given cutoff radius from neighborCutoffs map.
std::vector< std::size_t > numSymmetryFunctionDerivatives
Number of neighbor atom symmetry function derivatives per element.
bool useChargeNeuron
If an additional charge neuron in the short-range NN is present.
std::size_t indexStructure
Index number of structure this atom belongs to.
std::size_t element
Element index of this atom.
void allocate(bool all, double const maxCutoffRadius=0.0)
Allocate vectors related to symmetry functions (G, dEdG).
bool hasSymmetryFunctions
If symmetry function values are saved for this atom.
std::size_t calculateNumNeighbors(double const cutoffRadius) const
Calculate number of neighbors for a given cutoff radius.
std::vector< std::size_t > cacheSizePerElement
Cache size for each element.
std::vector< std::size_t > neighborsUnique
List of unique neighbor indices (don't count multiple PBC images).
List of symmetry functions corresponding to one cache identifier.
std::vector< NeuralNetwork::ActivationFunction > activationFunctionsPerLayer
Activation function type per layer.
std::vector< int > numNeuronsPerLayer
Number of neurons per layer.
int numLayers
Number of NN layers (including input and output layer).
Setup data for one neural network.
std::string keywordSuffix
Suffix for keywords (NN topology related).
std::vector< Topology > topology
Per-element NN topology.
std::string id
NN identifier, e.g. "short", "charge",...
Storage for one atomic configuration.
void toPhysicalUnits(double meanEnergy, double convEnergy, double convLength, double convCharge)
Switch to physical units, shift energy and change energy, length and charge unit.
Eigen::VectorXd const calculateForceLambdaElec() const
Calculate lambda_elec vector which is needed for the electrostatic force calculation in 4G NN.
std::vector< std::size_t > numAtomsPerElement
Number of atoms of each element in this structure.
void calculateMaxCutoffRadiusOverall(EwaldSetup &ewaldSetup, double rcutScreen, double maxCutoffRadius)
Calculate maximal cut-off if cut-off of screening and real part Ewald summation are also considered.
double charge
Charge determined by neural network potential.
double calculateElectrostaticEnergy(EwaldSetup &ewaldSetup, Eigen::VectorXd hardness, Eigen::MatrixXd gammaSqrt2, Eigen::VectorXd sigmaSqrtPi, ScreeningFunction const &fs, double const fourPiEps, ErfcBuf &erfcBuf)
Compute electrostatic energy with global charge equilibration.
double energyShort
Short-range part of the potential energy predicted by NNP.
bool hasSymmetryFunctionDerivatives
If symmetry function derivatives are saved for each atom.
Eigen::VectorXd const calculateForceLambdaTotal() const
Calculate lambda_total vector which is needed for the total force calculation in 4G NN.
double energyElec
Electrostatics part of the potential energy predicted by NNP.
double energy
Potential energy determined by neural network.
double energyRef
Reference potential energy.
void toNormalizedUnits(double meanEnergy, double convEnergy, double convLength, double convCharge)
Normalize structure, shift energy and change energy, length and charge unit.
void calculateElectrostaticEnergyDerivatives(Eigen::VectorXd hardness, Eigen::MatrixXd gammaSqrt2, Eigen::VectorXd sigmaSqrtPi, ScreeningFunction const &fs, double const fourPiEps)
Calculates partial derivatives of electrostatic Energy with respect to atom's coordinates and charges...
void calculateNeighborList(double cutoffRadius, bool sortByDistance=false)
Calculate neighbor list for all atoms.
std::size_t numAtoms
Total number of atoms present in this structure.
void calculateDAdrQ(EwaldSetup &ewaldSetup, Eigen::MatrixXd gammaSqrt2, double const fourPiEps, ErfcBuf &erfcBuf)
Calculates derivative of A-matrix with respect to the atoms positions and contract it with Q.
std::vector< Atom > atoms
Vector of all atoms in this structure.
bool hasSymmetryFunctions
If symmetry function values are saved for each atom.
Vector in 3 dimensional real space.