58 using Scalar =
typename MDTraits::Scalar;
59 using SolutionVector =
typename MDTraits::SolutionVector;
60 using PointSourceData =
typename BaseType::PointSourceTraits::PointSourceData;
62 static constexpr auto bulkIdx =
typename MDTraits::template SubDomain<0>::Index();
63 static constexpr auto lowDimIdx =
typename MDTraits::template SubDomain<1>::Index();
66 template<std::
size_t id>
using SubDomainTypeTag =
typename MDTraits::template SubDomain<id>::TypeTag;
69 template<std::
size_t id>
using GridView =
typename GridGeometry<id>::GridView;
70 template<std::
size_t id>
using Element =
typename GridView<id>::template Codim<0>::Entity;
73 using GlobalPosition =
typename Element<bulkIdx>::Geometry::GlobalCoordinate;
75 template<std::
size_t id>
76 static constexpr bool isBox()
79 static_assert(!isBox<bulkIdx>() && !isBox<lowDimIdx>(),
"The kernel coupling method is only implemented for the tpfa method");
80 static_assert(Dune::Capabilities::isCartesian<typename GridView<bulkIdx>::Grid>::v,
"The kernel coupling method is only implemented for structured grids");
83 bulkDim = GridView<bulkIdx>::dimension,
84 lowDimDim = GridView<lowDimIdx>::dimension,
85 dimWorld = GridView<bulkIdx>::dimensionworld
89 template <
typename T,
typename ...Ts>
90 using VariableKernelWidthDetector =
decltype(std::declval<T>().kernelWidthFactor(std::declval<Ts>()...));
92 template<
class T,
typename ...Args>
93 static constexpr bool hasKernelWidthFactor()
94 {
return Dune::Std::is_detected<VariableKernelWidthDetector, T, Args...>::value; }
99 using ParentType::ParentType;
101 void init(std::shared_ptr<Problem<bulkIdx>> bulkProblem,
102 std::shared_ptr<Problem<lowDimIdx>> lowDimProblem,
103 const SolutionVector&
curSol)
109 DUNE_THROW(Dune::NotImplemented,
"The current intersection detection may likely fail for refined grids.");
118 template<std::
size_t id,
class JacobianPattern>
121 extendedSourceStencil_.extendJacobianPattern(*
this, domainI, pattern);
131 template<std::
size_t i,
class LocalAssemblerI,
class JacobianMatrixDiagBlock,
class Gr
idVariables>
133 const LocalAssemblerI& localAssemblerI,
134 const typename LocalAssemblerI::LocalResidual::ElementResidualVector&,
135 JacobianMatrixDiagBlock& A,
136 GridVariables& gridVariables)
138 extendedSourceStencil_.evalAdditionalDomainDerivatives(*
this, domainI, localAssemblerI, A, gridVariables);
155 std::cout <<
"[coupling] Initializing the integration point source data structures..." << std::endl;
158 prepareDataStructures_();
159 std::cout <<
"[coupling] Resized data structures." << std::endl;
161 const auto& bulkGridGeometry = this->
problem(bulkIdx).gridGeometry();
162 const auto& lowDimGridGeometry = this->
problem(lowDimIdx).gridGeometry();
166 static const double characteristicRelativeLength =
getParam<double>(
"MixedDimension.KernelIntegrationCRL", 0.1);
169 static const bool writeIntegrationPointsToFile =
getParam<bool>(
"MixedDimension.WriteIntegrationPointsToFile",
false);
170 if (writeIntegrationPointsToFile)
172 std::ofstream ipPointFile(
"kernel_points.log", std::ios::trunc);
173 ipPointFile <<
"x,y,z\n";
174 std::cout <<
"[coupling] Initialized kernel_points.log." << std::endl;
177 for (
const auto& is : intersections(this->
glue()))
180 const auto& inside = is.targetEntity(0);
182 const auto intersectionGeometry = is.geometry();
183 const auto lowDimElementIdx = lowDimGridGeometry.elementMapper().index(inside);
188 const auto radius = this->
problem(lowDimIdx).spatialParams().radius(lowDimElementIdx);
189 const auto kernelWidthFactor = kernelWidthFactor_(this->
problem(lowDimIdx).spatialParams(), lowDimElementIdx);
190 const auto kernelWidth = kernelWidthFactor*
radius;
191 const auto a = intersectionGeometry.corner(0);
192 const auto b = intersectionGeometry.corner(1);
196 for (
int outsideIdx = 0; outsideIdx < is.numDomainNeighbors(); ++outsideIdx)
207 const auto& outside = is.domainEntity(outsideIdx);
208 const auto bulkElementIdx = bulkGridGeometry.elementMapper().index(outside);
209 const auto surfaceFactor = computeBulkSource(intersectionGeometry,
radius, kernelWidth,
id, lowDimElementIdx, bulkElementIdx, cylIntegration, is.numDomainNeighbors());
212 const auto center = intersectionGeometry.center();
213 this->
pointSources(lowDimIdx).emplace_back(
center,
id, surfaceFactor, intersectionGeometry.volume(), lowDimElementIdx);
214 this->
pointSources(lowDimIdx).back().setEmbeddings(is.numDomainNeighbors());
218 PointSourceData psData;
221 if constexpr (isBox<lowDimIdx>())
223 using ShapeValues = std::vector<Dune::FieldVector<Scalar, 1> >;
224 const auto lowDimGeometry = this->
problem(lowDimIdx).gridGeometry().element(lowDimElementIdx).geometry();
225 ShapeValues shapeValues;
227 psData.addLowDimInterpolation(shapeValues, this->
vertexIndices(lowDimIdx, lowDimElementIdx), lowDimElementIdx);
231 psData.addLowDimInterpolation(lowDimElementIdx);
235 if constexpr (isBox<bulkIdx>())
237 using ShapeValues = std::vector<Dune::FieldVector<Scalar, 1> >;
238 const auto bulkGeometry = this->
problem(bulkIdx).gridGeometry().element(bulkElementIdx).geometry();
239 ShapeValues shapeValues;
241 psData.addBulkInterpolation(shapeValues, this->
vertexIndices(bulkIdx, bulkElementIdx), bulkElementIdx);
245 psData.addBulkInterpolation(bulkElementIdx);
257 if constexpr (isBox<bulkIdx>())
259 const auto& vertices = this->
vertexIndices(bulkIdx, bulkElementIdx);
261 vertices.begin(), vertices.end());
265 this->
couplingStencils(lowDimIdx)[lowDimElementIdx].push_back(bulkElementIdx);
271 makeUniqueStencil_();
274 DUNE_THROW(Dune::InvalidStateException,
"Kernel method shouldn't have point sources in the bulk domain but only volume sources!");
276 std::cout <<
"[coupling] Finished preparing manager in " << watch.elapsed() <<
" seconds." << std::endl;
286 const std::vector<std::size_t>&
bulkSourceIds(GridIndex<bulkIdx> eIdx,
int scvIdx = 0)
const
287 {
return bulkSourceIds_[eIdx][scvIdx]; }
291 {
return bulkSourceWeights_[eIdx][scvIdx]; }
295 {
return fluxScalingFactor_[id]; }
302 const typename BaseType::template CouplingStencils<bulkIdx>::mapped_type&
305 const auto& sourceStencils = extendedSourceStencil_.stencil();
306 if (
auto stencil = sourceStencils.find(eIdx); stencil != sourceStencils.end())
307 return stencil->second;
314 template<
class Line,
class CylIntegration>
315 Scalar computeBulkSource(
const Line& line,
const Scalar radius,
const Scalar kernelWidth,
316 std::size_t
id, GridIndex<lowDimIdx> lowDimElementIdx, GridIndex<bulkIdx> coupledBulkElementIdx,
317 const CylIntegration& cylIntegration,
int embeddings)
320 static const auto bulkParamGroup = this->problem(bulkIdx).paramGroup();
324 const auto cylSamples = cylIntegration.size();
325 const auto& a = line.corner(0);
326 const auto& b = line.corner(1);
329 static const auto writeIntegrationPointsToFile =
getParam<bool>(
"MixedDimension.WriteIntegrationPointsToFile",
false);
330 if (writeIntegrationPointsToFile)
332 std::ofstream ipPointFile(
"kernel_points.log", std::ios::app);
333 for (
int i = 0; i < cylSamples; ++i)
335 const auto& point = cylIntegration.integrationPoint(i);
337 ipPointFile << point[0] <<
"," << point[1] <<
"," << point[2] <<
'\n';
341 Scalar integral = 0.0;
342 for (
int i = 0; i < cylSamples; ++i)
344 const auto& point = cylIntegration.integrationPoint(i);
351 assert(bulkElementIdx < this->problem(bulkIdx).gridGeometry().gridView().size(0));
352 const auto localWeight = evalConstKernel_(a, b, point, radius, kernelWidth)*cylIntegration.integrationElement(i)/Scalar(embeddings);
353 integral += localWeight;
354 if (!bulkSourceIds_[bulkElementIdx][0].empty() &&
id == bulkSourceIds_[bulkElementIdx][0].back())
356 bulkSourceWeights_[bulkElementIdx][0].back() += localWeight;
360 bulkSourceIds_[bulkElementIdx][0].emplace_back(
id);
361 bulkSourceWeights_[bulkElementIdx][0].emplace_back(localWeight);
362 addBulkSourceStencils_(bulkElementIdx, lowDimElementIdx, coupledBulkElementIdx);
368 const auto length = (a-b).two_norm()/Scalar(embeddings);
369 return integral/length;
372 void prepareDataStructures_()
377 bulkSourceIds_.clear();
378 bulkSourceWeights_.clear();
379 extendedSourceStencil_.stencil().clear();
382 this->precomputeVertexIndices(bulkIdx);
383 this->precomputeVertexIndices(lowDimIdx);
385 bulkSourceIds_.resize(this->gridView(bulkIdx).size(0));
386 bulkSourceWeights_.resize(this->gridView(bulkIdx).size(0));
392 this->pointSourceData().reserve(this->glue().size());
393 this->averageDistanceToBulkCell().reserve(this->glue().size());
394 fluxScalingFactor_.reserve(this->glue().size());
397 const auto numBulkElements = this->gridView(bulkIdx).size(0);
398 for (GridIndex<bulkIdx> bulkElementIdx = 0; bulkElementIdx < numBulkElements; ++bulkElementIdx)
400 this->couplingStencils(bulkIdx)[bulkElementIdx].reserve(10);
401 extendedSourceStencil_.stencil()[bulkElementIdx].reserve(10);
402 bulkSourceIds_[bulkElementIdx][0].reserve(10);
403 bulkSourceWeights_[bulkElementIdx][0].reserve(10);
408 void makeUniqueStencil_()
411 for (
auto&& stencil : extendedSourceStencil_.stencil())
413 std::sort(stencil.second.begin(), stencil.second.end());
414 stencil.second.erase(std::unique(stencil.second.begin(), stencil.second.end()), stencil.second.end());
417 if constexpr (isBox<bulkIdx>())
419 const auto& indices = this->vertexIndices(bulkIdx, stencil.first);
420 stencil.second.erase(std::remove_if(stencil.second.begin(), stencil.second.end(),
421 [&](
auto i){ return std::find(indices.begin(), indices.end(), i) != indices.end(); }),
422 stencil.second.end());
427 stencil.second.erase(std::remove_if(stencil.second.begin(), stencil.second.end(),
428 [&](
auto i){ return i == stencil.first; }),
429 stencil.second.end());
434 using namespace Dune::Hybrid;
435 forEach(integralRange(Dune::index_constant<2>{}), [&](
const auto domainIdx)
437 for (
auto&& stencil : this->couplingStencils(domainIdx))
439 std::sort(stencil.second.begin(), stencil.second.end());
440 stencil.second.erase(std::unique(stencil.second.begin(), stencil.second.end()), stencil.second.end());
446 void addBulkSourceStencils_(GridIndex<bulkIdx> bulkElementIdx, GridIndex<lowDimIdx> coupledLowDimElementIdx, GridIndex<bulkIdx> coupledBulkElementIdx)
449 if constexpr (isBox<lowDimIdx>())
451 const auto& vertices = this->vertexIndices(lowDimIdx, coupledLowDimElementIdx);
452 this->couplingStencils(bulkIdx)[bulkElementIdx].insert(this->couplingStencils(bulkIdx)[bulkElementIdx].end(),
453 vertices.begin(), vertices.end());
458 auto& s = this->couplingStencils(bulkIdx)[bulkElementIdx];
459 s.push_back(coupledLowDimElementIdx);
464 if constexpr (isBox<bulkIdx>())
466 const auto& vertices = this->vertexIndices(bulkIdx, coupledBulkElementIdx);
467 extendedSourceStencil_.stencil()[bulkElementIdx].insert(extendedSourceStencil_.stencil()[bulkElementIdx].end(),
468 vertices.begin(), vertices.end());
472 auto& s = extendedSourceStencil_.stencil()[bulkElementIdx];
473 s.push_back(coupledBulkElementIdx);
478 Scalar evalConstKernel_(
const GlobalPosition& a,
479 const GlobalPosition& b,
480 const GlobalPosition& point,
482 const Scalar rho)
const noexcept
485 const auto ab = b - a;
486 const auto t = (point - a)*ab/ab.two_norm2();
489 if (t < 0.0 || t > 1.0)
493 auto proj = a; proj.axpy(t, ab);
494 const auto radiusSquared = (proj - point).two_norm2();
496 if (radiusSquared > rho*rho)
499 return 1.0/(M_PI*rho*rho);
505 template<
class SpatialParams>
506 auto kernelWidthFactor_(
const SpatialParams& spatialParams,
unsigned int eIdx)
507 -> std::enable_if_t<hasKernelWidthFactor<SpatialParams, unsigned int>(), Scalar>
508 {
return spatialParams.kernelWidthFactor(eIdx); }
513 template<
class SpatialParams>
514 auto kernelWidthFactor_(
const SpatialParams& spatialParams,
unsigned int eIdx)
515 -> std::enable_if_t<!hasKernelWidthFactor<SpatialParams, unsigned int>(), Scalar>
517 static const Scalar kernelWidthFactor =
getParam<Scalar>(
"MixedDimension.KernelWidthFactor");
518 return kernelWidthFactor;
522 EmbeddedCoupling::ExtendedSourceStencil<ThisType> extendedSourceStencil_;
525 std::vector<std::array<std::vector<std::size_t>, isBox<bulkIdx>() ? 1<<bulkDim : 1>> bulkSourceIds_;
527 std::vector<std::array<std::vector<Scalar>, isBox<bulkIdx>() ? 1<<bulkDim : 1>> bulkSourceWeights_;
530 std::vector<Scalar> fluxScalingFactor_;