source: src/Potentials/PartialNucleiChargeFitter.cpp@ 96c7d2

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 96c7d2 was f60d95, checked in by Frederik Heber <heber@…>, 11 years ago

PartialNucleiChargeFitter writes potential, solution, and residuum to CSV files in DEBUG.

  • Property mode set to 100644
File size: 13.1 KB
Line 
1/*
2 * Project: MoleCuilder
3 * Description: creates and alters molecular systems
4 * Copyright (C) 2013 Frederik Heber. All rights reserved.
5 * Please see the LICENSE file or "Copyright notice" in builder.cpp for details.
6 *
7 *
8 * This file is part of MoleCuilder.
9 *
10 * MoleCuilder is free software: you can redistribute it and/or modify
11 * it under the terms of the GNU General Public License as published by
12 * the Free Software Foundation, either version 2 of the License, or
13 * (at your option) any later version.
14 *
15 * MoleCuilder is distributed in the hope that it will be useful,
16 * but WITHOUT ANY WARRANTY; without even the implied warranty of
17 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
18 * GNU General Public License for more details.
19 *
20 * You should have received a copy of the GNU General Public License
21 * along with MoleCuilder. If not, see <http://www.gnu.org/licenses/>.
22 */
23
24/*
25 * PartialNucleiChargeFitter.cpp
26 *
27 * Created on: 12.05.2013
28 * Author: heber
29 */
30
31// include config.h
32#ifdef HAVE_CONFIG_H
33#include <config.h>
34#endif
35
36#include "CodePatterns/MemDebug.hpp"
37
38#include "PartialNucleiChargeFitter.hpp"
39
40#include <cmath>
41#include <fstream>
42#include <limits>
43#include <numeric>
44
45#include "LinearAlgebra/MatrixContent.hpp"
46#include "LinearAlgebra/VectorContent.hpp"
47
48#include "CodePatterns/Assert.hpp"
49#include "CodePatterns/Log.hpp"
50
51#include "Fragmentation/Summation/SetValues/SamplingGrid.hpp"
52
53PartialNucleiChargeFitter::dimensions_t
54PartialNucleiChargeFitter::getGridDimensions(const SamplingGrid &grid) const
55{
56 // convert sampled potential into a vector
57 const double round_offset =
58 (std::numeric_limits<size_t>::round_style == std::round_toward_zero) ?
59 0.5 : 0.; // need offset to get to round_toward_nearest behavior
60 dimensions_t total(3,0);
61 for(size_t index=0;index<3;++index) {
62 const double delta = grid.getDeltaPerAxis(index);
63 // delta is conversion factor from box length to discrete length, i.e. number of points
64 total[index] = (grid.end[index] - grid.begin[index])/delta+round_offset;
65 }
66 return total;
67}
68
69PartialNucleiChargeFitter::PartialNucleiChargeFitter(
70 const SamplingGrid &grid,
71 const positions_t &_positions,
72 const double _threshold) :
73 total(getGridDimensions(grid)),
74 SampledPotential(std::accumulate(total.begin(), total.end(), 1, std::multiplies<double>())),
75 grid_properties(static_cast<const SamplingGridProperties &>(grid)),
76 positions(_positions),
77 PotentialFromCharges(NULL),
78 PartialCharges(NULL),
79 threshold(_threshold)
80{
81 // we must take care of the "window", i.e. there may be less entries in sampled_grid
82 // vector as we would expect from size of grid ((2^level)^3) as 0-entries have been
83 // omitted.
84 size_t pre_offset[3];
85 size_t post_offset[3];
86 size_t length[3];
87 size_t total[3];
88 grid.getDiscreteWindowIndices(
89 grid.begin, grid.end,
90 grid.begin_window, grid.end_window,
91 pre_offset,
92 post_offset,
93 length,
94 total
95 );
96 const size_t calculated_size = length[0]*length[1]*length[2];
97 ASSERT( calculated_size == grid.sampled_grid.size(),
98 "PartialNucleiChargeFitter::PartialNucleiChargeFitter() - grid does not match size indicated by its window.");
99
100 const double potential_sum = std::accumulate(grid.sampled_grid.begin(), grid.sampled_grid.end(), 0.);
101 if ( fabs(potential_sum) > std::numeric_limits<double>::epsilon()*1e4 ) {
102 ELOG(2, "Potential sum is not less then "
103 << std::numeric_limits<double>::epsilon()*1e4 << " but "
104 << potential_sum << ".");
105 }
106
107 SamplingGrid::sampledvalues_t::const_iterator griditer = grid.sampled_grid.begin();
108 size_t index=0;
109 size_t N[3];
110 Vector grid_position; // position of grid point in real domain
111 size_t masked_points = 0;
112 // store step length per axis
113 double delta[3];
114 for (size_t i=0;i<3;++i)
115 delta[i] = grid_properties.getDeltaPerAxis(i);
116 /// convert sampled potential into a vector
117 grid_position[0] = grid_properties.begin[0];
118 for(N[0]=0; N[0] < pre_offset[0]; ++N[0]) {
119 grid_position[1] = grid_properties.begin[1];
120 for(N[1]=0; N[1] < total[1]; ++N[1]) {
121 grid_position[2] = grid_properties.begin[2];
122 for(N[2]=0; N[2] < total[2]; ++N[2]) {
123 const_cast<VectorContent &>(SampledPotential)[index++] = 0.;
124 grid_position[2] += delta[2];
125 }
126 grid_position[1] += delta[1];
127 }
128 grid_position[0] += delta[0];
129 }
130 for(N[0]=0; N[0] < length[0]; ++N[0]) {
131 grid_position[1] = grid_properties.begin[1];
132 for(N[1]=0; N[1] < pre_offset[1]; ++N[1]) {
133 grid_position[2] = grid_properties.begin[2];
134 for(N[2]=0; N[2] < total[2]; ++N[2]) {
135 const_cast<VectorContent &>(SampledPotential)[index++] = 0.;
136 grid_position[2] += delta[2];
137 }
138 grid_position[1] += delta[1];
139 }
140 for(N[1]=0; N[1] < length[1]; ++N[1]) {
141 grid_position[2] = grid_properties.begin[2];
142 for(N[2]=0; N[2] < pre_offset[2]; ++N[2]) {
143 const_cast<VectorContent &>(SampledPotential)[index++] = 0.;
144 grid_position[2] += delta[2];
145 }
146 for(N[2]=0; N[2] < length[2]; ++N[2]) {
147 if (isGridPointSettable(positions, grid_position))
148 const_cast<VectorContent &>(SampledPotential)[index++] = *griditer++;
149 else {
150 // skip point
151 ++griditer;
152 ++masked_points;
153 const_cast<VectorContent &>(SampledPotential)[index++] = 0.;
154 }
155 grid_position[2] += delta[2];
156 }
157 for(N[2]=0; N[2] < post_offset[2]; ++N[2]) {
158 const_cast<VectorContent &>(SampledPotential)[index++] = 0.;
159 grid_position[2] += delta[2];
160 }
161 grid_position[1] += delta[1];
162 }
163 for(N[1]=0; N[1] < post_offset[1]; ++N[1]) {
164 grid_position[2] = grid_properties.begin[2];
165 for(N[2]=0; N[2] < total[2]; ++N[2]) {
166 const_cast<VectorContent &>(SampledPotential)[index++] = 0.;
167 grid_position[2] += delta[2];
168 }
169 grid_position[1] += delta[1];
170 }
171 grid_position[0] += delta[0];
172 }
173 for(N[0]=0; N[0] < post_offset[0]; ++N[0]) {
174 grid_position[1] = grid_properties.begin[1];
175 for(N[1]=0; N[1] < total[1]; ++N[1]) {
176 grid_position[2] = grid_properties.begin[2];
177 for(N[2]=0; N[2] < total[2]; ++N[2]) {
178 const_cast<VectorContent &>(SampledPotential)[index++] = 0.;
179 grid_position[2] += delta[2];
180 }
181 grid_position[1] += delta[1];
182 }
183 grid_position[0] += delta[0];
184 }
185 // set remainder of points to zero
186 ASSERT( index == SampledPotential.getDimension(),
187 "PartialNucleiChargeFitter::PartialNucleiChargeFitter() - not enough or more than calculated sample points.");
188
189#ifndef NDEBUG
190 // write vector as paraview csv file file
191 {
192 size_t N[3];
193 size_t index = 0;
194 std::ofstream paraview_output("solution.csv");
195 paraview_output << "x coord,y coord,z coord,scalar" << std::endl;
196 for(N[0]=0; N[0] < total[0]; ++N[0]) {
197 for(N[1]=0; N[1] < total[1]; ++N[1]) {
198 for(N[2]=0; N[2] < total[2]; ++N[2]) {
199 paraview_output
200 << (double)N[0]/(double)total[0] << ","
201 << (double)N[1]/(double)total[1] << ","
202 << (double)N[2]/(double)total[2] << ","
203 << SampledPotential.at(index++) << std::endl;
204 }
205 }
206 }
207 paraview_output.close();
208 }
209#endif
210
211 LOG(1, "INFO: I masked " << masked_points << " points in right-hand-side.");
212// LOG(4, "DEBUG: Right-hand side vector is " << SampledPotential << ".");
213}
214
215bool PartialNucleiChargeFitter::isGridPointSettable(
216 const positions_t &_positions,
217 const Vector &grid_position) const
218{
219 bool status = true;
220 for (positions_t::const_iterator iter = _positions.begin();
221 iter != _positions.end(); ++iter) {
222 status &= grid_position.DistanceSquared(*iter) > threshold*threshold;
223 }
224 return status;
225}
226
227PartialNucleiChargeFitter::~PartialNucleiChargeFitter()
228{
229 delete PotentialFromCharges;
230}
231
232
233void PartialNucleiChargeFitter::constructMatrix()
234{
235 const size_t rows = SampledPotential.getDimension();
236 const size_t cols = positions.size();
237 PotentialFromCharges = new MatrixContent( rows, cols );
238 // store step length per axis
239 double delta[3];
240 for (size_t i=0;i<3;++i)
241 delta[i] = grid_properties.getDeltaPerAxis(i);
242 // then for each charge ...
243 size_t masked_points = 0;
244 for (size_t nuclei_index = 0; nuclei_index < cols; ++nuclei_index) {
245 // ... calculate potential at each grid position,
246 // i.e. step through grid and calculate distance to charge position
247 Vector grid_position; // position of grid point in real domain
248 grid_position[0] = grid_properties.begin[0];
249 size_t N[3]; // discrete grid position
250 size_t index = 0; // component of column vector
251 for(N[0]=0; N[0] < total[0]; ++N[0]) {
252 grid_position[1] = grid_properties.begin[1];
253 for(N[1]=0; N[1] < total[1]; ++N[1]) {
254 grid_position[2] = grid_properties.begin[2];
255 for(N[2]=0; N[2] < total[2]; ++N[2]) {
256 if (isGridPointSettable(positions, grid_position)) {
257 const double distance = positions[nuclei_index].distance(grid_position);
258 ASSERT( distance >= 0,
259 "PartialNucleiChargeFitter::constructMatrix() - distance is negative?");
260 // Coulomb's constant is 1 in atomic units, see http://en.wikipedia.org/wiki/Atomic_units
261 const double epsilon0_au = 1.; //4.*M_PI*0.007957747154594767;
262 // ... with epsilon_0 in atom units from http://folk.uio.no/michalj/node72.html
263 const double value = 1./(epsilon0_au*distance);
264 PotentialFromCharges->at(index++, nuclei_index) = value;
265 } else {
266 ++masked_points;
267 PotentialFromCharges->at(index++, nuclei_index) = 0.;
268 }
269 grid_position[2] += delta[2];
270 }
271 grid_position[1] += delta[1];
272 }
273 grid_position[0] += delta[0];
274 }
275 ASSERT( index == PotentialFromCharges->getRows(),
276 "PartialNucleiChargeFitter::operator() - number of sampled positions "
277 +toString(index)+" unequal to set rows "
278 +toString(PotentialFromCharges->getRows())+".");
279 }
280
281 LOG(1, "INFO: I masked " << masked_points/cols << " points in matrix.");
282}
283
284double PartialNucleiChargeFitter::operator()()
285{
286 // prepare PartialCharges
287 PartialCharges = new VectorContent(positions.size());
288
289 // set up over-determined system's problem matrix A for Ax=b
290 // i.e. columns represent potential of a single charge at grid positions
291 constructMatrix();
292
293 // solve for x
294 *PartialCharges =
295 PotentialFromCharges->solveOverdeterminedLinearEquation(
296 SampledPotential);
297
298// LOG(4, "DEBUG: Solution vector is " << (*PotentialFromCharges) * (*PartialCharges) << ".");
299
300 // calculate residual vector
301 VectorContent residuum = (*PotentialFromCharges) * (*PartialCharges) - SampledPotential;
302
303#ifndef NDEBUG
304 // write solution to file
305 writeMatrix();
306
307 // write vector as paraview csv file file
308 {
309 size_t N[3];
310 size_t index = 0;
311 std::ofstream paraview_output("residuum.csv");
312 paraview_output << "x coord,y coord,z coord,scalar" << std::endl;
313 for(N[0]=0; N[0] < total[0]; ++N[0]) {
314 for(N[1]=0; N[1] < total[1]; ++N[1]) {
315 for(N[2]=0; N[2] < total[2]; ++N[2]) {
316 paraview_output
317 << (double)N[0]/(double)total[0] << ","
318 << (double)N[1]/(double)total[1] << ","
319 << (double)N[2]/(double)total[2] << ","
320 << residuum.at(index++) << std::endl;
321 }
322 }
323 }
324 paraview_output.close();
325 }
326#endif
327
328 // calculate L1 and L2 errors
329 double residuum_l1 = 0.;
330 for (size_t i=0; i< residuum.getDimension(); ++i)
331 if (residuum_l1 < residuum[i])
332 residuum_l1 = residuum[i];
333 LOG(1, "INFO: L2-Norm of residuum is " << residuum.Norm() << ".");
334 LOG(1, "INFO: L1-Norm of residuum is " << residuum_l1 << ".");
335
336 return residuum.Norm();
337}
338
339void PartialNucleiChargeFitter::writeMatrix()
340{
341 constructMatrix();
342
343 // write matrix as paraview csv file file
344 size_t N[3];
345 size_t index=0;
346 std::string filename = std::string("potential.csv");
347 std::ofstream paraview_output(filename.c_str());
348 paraview_output << "x coord,y coord,z coord,scalar" << std::endl;
349 for(N[0]=0; N[0] < total[0]; ++N[0]) {
350 for(N[1]=0; N[1] < total[1]; ++N[1]) {
351 for(N[2]=0; N[2] < total[2]; ++N[2]) {
352 double sum = 0.;
353 for (size_t nuclei_index = 0; nuclei_index < positions.size(); ++nuclei_index) {
354 sum+= PotentialFromCharges->at(index, nuclei_index)*PartialCharges->at(nuclei_index);
355 }
356 paraview_output
357 << (double)N[0]/(double)total[0] << ","
358 << (double)N[1]/(double)total[1] << ","
359 << (double)N[2]/(double)total[2] << ","
360 << sum << std::endl;
361 index++;
362 }
363 }
364 }
365 paraview_output.close();
366}
367
368PartialNucleiChargeFitter::charges_t
369PartialNucleiChargeFitter::getSolutionAsCharges_t() const
370{
371 ASSERT( PartialCharges != NULL,
372 "PartialNucleiChargeFitter::getSolutionAsCharges_t() - PartialCharges requested prior to calculation.");
373 charges_t return_charges(positions.size(), 0.);
374 for (size_t i = 0; i < return_charges.size(); ++i)
375 return_charges[i] = PartialCharges->at(i);
376 return return_charges;
377}
Note: See TracBrowser for help on using the repository browser.