/* * Project: MoleCuilder * Description: creates and alters molecular systems * Copyright (C) 2012 University of Bonn. All rights reserved. * * * This file is part of MoleCuilder. * * MoleCuilder is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 2 of the License, or * (at your option) any later version. * * MoleCuilder is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with MoleCuilder. If not, see . */ /* * VMGJob.cpp * * Created on: Jul 13, 2012 * Author: heber */ // include config.h #ifdef HAVE_CONFIG_H #include #endif #ifdef HAVE_MPI #include "mpi.h" #endif // include headers that implement a archive in simple text format // otherwise BOOST_CLASS_EXPORT_IMPLEMENT has no effect #include #include #include "mg.hpp" #include "base/object.hpp" #include "base/defs.hpp" #ifdef HAVE_MPI #include "comm/comm_mpi.hpp" #include "comm/domain_decomposition_mpi.hpp" #else #include "comm/comm_serial.hpp" #endif #include "cycles/cycle_cs_periodic.hpp" #include "grid/multigrid.hpp" //#include "grid/tempgrid.hpp" #include "level/level_operator_cs.hpp" #include "level/stencils.hpp" #include "samples/discretization_poisson_fd.hpp" #include "smoother/gsrb_poisson_4.hpp" #include "solver/givens.hpp" //#include "solver/solver_regular.hpp" #include "solver/solver_singular.hpp" #include "units/particle/comm_mpi_particle.hpp" #include "CodePatterns/MemDebug.hpp" #include "Jobs/VMGJob.hpp" #include "LinearAlgebra/defs.hpp" #include "Jobs/InterfaceVMGJob.hpp" #include "CodePatterns/Assert.hpp" using namespace VMG; VMGJob::VMGJob( const JobId_t _JobId, const SamplingGrid &_density_grid, const std::vector< std::vector< double > > &_particle_positions, const std::vector< double > &_particle_charges, const size_t _near_field_cells) : FragmentJob(_JobId), density_grid(_density_grid), particle_positions(_particle_positions), particle_charges(_particle_charges), near_field_cells(_near_field_cells), returndata(static_cast(_density_grid)), num_particles(0), f(NULL), x(NULL), p(NULL), q(NULL) {} VMGJob::VMGJob() : FragmentJob(JobId::IllegalJob), near_field_cells(5), num_particles(0), f(NULL), x(NULL), p(NULL), q(NULL) {} VMGJob::~VMGJob() { delete[] f; delete[] x; delete[] p; delete[] q; } void VMGJob::InitVMGArrays() { num_particles = particle_charges.size(); f = new double[num_particles*3]; x = new double[num_particles*3]; p = new double[num_particles]; q = new double[num_particles]; { size_t index=0; for (std::vector< std::vector< double > >::const_iterator iter = particle_positions.begin(); iter != particle_positions.end(); ++iter) { for (std::vector< double >::const_iterator positer = (*iter).begin(); positer != (*iter).end(); ++positer) { f[index] = 0.; x[index++] = *positer; } } ASSERT( index == num_particles*3, "VMGJob::VMGJob() - too many particles or components in particle_positions."); } { size_t index = 0; for (std::vector< double >::const_iterator iter = particle_charges.begin(); iter != particle_charges.end(); ++iter) { p[index] = 0.; q[index++] = *iter; } ASSERT( index == num_particles, "VMGJob::VMGJob() - too many charges in particle_charges."); } } FragmentResult::ptr VMGJob::Work() { // initialize VMG library solver InitVMG(); /* * Start the multigrid solver */ MG::Solve(); /// create and fill result object // place into returnstream std::stringstream returnstream; boost::archive::text_oarchive oa(returnstream); oa << returndata; FragmentResult::ptr ptr( new FragmentResult(getId(), returnstream.str()) ); /* * Delete all data. */ MG::Destroy(); return ptr; } /** Initialization of VMG library. * * The contents is heavily inspired from interface_fcs.cpp: VMG_fcs_init() of * the ScaFaCoS project. * */ void VMGJob::InitVMG() { // TODO: As a matter of fact should use open boundary conditions const Boundary boundary(Periodic, Periodic, Periodic); /* * Choose multigrid components (self-registering) */ #ifdef HAVE_MPI new Particle::CommMPI(boundary, new DomainDecompositionMPI()); #else new CommSerial(boundary); #endif new DiscretizationPoissonFD(4); new VMGInterfaces::InterfaceVMGJob( density_grid.sampled_grid, returndata, particle_positions, particle_charges, boundary, 2, density_grid.level, Vector(density_grid.begin), density_grid.size, near_field_cells); new LevelOperatorCS(Stencils::RestrictionFullWeight, Stencils::InterpolationTrilinear); new Givens(); const int cycle_type = 1; // V-type new CycleCSPeriodic(cycle_type); new GaussSeidelRBPoisson4(); /* * Register required parameters */ new ObjectStorage("PRESMOOTHSTEPS", 3); new ObjectStorage("POSTSMOOTHSTEPS", 3); new ObjectStorage("PRECISION", 1.0e-10); new ObjectStorage("MAX_ITERATION", 15); new ObjectStorage("PARTICLE_NEAR_FIELD_CELLS", near_field_cells); new ObjectStorage("PARTICLE_INTERPOLATION_DEGREE", 3); // first initialize f,x,p,q,... from STL vectors InitVMGArrays(); // then initialize as objects with VMG's storage new ObjectStorage("PARTICLE_POS_ARRAY", x); new ObjectStorage("PARTICLE_CHARGE_ARRAY", q); new ObjectStorage("PARTICLE_POTENTIAL_ARRAY", p); new ObjectStorage("PARTICLE_FIELD_ARRAY", f); new ObjectStorage("PARTICLE_NUM_LOCAL", num_particles); /* * Post init */ MG::PostInit(); /* * Check whether the library is correctly initialized now. */ MG::IsInitialized(); } // we need to explicitly instantiate the serialization functions as // its is only serialized through its base class FragmentJob BOOST_CLASS_EXPORT_IMPLEMENT(VMGJob)