source: src/Actions/FragmentationAction/FragmentationAutomationAction.cpp@ 27594e

Action_Thermostats Add_AtomRandomPerturbation Add_FitFragmentPartialChargesAction Add_RotateAroundBondAction Add_SelectAtomByNameAction Added_ParseSaveFragmentResults AddingActions_SaveParseParticleParameters Adding_Graph_to_ChangeBondActions Adding_MD_integration_tests Adding_ParticleName_to_Atom Adding_StructOpt_integration_tests AtomFragments Automaking_mpqc_open AutomationFragmentation_failures Candidate_v1.5.4 Candidate_v1.6.0 Candidate_v1.6.1 ChangeBugEmailaddress ChangingTestPorts ChemicalSpaceEvaluator CombiningParticlePotentialParsing Combining_Subpackages Debian_Package_split Debian_package_split_molecuildergui_only Disabling_MemDebug Docu_Python_wait EmpiricalPotential_contain_HomologyGraph EmpiricalPotential_contain_HomologyGraph_documentation Enable_parallel_make_install Enhance_userguide Enhanced_StructuralOptimization Enhanced_StructuralOptimization_continued Example_ManyWaysToTranslateAtom Exclude_Hydrogens_annealWithBondGraph FitPartialCharges_GlobalError Fix_BoundInBox_CenterInBox_MoleculeActions Fix_ChargeSampling_PBC Fix_ChronosMutex Fix_FitPartialCharges Fix_FitPotential_needs_atomicnumbers Fix_ForceAnnealing Fix_IndependentFragmentGrids Fix_ParseParticles Fix_ParseParticles_split_forward_backward_Actions Fix_PopActions Fix_QtFragmentList_sorted_selection Fix_Restrictedkeyset_FragmentMolecule Fix_StatusMsg Fix_StepWorldTime_single_argument Fix_Verbose_Codepatterns Fix_fitting_potentials Fixes ForceAnnealing_goodresults ForceAnnealing_oldresults ForceAnnealing_tocheck ForceAnnealing_with_BondGraph ForceAnnealing_with_BondGraph_continued ForceAnnealing_with_BondGraph_continued_betteresults ForceAnnealing_with_BondGraph_contraction-expansion FragmentAction_writes_AtomFragments FragmentMolecule_checks_bonddegrees GeometryObjects Gui_Fixes Gui_displays_atomic_force_velocity ImplicitCharges IndependentFragmentGrids IndependentFragmentGrids_IndividualZeroInstances IndependentFragmentGrids_IntegrationTest IndependentFragmentGrids_Sole_NN_Calculation JobMarket_RobustOnKillsSegFaults JobMarket_StableWorkerPool JobMarket_unresolvable_hostname_fix MoreRobust_FragmentAutomation ODR_violation_mpqc_open PartialCharges_OrthogonalSummation PdbParser_setsAtomName PythonUI_with_named_parameters QtGui_reactivate_TimeChanged_changes Recreated_GuiChecks Rewrite_FitPartialCharges RotateToPrincipalAxisSystem_UndoRedo SaturateAtoms_findBestMatching SaturateAtoms_singleDegree StoppableMakroAction Subpackage_CodePatterns Subpackage_JobMarket Subpackage_LinearAlgebra Subpackage_levmar Subpackage_mpqc_open Subpackage_vmg Switchable_LogView ThirdParty_MPQC_rebuilt_buildsystem TrajectoryDependenant_MaxOrder TremoloParser_IncreasedPrecision TremoloParser_MultipleTimesteps TremoloParser_setsAtomName Ubuntu_1604_changes stable
Last change on this file since 27594e was 27594e, checked in by Frederik Heber <heber@…>, 12 years ago

Added FragmentationChargeDensity to contain summed up charge density.

  • Property mode set to 100644
File size: 17.1 KB
Line 
1/*
2 * Project: MoleCuilder
3 * Description: creates and alters molecular systems
4 * Copyright (C) 2010-2012 University of Bonn. All rights reserved.
5 *
6 *
7 * This file is part of MoleCuilder.
8 *
9 * MoleCuilder is free software: you can redistribute it and/or modify
10 * it under the terms of the GNU General Public License as published by
11 * the Free Software Foundation, either version 2 of the License, or
12 * (at your option) any later version.
13 *
14 * MoleCuilder is distributed in the hope that it will be useful,
15 * but WITHOUT ANY WARRANTY; without even the implied warranty of
16 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
17 * GNU General Public License for more details.
18 *
19 * You should have received a copy of the GNU General Public License
20 * along with MoleCuilder. If not, see <http://www.gnu.org/licenses/>.
21 */
22
23/*
24 * FragmentationAutomationAction.cpp
25 *
26 * Created on: May 18, 2012
27 * Author: heber
28 */
29
30// include config.h
31#ifdef HAVE_CONFIG_H
32#include <config.h>
33#endif
34
35#include <boost/archive/text_iarchive.hpp>
36// boost asio needs specific operator new
37#include <boost/asio.hpp>
38
39#include "CodePatterns/MemDebug.hpp"
40
41#include <boost/mpl/remove.hpp>
42#include <boost/lambda/lambda.hpp>
43
44#include "CodePatterns/Assert.hpp"
45#include "CodePatterns/Info.hpp"
46#include "CodePatterns/Log.hpp"
47#include "JobMarket/Jobs/FragmentJob.hpp"
48
49#include "Fragmentation/Automation/createMatrixNrLookup.hpp"
50#include "Fragmentation/Automation/extractJobIds.hpp"
51#include "Fragmentation/Automation/FragmentationChargeDensity.hpp"
52#include "Fragmentation/Automation/FragmentationResults.hpp"
53#include "Fragmentation/Automation/MPQCFragmentController.hpp"
54#include "Fragmentation/Automation/VMGDebugGridFragmentController.hpp"
55#include "Fragmentation/Automation/VMGFragmentController.hpp"
56#include "Fragmentation/EnergyMatrix.hpp"
57#include "Fragmentation/ForceMatrix.hpp"
58#include "Fragmentation/Fragmentation.hpp"
59#include "Fragmentation/SetValues/Fragment.hpp"
60#include "Fragmentation/SetValues/Histogram.hpp"
61#include "Fragmentation/SetValues/IndexedVectors.hpp"
62#include "Fragmentation/HydrogenSaturation_enum.hpp"
63#include "Fragmentation/KeySet.hpp"
64#include "Fragmentation/KeySetsContainer.hpp"
65#include "Fragmentation/Summation/writeTable.hpp"
66#include "Graph/DepthFirstSearchAnalysis.hpp"
67#include "Helpers/defs.hpp"
68#include "Jobs/MPQCJob.hpp"
69#include "Jobs/MPQCData.hpp"
70#include "Jobs/MPQCData_printKeyNames.hpp"
71#ifdef HAVE_VMG
72#include "Jobs/VMGDebugGridJob.hpp"
73#include "Jobs/VMGJob.hpp"
74#include "Jobs/VMGData.hpp"
75#include "Jobs/VMGDataFused.hpp"
76#include "Jobs/VMGDataMap.hpp"
77#include "Jobs/VMGData_printKeyNames.hpp"
78#endif
79#include "World.hpp"
80
81#include <fstream>
82#include <iostream>
83#include <string>
84#include <vector>
85
86#include <boost/mpl/for_each.hpp>
87
88#include "Actions/FragmentationAction/FragmentationAutomationAction.hpp"
89
90using namespace MoleCuilder;
91
92// and construct the stuff
93#include "FragmentationAutomationAction.def"
94#include "Action_impl_pre.hpp"
95/** =========== define the function ====================== */
96
97class controller_AddOn;
98
99// needs to be defined for using the FragmentController
100controller_AddOn *getAddOn()
101{
102 return NULL;
103}
104
105/** Helper function to get number of atoms somehow.
106 *
107 * Here, we just parse the number of lines in the adjacency file as
108 * it should correspond to the number of atoms, except when some atoms
109 * are not bonded, but then fragmentation makes no sense.
110 *
111 * @param path path to the adjacency file
112 */
113size_t getNoAtomsFromAdjacencyFile(const std::string &path)
114{
115 size_t NoAtoms = 0;
116
117 // parse in special file to get atom count (from line count)
118 std::string filename(path);
119 filename += FRAGMENTPREFIX;
120 filename += ADJACENCYFILE;
121 std::ifstream adjacency(filename.c_str());
122 if (adjacency.fail()) {
123 LOG(0, endl << "getNoAtomsFromAdjacencyFile() - Unable to open " << filename << ", is the directory correct?");
124 return false;
125 }
126 std::string buffer;
127 while (getline(adjacency, buffer))
128 NoAtoms++;
129 LOG(1, "INFO: There are " << NoAtoms << " atoms.");
130
131 return NoAtoms;
132}
133
134
135
136/** Place results from FragmentResult into EnergyMatrix and ForceMatrix.
137 *
138 * @param fragmentData MPQCData resulting from the jobs
139 * @param MatrixNrLookup Lookup up-map from job id to fragment number
140 * @param FragmentCounter total number of fragments
141 * @param NoAtoms total number of atoms
142 * @param Energy energy matrix to be filled on return
143 * @param Force force matrix to be filled on return
144 * @return true - everything ok, false - else
145 */
146bool putResultsintoMatrices(
147 const std::map<JobId_t, MPQCData> &fragmentData,
148 const std::map< JobId_t, size_t > &MatrixNrLookup,
149 const size_t FragmentCounter,
150 const size_t NoAtoms,
151 EnergyMatrix &Energy,
152 ForceMatrix &Force)
153{
154 for (std::map<JobId_t, MPQCData>::const_iterator dataiter = fragmentData.begin();
155 dataiter != fragmentData.end(); ++dataiter) {
156 const MPQCData &extractedData = dataiter->second;
157 const JobId_t &jobid = dataiter->first;
158 std::map< JobId_t, size_t >::const_iterator nriter = MatrixNrLookup.find(jobid);
159 ASSERT( nriter != MatrixNrLookup.end(),
160 "putResultsintoMatrices() - MatrixNrLookup does not contain id "
161 +toString(jobid)+".");
162 // place results into EnergyMatrix ...
163 {
164 MatrixContainer::MatrixArray matrix;
165 matrix.resize(1);
166 matrix[0].resize(1, extractedData.energies.total);
167 if (!Energy.AddMatrix(
168 std::string("MPQCJob ")+toString(jobid),
169 matrix,
170 nriter->second)) {
171 ELOG(1, "Adding energy matrix failed.");
172 return false;
173 }
174 }
175 // ... and ForceMatrix (with two empty columns in front)
176 {
177 MatrixContainer::MatrixArray matrix;
178 const size_t rows = extractedData.forces.size();
179 matrix.resize(rows);
180 for (size_t i=0;i<rows;++i) {
181 const size_t columns = 2+extractedData.forces[i].size();
182 matrix[i].resize(columns, 0.);
183 // for (size_t j=0;j<2;++j)
184 // matrix[i][j] = 0.;
185 for (size_t j=2;j<columns;++j)
186 matrix[i][j] = extractedData.forces[i][j-2];
187 }
188 if (!Force.AddMatrix(
189 std::string("MPQCJob ")+toString(jobid),
190 matrix,
191 nriter->second)) {
192 ELOG(1, "Adding force matrix failed.");
193 return false;
194 }
195 }
196 }
197 // add one more matrix (not required for energy)
198 MatrixContainer::MatrixArray matrix;
199 matrix.resize(1);
200 matrix[0].resize(1, 0.);
201 if (!Energy.AddMatrix(std::string("MPQCJob total"), matrix, FragmentCounter))
202 return false;
203 // but for energy because we need to know total number of atoms
204 matrix.resize(NoAtoms);
205 for (size_t i = 0; i< NoAtoms; ++i)
206 matrix[i].resize(2+NDIM, 0.);
207 if (!Force.AddMatrix(std::string("MPQCJob total"), matrix, FragmentCounter))
208 return false;
209
210 return true;
211}
212
213/** Print MPQCData from received results.
214 *
215 * @param fragmentData MPQCData resulting from the jobs, associated to job id
216 * @param KeySetFilename filename with keysets to associate forces correctly
217 * @param NoAtoms total number of atoms
218 * @param full_sample summed up charge from fragments on return
219 */
220bool printReceivedMPQCResults(
221 const std::map<JobId_t, MPQCData> &fragmentData,
222 const std::string &KeySetFilename,
223 size_t NoAtoms)
224{
225 // create a vector of all job ids
226 std::vector<JobId_t> jobids;
227 std::transform(fragmentData.begin(),fragmentData.end(),
228 std::back_inserter(jobids),
229 boost::bind( &std::map<JobId_t,MPQCData>::value_type::first, boost::lambda::_1 )
230 );
231
232 // create lookup from job nr to fragment number
233 size_t FragmentCounter = 0;
234 const std::map< JobId_t, size_t > MatrixNrLookup=
235 createMatrixNrLookup(jobids, FragmentCounter);
236
237 // place results into maps
238 EnergyMatrix Energy;
239 ForceMatrix Force;
240 if (!putResultsintoMatrices(fragmentData, MatrixNrLookup, FragmentCounter, NoAtoms, Energy, Force))
241 return false;
242
243 // initialise keysets
244 KeySetsContainer KeySet;
245 KeySetsContainer ForceKeySet;
246 if (!Energy.InitialiseIndices()) return false;
247
248 if (!Force.ParseIndices(KeySetFilename.c_str())) return false;
249
250 {
251 // else needs keysets without hydrogens
252 std::stringstream filename;
253 filename << FRAGMENTPREFIX << KEYSETFILE;
254 if (!KeySet.ParseKeySets(KeySetFilename, filename.str(), FragmentCounter)) return false;
255 }
256
257 {
258 // forces need keysets including hydrogens
259 std::stringstream filename;
260 filename << FRAGMENTPREFIX << FORCESFILE;
261 if (!ForceKeySet.ParseKeySets(KeySetFilename, filename.str(), FragmentCounter)) return false;
262 }
263
264 // combine all found data
265 if (!KeySet.ParseManyBodyTerms()) return false;
266
267 EnergyMatrix EnergyFragments;
268 ForceMatrix ForceFragments;
269 if (!EnergyFragments.AllocateMatrix(Energy.Header, Energy.MatrixCounter, Energy.RowCounter, Energy.ColumnCounter)) return false;
270 if (!ForceFragments.AllocateMatrix(Force.Header, Force.MatrixCounter, Force.RowCounter, Force.ColumnCounter)) return false;
271
272 if(!Energy.SetLastMatrix(0., 0)) return false;
273 if(!Force.SetLastMatrix(0., 2)) return false;
274
275 for (int BondOrder=0;BondOrder<KeySet.Order;BondOrder++) {
276 // --------- sum up energy --------------------
277 LOG(1, "INFO: Summing energy of order " << BondOrder+1 << " ...");
278 if (!EnergyFragments.SumSubManyBodyTerms(Energy, KeySet, BondOrder)) return false;
279 if (!Energy.SumSubEnergy(EnergyFragments, NULL, KeySet, BondOrder, 1.)) return false;
280
281 // --------- sum up Forces --------------------
282 LOG(1, "INFO: Summing forces of order " << BondOrder+1 << " ...");
283 if (!ForceFragments.SumSubManyBodyTerms(Force, KeySet, BondOrder)) return false;
284 if (!Force.SumSubForces(ForceFragments, KeySet, BondOrder, 1.)) return false;
285 }
286
287 // for debugging print resulting energy and forces
288 LOG(1, "INFO: Resulting energy is " << Energy.Matrix[ FragmentCounter ][0][0]);
289 std::stringstream output;
290 for (int i=0; i< Force.RowCounter[FragmentCounter]; ++i) {
291 for (int j=0; j< Force.ColumnCounter[FragmentCounter]; ++j)
292 output << Force.Matrix[ FragmentCounter ][i][j] << " ";
293 output << "\n";
294 }
295 LOG(1, "INFO: Resulting forces are " << std::endl << output.str());
296
297 return true;
298}
299
300void writeToFile(const std::string &filename, const std::string contents)
301{
302 std::ofstream tablefile(filename.c_str());
303 tablefile << contents;
304 tablefile.close();
305}
306
307/** Print MPQCData from received results.
308 *
309 * @param results summed up results container
310 */
311void printReceivedFullResults(
312 const FragmentationResults &results)
313{
314 // print tables (without eigenvalues, they go extra)
315 {
316 typedef boost::mpl::remove<MPQCDataEnergyVector_t, MPQCDataFused::energy_eigenvalues>::type
317 MPQCDataEnergyVector_noeigenvalues_t;
318 const std::string energyresult =
319 writeTable<MPQCDataEnergyMap_t, MPQCDataEnergyVector_noeigenvalues_t >()(
320 results.Result_Energy_fused, results.getMaxLevel());
321 LOG(0, "Energy table is \n" << energyresult);
322 std::string filename;
323 filename += FRAGMENTPREFIX + std::string("_Energy.dat");
324 writeToFile(filename, energyresult);
325 }
326
327 {
328 const std::string gridresult =
329 writeTable<VMGDataLongRangeMap_t, VMGDataLongRangeVector_t >()(
330 results.Result_LongRangeIntegrated_fused, results.getMaxLevel());
331 LOG(0, "LongRange table is \n" << gridresult);
332 std::string filename;
333 filename += FRAGMENTPREFIX + std::string("_LongRangeEnergy.dat");
334 writeToFile(filename, gridresult);
335 }
336
337 {
338 const std::string eigenvalueresult;
339 LOG(0, "Eigenvalue table is \n" << eigenvalueresult);
340 std::string filename;
341 filename += FRAGMENTPREFIX + std::string("_Eigenvalues.dat");
342 writeToFile(filename, eigenvalueresult);
343 }
344
345 {
346 const std::string forceresult =
347 writeTable<MPQCDataForceMap_t, MPQCDataForceVector_t>()(
348 results.Result_Force_fused, results.getMaxLevel());
349 LOG(0, "Force table is \n" << forceresult);
350 std::string filename;
351 filename += FRAGMENTPREFIX + std::string("_Forces.dat");
352 writeToFile(filename, forceresult);
353 }
354 // we don't want to print grid to a table
355 {
356 // print times (without flops for now)
357 typedef boost::mpl::remove<
358 boost::mpl::remove<MPQCDataTimeVector_t, MPQCDataFused::times_total_flops>::type,
359 MPQCDataFused::times_gather_flops>::type
360 MPQCDataTimeVector_noflops_t;
361 const std::string timesresult =
362 writeTable<MPQCDataTimeMap_t, MPQCDataTimeVector_noflops_t >()(
363 results.Result_Time_fused, results.getMaxLevel());
364 LOG(0, "Times table is \n" << timesresult);
365 std::string filename;
366 filename += FRAGMENTPREFIX + std::string("_Times.dat");
367 writeToFile(filename, timesresult);
368 }
369}
370
371
372Action::state_ptr FragmentationFragmentationAutomationAction::performCall() {
373 boost::asio::io_service io_service;
374 MPQCFragmentController mpqccontroller(io_service);
375 mpqccontroller.setHost(params.host.get());
376 mpqccontroller.setPort(params.port.get());
377 mpqccontroller.setLevel(params.level.get());
378 VMGFragmentController vmgcontroller(io_service);
379 vmgcontroller.setHost(params.host.get());
380 vmgcontroller.setPort(params.port.get());
381 VMGDebugGridFragmentController debugcontroller(io_service);
382 debugcontroller.setHost(params.host.get());
383 debugcontroller.setPort(params.port.get());
384
385 // TODO: Have io_service run in second thread and merge with current again eventually
386
387 // Phase One: obtain ids
388 std::vector< boost::filesystem::path > jobfiles = params.jobfiles.get();
389 mpqccontroller.requestIds(jobfiles.size());
390
391 // Phase Two: create and add MPQCJobs
392 if (!mpqccontroller.addJobsFromFiles(params.executable.get().string(), jobfiles))
393 return Action::failure;
394
395 // Phase Three: calculate result
396 mpqccontroller.waitforResults(jobfiles.size());
397 std::map<JobId_t, MPQCData> fragmentData;
398 mpqccontroller.getResults(fragmentData);
399
400#ifdef HAVE_VMG
401 if (params.DoLongrange.get()) {
402 ASSERT( World::getInstance().getAllAtoms().size() != 0,
403 "FragmentationFragmentationAutomationAction::performCall() - please load the full molecule into the World before.");
404
405 // obtain combined charge density
406 LOG(1, "INFO: Parsing fragment files from " << params.path.get() << ".");
407 FragmentationChargeDensity summedChargeDensity(
408 fragmentData,
409 params.path.get());
410 const std::vector<SamplingGrid> full_sample = summedChargeDensity.getFullSampledGrid();
411
412 // Phase Four: obtain more ids
413 const size_t NoJobs = fragmentData.size()+full_sample.size();
414 vmgcontroller.requestIds(NoJobs);
415
416 // Phase Five: create VMGJobs
417 const size_t near_field_cells = params.near_field_cells.get();
418 if (!vmgcontroller.createLongRangeJobs(
419 fragmentData,
420 full_sample,
421 summedChargeDensity.getFragment(), near_field_cells))
422 return Action::failure;
423
424 // Phase Six: calculate result
425 vmgcontroller.waitforResults(NoJobs);
426 std::map<JobId_t, VMGData> longrangeData;
427 vmgcontroller.getResults(longrangeData);
428 ASSERT( NoJobs == longrangeData.size(),
429 "FragmentationFragmentationAutomationAction::performCall() - number of MPQCresults+"
430 +toString(full_sample)+" "
431 +toString(NoJobs)
432 +" and VMGresults "+toString(longrangeData.size())+" don't match.");
433
434 // remove full solution corresponding to full_sample from map (must be highest ids), has to be treated extra
435 std::map<JobId_t, VMGData>::iterator iter = longrangeData.end();
436 for (size_t i=0;i<full_sample.size();++i)
437 --iter;
438 std::map<JobId_t, VMGData>::iterator remove_iter = iter;
439 std::vector<VMGData> fullsolutionData;
440 for (; iter != longrangeData.end(); ++iter)
441 fullsolutionData.push_back(iter->second);
442 longrangeData.erase(remove_iter, longrangeData.end());
443
444 // Final phase: sum up and print result
445 {
446 FragmentationResults results(
447 fragmentData,
448 longrangeData,
449 fullsolutionData,
450 params.path.get(),
451 getNoAtomsFromAdjacencyFile(params.path.get()),
452 full_sample);
453
454 LOG(1, "INFO: Parsing fragment files from " << params.path.get() << ".");
455 printReceivedFullResults(results);
456
457 if (!full_sample.empty()) {
458 // create debug jobs for each level to print the summed-up potential to vtk files
459 debugcontroller.requestIds(full_sample.size());
460 if (!debugcontroller.createDebugJobs(full_sample))
461 return Action::failure;
462 debugcontroller.waitforResults(full_sample.size());
463 std::map<JobId_t, std::string> debugData;
464 debugcontroller.getResults(debugData);
465 }
466 }
467 }
468#else
469 // Final phase: print result
470 {
471 LOG(1, "INFO: Parsing fragment files from " << params.path.get() << ".");
472 printReceivedMPQCResults(
473 fragmentData,
474 params.path.get(),
475 getNoAtomsFromAdjacencyFile(params.path.get()));
476 }
477#endif
478 size_t Exitflag = vmgcontroller.getExitflag() + mpqccontroller.getExitflag();
479
480 return (Exitflag == 0) ? Action::success : Action::failure;
481}
482
483Action::state_ptr FragmentationFragmentationAutomationAction::performUndo(Action::state_ptr _state) {
484 return Action::success;
485}
486
487Action::state_ptr FragmentationFragmentationAutomationAction::performRedo(Action::state_ptr _state){
488 return Action::success;
489}
490
491bool FragmentationFragmentationAutomationAction::canUndo() {
492 return false;
493}
494
495bool FragmentationFragmentationAutomationAction::shouldUndo() {
496 return false;
497}
498/** =========== end of function ====================== */
Note: See TracBrowser for help on using the repository browser.