6#include <dune/pdelab/backend/istl/geneo/geneo.hh>
12 template<
class V,
class GFS,
class GFS_EXT,
class GO,
class GO_EXT,
class LOP,
class CON,
class MBE>
14 public PDELab::OVLPScalarProductImplementation<GFS>,
15 public PDELab::LinearResultStorage
22 CG_GenEO(V& x0_,
const GFS& gfs_,
const GFS_EXT gfs_ext_, LOP& lop_, CON& constraints_, MBE& mbe_, Composites::SolverInfo & si_, Dune::MPIHelper& helper_,
int cells_) :
23 PDELab::OVLPScalarProductImplementation<GFS>(gfs_),
28 constraints(constraints_),
37 typedef typename GFS::template ConstraintsContainer<RF>::Type CC;
39 typedef typename GFS_EXT::template ConstraintsContainer<RF>::Type CC_EXTERIOR;
40 auto cg_ext = CC_EXTERIOR();
41 auto cc_bnd_neu_int_dir = CC();
44 Dune::PDELab::constraints(constraints,gfs,cg);
45 Dune::PDELab::constraints(constraints,gfs_ext,cg_ext);
47 Dune::PDELab::NoDirichletConstraintsParameters pnbc;
48 Dune::PDELab::constraints(pnbc,gfs,cc_bnd_neu_int_dir);
50 std::vector<double> times;
55 typedef Dune::PDELab::Backend::Vector<GFS_EXT, RF> V_EXTERIOR;
56 V_EXTERIOR x0_ext(gfs_ext, Dune::PDELab::Backend::unattached_container());
57 x0_ext.attach(x0.storage());
60 typedef typename GO::Jacobian J;
61 typedef typename GO_EXT::Jacobian J_ext;
64 GO go(gfs,cg,gfs,cg,lop,mbe);
70 GO_EXT go_neu(gfs_ext,cg_ext,gfs_ext,cg_ext,lop,mbe);
73 go_neu.jacobian(x0_ext,AF_neu);
76 typedef Dune::PDELab::LocalOperatorOvlpRegion<LOP, GFS_EXT> LOP_ovlp;
77 LOP_ovlp lop_ovlp(lop,gfs_ext);
78 typedef Dune::PDELab::GridOperator<GFS_EXT,GFS_EXT,LOP_ovlp,MBE,RF,RF,RF,CC,CC> GO_ovlp;
79 GO_ovlp go_ovlp(gfs_ext,cg_ext,gfs_ext,cg_ext,lop_ovlp,mbe);
80 J_ext AF_ovlp(go_ovlp);
82 go_ovlp.jacobian(x0_ext,AF_ovlp);
85 typedef Dune::PDELab::OverlappingOperator<CC,J,V,V> POP;
87 typedef Dune::PDELab::ISTL::ParallelHelper<GFS> PIH;
89 typedef Dune::PDELab::OverlappingScalarProduct<GFS,V> OSP;
94 double eigenvalue_threshold = solver_info.eigenvalue_threshold*(double)solver_info.overlap/(cells+solver_info.overlap);
95 if(helper.rank() == 0){
96 if(eigenvalue_threshold > 0) std::cout <<
"Eigenvalue threshhold: " << eigenvalue_threshold << std::endl;
97 else std::cout <<
"Using " << solver_info.nev <<
" EV in each subdomain" << std::endl;
99 typedef Dune::PDELab::LocalFunctionSpace<GFS, Dune::PDELab::AnySpaceTag> LFSU;
100 typedef typename LFSU::template Child<0>::Type LFS;
102 LFS lfs = lfsu.template child<0>();
104 std::shared_ptr<V> part_unity;
105 part_unity = std::make_shared<V>(standardPartitionOfUnity<V>(gfs, cc_bnd_neu_int_dir));
108 Dune::PDELab::set_constrained_dofs(cg_ext,0.0,dummy);
110 int nev = solver_info.nev;
112 std::shared_ptr<Dune::PDELab::SubdomainBasis<V> > subdomain_basis;
114 subdomain_basis = std::make_shared<Dune::PDELab::GenEOBasis<GFS,J_ext,V,3> >(gfs, AF_neu, AF_ovlp, eigenvalue_threshold, *part_unity, solver_info.nev, solver_info.nev_arpack, 0.001,
false, solver_info.verb);
116 subdomain_basis = std::make_shared<Dune::PDELab::ZEMBasis<GFS,LFS,V,3,3> >(gfs, lfs, *part_unity);
119 subdomain_basis = std::make_shared<Dune::PDELab::SubdomainBasis<V> >(*part_unity);
122 auto partunityspace = std::make_shared<Dune::PDELab::SubdomainProjectedCoarseSpace<GFS,J_ext,V,PIH> >(gfs, AF_neu, subdomain_basis, pihf);
123 typedef Dune::PDELab::ISTL::TwoLevelOverlappingAdditiveSchwarz<GFS,J,V,V> PREC_PCG;
124 std::shared_ptr<PREC_PCG> prec;
125 prec = std::make_shared<PREC_PCG>(gfs, AF, partunityspace,solver_info.coarseSpaceActive);
127 MPI_Barrier(gfs.gridView().comm());
128 if(helper.rank()==0) std::cout <<
"Geneo setup time " << watch.elapsed() << std::endl;
129 times.push_back(watch.elapsed());
138 std::shared_ptr<CGSolver<V>> solver;
139 solver = std::make_shared<CGSolver<V> >(popf,ospf,*prec,solver_info.KrylovTol,solver_info.MaxIt,(helper.rank()==0) ? solver_info.verb : 0,
true);
140 Dune::InverseOperatorResult result;
141 solver->apply(v,d,result);
143 times.push_back(watch.elapsed());
147 solver_info.setTimes(times);
148 solver_info.recordResult(result);
151 std::cout <<
"Solver not available . . . Please install UMFPACK as part of SuiteSparse" << std::endl;
157 template<
class M,
class V2,
class W>
158 void apply(M& A, V2& z, W& r,
typename Dune::template FieldTraits<typename V2::ElementType >::real_type reduction)
160 this->apply(z,r,reduction);
166 const GFS_EXT& gfs_ext;
168 const CON& constraints;
170 Composites::SolverInfo& solver_info;
171 Dune::MPIHelper& helper;