Finite Element Domain Decomposition Library
FEDDLib
Loading...
Searching...
No Matches
BCBuilder_def.hpp
1#ifndef BCBuilder_def_hpp
2#define BCBuilder_def_hpp
3
4#include "BCBuilder_decl.hpp"
5#include "feddlib/core/Utils/FEDDUtils.hpp"
6#include <Teuchos_RCPDecl.hpp>
7#include <Teuchos_ScalarTraitsDecl.hpp>
8#include <Xpetra_MatrixFactory.hpp>
9
18
19namespace FEDD {
20template<class SC,class LO,class GO,class NO>
21BCBuilder<SC,LO,GO,NO>::BCBuilder():
22 vecBC_func_(),
23 vecFlag_(),
24 vecBlockID_(),
25 vecDomain_(),
26 vecBCType_(),
27 vecDofs_(),
28 vecBC_Parameters_(),
29 vecExternalSol_(0),
30 resultPtr_(),
31 pointPtr_()
32#ifdef BCBuilder_TIMER
33,SetSystemRowTimer_(Teuchos::TimeMonitor::getNewCounter("BCBuilder: SetSystemRow")),
34BlockRowHasDirichletTimer_(Teuchos::TimeMonitor::getNewCounter("BCBuilder: SetSystemRow: BlockHasDiri")),
35FindFlagTimer_(Teuchos::TimeMonitor::getNewCounter("BCBuilder: SetSystemRow: BlockHasDiri: FindFlag"))
36#endif
37{
38
39}
40
41template<class SC,class LO,class GO,class NO>
42void BCBuilder<SC,LO,GO,NO>::addBC(BC_func_Type funcBC, int flag, int block, const DomainPtr_Type &domain, std::string type, int dofs){
43
44 vecBC_func_.push_back(funcBC);
45 vecFlag_.push_back(flag);
46 vecBlockID_.push_back(block);
47 vecDomain_.push_back(domain);
48 vecBCType_.push_back(type);
49 vecDofs_.push_back(dofs);
50 vec_dbl_Type dummy(1,0.);
51 vecBC_Parameters_.push_back(dummy);
52 MultiVectorConstPtr_Type dummyMV;
53 vecExternalSol_.push_back(dummyMV);
54
55}
56
57template<class SC,class LO,class GO,class NO>
58void BCBuilder<SC,LO,GO,NO>::addBC(BC_func_Type funcBC, int flag, int block, const DomainPtr_Type &domain, std::string type, int dofs, vec_dbl_Type &parameter_vec){
59
60 vecBC_func_.push_back(funcBC);
61 vecFlag_.push_back(flag);
62 vecBlockID_.push_back(block);
63 vecDomain_.push_back(domain);
64 vecBCType_.push_back(type);
65 vecDofs_.push_back(dofs);
66 vecBC_Parameters_.push_back(parameter_vec);
67 MultiVectorConstPtr_Type dummyMV;
68 vecExternalSol_.push_back(dummyMV);
69
70}
71
72template<class SC,class LO,class GO,class NO>
73void BCBuilder<SC,LO,GO,NO>::addBC(BC_func_Type funcBC, int flag, int block, const DomainPtr_Type &domain, std::string type, int dofs, vec_dbl_Type &parameter_vec, MultiVectorConstPtr_Type& externalSol){
74
75 vecBC_func_.push_back(funcBC);
76 vecFlag_.push_back(flag);
77 vecBlockID_.push_back(block);
78 vecDomain_.push_back(domain);
79 vecBCType_.push_back(type);
80 vecDofs_.push_back(dofs);
81 vecBC_Parameters_.push_back(parameter_vec);
82 vecExternalSol_.push_back(externalSol);
83
84}
85
86
87template<class SC,class LO,class GO,class NO>
88void BCBuilder<SC,LO,GO,NO>::set(const BlockMatrixPtr_Type &blockMatrix, const BlockMultiVectorPtr_Type &blockMV, double t) const{
89
90 setSystem(blockMatrix);
91
92 setRHS(blockMV, t);
93
94}
95
96template<class SC,class LO,class GO,class NO>
97void BCBuilder<SC,LO,GO,NO>::setRHS(const BlockMultiVectorPtr_Type &blockMV, double t) const{
98 vec_dbl_Type result;
99 vec_dbl_Type point;
101 TEUCHOS_TEST_FOR_EXCEPTION( blockMV->getNumVectors()>1, std::runtime_error, "BCBuilder::setRHS() only for getNumVectors == 1.");
102
103 int loc = 0;
104 for (int block = 0; block < blockMV->size(); block++) { // blocks of RHS vector
105 if(blockHasDirichletBC(block, loc)){
106
107 vec_int_ptr_Type bcFlags = vecDomain_.at(loc)->getBCFlagUnique();
108 vec2D_dbl_ptr_Type vecPoints = vecDomain_.at(loc)->getPointsUnique();
109 int dim = vecDomain_.at(loc)->getDimension();
110 int dofs = vecDofs_.at(loc);
111
112 result.resize(dofs,0.);
113 point.resize(dim,0.);
115 resultPtr_ = Teuchos::rcp(new vec_dbl_Type(dofs,0.));
116 pointPtr_ = Teuchos::rcp(new vec_dbl_Type(dim,0.));
117
118 Teuchos::ArrayRCP<SC> valuesRHS = blockMV->getBlock(block)->getDataNonConst(0);
119 for (int i = 0 ; i < bcFlags->size(); i++) {
120 int flag = bcFlags->at(i);
121 if(findFlag(flag, block, loc)){ // loc changed to postion corresponding to bcFlag
122 if (!vecBCType_.at(loc).compare("Dirichlet") ||
123 !vecBCType_.at(loc).compare("Dirichlet_X") ||
124 !vecBCType_.at(loc).compare("Dirichlet_Y") ||
125 !vecBCType_.at(loc).compare("Dirichlet_Z")||
126 !vecBCType_.at(loc).compare("Dirichlet_X_Y") ||
127 !vecBCType_.at(loc).compare("Dirichlet_X_Z") ||
128 !vecBCType_.at(loc).compare("Dirichlet_Y_Z")) {
129 if (!vecExternalSol_[loc].is_null()) { // we use the external vector here
130 std::string type = "standard";
131 this->setDirichletBoundaryFromExternal(valuesRHS, i, loc, t, type);
132 }
133 else {
134 for (int d=0; d < dim; d++) {
135 point.at(d) = vecPoints->at(i).at(d);
136 }
137 for (int d=0; d < dofs; d++) {
138 result.at(d) = vecPoints->at(i).at(d);
139 }
140 vecBC_func_.at(loc)(&point.at(0), &result.at(0), t, &(vecBC_Parameters_.at(loc).at(0)));
141
142 for (UN j=0; j<blockMV->getNumVectors(); j++) {
143 Teuchos::ArrayRCP<SC> values = blockMV->getBlock(block)->getDataNonConst(j);
144 if ( !vecBCType_.at(loc).compare("Dirichlet") ) {
145 for (int dd=0; dd < dofs; dd++)
146 values[ dofs * i + dd ] = result.at(dd);
147 }
148 else if ( !vecBCType_.at(loc).compare("Dirichlet_X") ) {
149 values[ dofs * i + 0 ] = result.at(0);
150 }
151 else if ( !vecBCType_.at(loc).compare("Dirichlet_Y") ) {
152 values[ dofs * i + 1 ] = result.at(1);
154 else if ( !vecBCType_.at(loc).compare("Dirichlet_Z") ) {
155 values[ dofs * i + 2 ] = result.at(2);
156 }
157 else if ( !vecBCType_.at(loc).compare("Dirichlet_X_Y") ) {
158 values[ dofs * i + 0 ] = result.at(0);
159 values[ dofs * i + 1 ] = result.at(1);
161 else if ( !vecBCType_.at(loc).compare("Dirichlet_X_Z") ) {
162 values[ dofs * i + 0 ] = result.at(0);
163 values[ dofs * i + 2 ] = result.at(2);
164 }
165 else if ( !vecBCType_.at(loc).compare("Dirichlet_Y_Z") ) {
166 values[ dofs * i + 1 ] = result.at(1);
167 values[ dofs * i + 2 ] = result.at(2);
168 }
169 }
170 }
172 }
173 }
174 }
175 //Code below is experimental
176 for (int i=0; i<vecBCType_.size(); i++) {
177 if( vecBCType_.at(i) == "Neumann" && vecBlockID_.at(i)==block){
178 DomainPtr_Type domain = vecDomain_.at(i);
179 FEFacPtr_Type feFactory = Teuchos::rcp( new FEFac_Type() );
180 feFactory->addFE(domain);
181
182 std::vector<SC> funcParameter(3 , 0.);//0: order, 1:time, 2:surface flag (no specific flag is used in function above)
183 funcParameter[1] = t;
184 funcParameter[2] = vecFlag_.at(i);
185 int dim = domain->getDimension();
186 int dofs = vecDofs_.at(i);
187 MultiVectorPtr_Type a;
188 MultiVectorPtr_Type aUnique;
189 if (dofs>1){
190 a = Teuchos::rcp(new MultiVector_Type ( domain->getMapVecFieldRepeated() ) );
191 aUnique = Teuchos::rcp(new MultiVector_Type ( domain->getMapVecFieldUnique() ) );
192 feFactory->assemblySurfaceIntegralFlag( dim, domain->getFEType(), a, "Vector", vecBC_func_.at(i), funcParameter );
193 }
194 else{
195 a = Teuchos::rcp(new MultiVector_Type ( domain->getMapRepeated() ) );
196 aUnique = Teuchos::rcp(new MultiVector_Type ( domain->getMapUnique() ) );
197 feFactory->assemblySurfaceIntegralFlag( dim, domain->getFEType(), a, "Scalar", vecBC_func_.at(i), funcParameter );
198 }
199 aUnique->exportFromVector( a, false, "Add" );
200 MultiVectorPtr_Type mv = blockMV->getBlockNonConst(block);
201 mv->update(1.,*aUnique,1.);
202 }
203 }
204 }
205}
207template<class SC,class LO,class GO,class NO>
208void BCBuilder<SC,LO,GO,NO>::setDirichletBoundaryFromExternal(Teuchos::ArrayRCP<SC>& values/*values will be set to this vector*/, LO index, int loc, double time, std::string type, Teuchos::ArrayRCP<SC> valuesSubstract) const{
209
210 Teuchos::ArrayRCP<SC> valuesEx = vecExternalSol_[loc]->getDataNonConst(0);//We assume that our Multivector only has one column
211
212 (*pointPtr_)[0] = valuesEx[index]; // laplace solution in current point
213
214 vecBC_func_.at(loc)( &((*pointPtr_)[0]), &((*resultPtr_)[0]), time, &(vecBC_Parameters_.at(loc).at(0)));
215
216 int dofs = resultPtr_->size();
217 if ( !vecBCType_.at(loc).compare("Dirichlet") ) {
218 if (type == "standard") {
219 for (int dd=0; dd < dofs; dd++){
220 values[ dofs * index + dd ] = (*resultPtr_)[dd];
221 }
222 }
223 else if (type == "BCMinusVec"){
224 for (int dd=0; dd < dofs; dd++)
225 values[ dofs * index + dd ] = (*resultPtr_)[dd] - valuesSubstract[ dofs * index + dd ];
226 }
227 else if (type == "VecMinusBC"){
228 for (int dd=0; dd < dofs; dd++)
229 values[ dofs * index + dd ] = valuesSubstract[ dofs * index + dd ] - (*resultPtr_)[dd];
230 }
231 }
232 else {
233 TEUCHOS_TEST_FOR_EXCEPTION( true, std::runtime_error, "BCBuilder::setDirichletBoundaryFromExternal only implemented for full Dirichlet.");
234 }
235}
236
237template<class SC,class LO,class GO,class NO>
238void BCBuilder<SC,LO,GO,NO>::setBCMinusVector(const BlockMultiVectorPtr_Type &outBlockMV, const BlockMultiVectorPtr_Type &substractBlockMV, double t ) const{
239
240 TEUCHOS_TEST_FOR_EXCEPTION( outBlockMV->getNumVectors()>1, std::runtime_error, "BCBuilder::setRHS() only for getNumVectors == 1.");
241
242 vec_dbl_Type result;
243 vec_dbl_Type point;
244 int loc = 0;
245
246 for (int block = 0; block < outBlockMV->size(); block++) { // blocks of RHS vector
247 if(blockHasDirichletBC(block, loc)){
248 vec_int_ptr_Type bcFlags = vecDomain_.at(loc)->getBCFlagUnique();
249 vec2D_dbl_ptr_Type vecPoints = vecDomain_.at(loc)->getPointsUnique();
250 int dim = vecDomain_.at(loc)->getDimension();
251 int dofs = vecDofs_.at(loc);
252
253 result.resize(dofs,0.);
254 point.resize(dim,0.);
255
256 resultPtr_ = Teuchos::rcp(new vec_dbl_Type(dofs,0.));
257 pointPtr_ = Teuchos::rcp(new vec_dbl_Type(dim,0.));
258
259 Teuchos::ArrayRCP<SC> valuesRHS = outBlockMV->getBlock(block)->getDataNonConst(0);
260 Teuchos::ArrayRCP<SC> valuesSubstract = substractBlockMV->getBlock(block)->getDataNonConst(0);
261
262 for (int i = 0 ; i < bcFlags->size(); i++) {
263 int flag = bcFlags->at(i);
264 if(findFlag(flag,block, loc)){ // loc changed to postion corresponding to bcFlag
265 if (!vecBCType_.at(loc).compare("Dirichlet") ||
266 !vecBCType_.at(loc).compare("Dirichlet_X") ||
267 !vecBCType_.at(loc).compare("Dirichlet_Y") ||
268 !vecBCType_.at(loc).compare("Dirichlet_Z")||
269 !vecBCType_.at(loc).compare("Dirichlet_X_Y") ||
270 !vecBCType_.at(loc).compare("Dirichlet_X_Z") ||
271 !vecBCType_.at(loc).compare("Dirichlet_Y_Z")) {
272 if (!vecExternalSol_[loc].is_null()) { // we use the external vector here
273 std::string type = "BCMinusVec";
274 this->setDirichletBoundaryFromExternal(valuesRHS, i, loc, t, type, valuesSubstract);
275 }
276 else{
277 for (int d=0; d < dim; d++) {
278 point.at(d) = vecPoints->at(i).at(d);
279 }
280 for (int d=0; d < dofs; d++) {
281 result.at(d) = vecPoints->at(i).at(d);
282 }
283 vecBC_func_.at(loc)(&point.at(0), &result.at(0), t, &(vecBC_Parameters_.at(loc).at(0)));
284
285 for (UN j=0; j<outBlockMV->getNumVectors(); j++) {
286 Teuchos::ArrayRCP<SC> values = outBlockMV->getBlock(block)->getDataNonConst(j);
287 Teuchos::ArrayRCP<const SC> substractValues = substractBlockMV->getBlock(block)->getData(j);
288 if ( !vecBCType_.at(loc).compare("Dirichlet") ) {
289 for (int dd=0; dd < dofs; dd++)
290 values[ dofs * i + dd ] = result.at(dd) - substractValues[ dofs * i + dd ];
291 }
292 else if ( !vecBCType_.at(loc).compare("Dirichlet_X") ) {
293 values[ dofs * i + 0 ] = result.at(0) - substractValues[ dofs * i + 0 ];
294 }
295 else if ( !vecBCType_.at(loc).compare("Dirichlet_Y") ) {
296 values[ dofs * i + 1 ] = result.at(1) - substractValues[ dofs * i + 1 ];
297 }
298 else if ( !vecBCType_.at(loc).compare("Dirichlet_Z") ) {
299 values[ dofs * i + 2 ] = result.at(2) - substractValues[ dofs * i + 2 ];
300 }
301 else if ( !vecBCType_.at(loc).compare("Dirichlet_X_Y") ) {
302 values[ dofs * i + 0 ] = result.at(0) - substractValues[ dofs * i + 0 ];
303 values[ dofs * i + 1 ] = result.at(1) - substractValues[ dofs * i + 1 ];
304 }
305 else if ( !vecBCType_.at(loc).compare("Dirichlet_X_Z") ) {
306 values[ dofs * i + 0 ] = result.at(0) - substractValues[ dofs * i + 0 ];
307 values[ dofs * i + 2 ] = result.at(2) - substractValues[ dofs * i + 2 ];
308 }
309 else if ( !vecBCType_.at(loc).compare("Dirichlet_Y_Z") ) {
310 values[ dofs * i + 1 ] = result.at(1) - substractValues[ dofs * i + 1 ];
311 values[ dofs * i + 2 ] = result.at(2) - substractValues[ dofs * i + 2 ];
312 }
313 }
314 }
315 }
316 }
317 }
318 }
319 //Code below is experimental
320 for (int i=0; i<vecBCType_.size(); i++) {
321 if( vecBCType_.at(i) == "Neumann" && vecBlockID_.at(i)==block){
322 DomainPtr_Type domain = vecDomain_.at(i);
323 FEFacPtr_Type feFactory = Teuchos::rcp( new FEFac_Type() );
324 feFactory->addFE(domain);
325
326 std::vector<SC> funcParameter(3 , 0.);//0: order, 1:time, 2:surface flag (no specific flag is used in function above)
327 funcParameter[1] = t;
328 funcParameter[2] = vecFlag_.at(i);
329 int dim = domain->getDimension();
330 int dofs = vecDofs_.at(i);
331 MultiVectorPtr_Type a;
332 MultiVectorPtr_Type aUnique;
333 if (dofs>1){
334 a = Teuchos::rcp(new MultiVector_Type ( domain->getMapVecFieldRepeated() ) );
335 aUnique = Teuchos::rcp(new MultiVector_Type ( domain->getMapVecFieldUnique() ) );
336 feFactory->assemblySurfaceIntegralFlag( dim, domain->getFEType(), a, "Vector", vecBC_func_.at(i), funcParameter );
337 }
338 else{
339 a = Teuchos::rcp(new MultiVector_Type ( domain->getMapRepeated() ) );
340 aUnique = Teuchos::rcp(new MultiVector_Type ( domain->getMapUnique() ) );
341 feFactory->assemblySurfaceIntegralFlag( dim, domain->getFEType(), a, "Scalar", vecBC_func_.at(i), funcParameter );
342 }
343 aUnique->exportFromVector( a, false, "Add" );
344 MultiVectorPtr_Type mv = outBlockMV->getBlockNonConst(block);
345 mv->update(1.,*aUnique,1.);
346 MultiVectorPtr_Type mvMinus = substractBlockMV->getBlockNonConst(block);
347 mv->update(-1.,*mvMinus,1.);
348 }
349 }
350 }
351}
352
353template<class SC,class LO,class GO,class NO>
354void BCBuilder<SC,LO,GO,NO>::setVectorMinusBC(const BlockMultiVectorPtr_Type &outBlockMV, const BlockMultiVectorPtr_Type &substractBlockMV, double t) const{
355 vec_dbl_Type result;
356 vec_dbl_Type point;
357 int loc = 0;
358
359 TEUCHOS_TEST_FOR_EXCEPTION( outBlockMV->getNumVectors()>1, std::runtime_error, "BCBuilder::setRHS() only for getNumVectors == 1.");
360
361 for (int block = 0; block < outBlockMV->size(); block++) { // blocks of RHS vector
362 if(blockHasDirichletBC(block, loc)){
363 vec_int_ptr_Type bcFlags = vecDomain_.at(loc)->getBCFlagUnique();
364 vec2D_dbl_ptr_Type vecPoints = vecDomain_.at(loc)->getPointsUnique();
365 int dim = vecDomain_.at(loc)->getDimension();
366 int dofs = vecDofs_.at(loc);
367
368 result.resize(dofs,0.);
369 point.resize(dim,0.);
370
371 resultPtr_ = Teuchos::rcp(new vec_dbl_Type(dofs,0.));
372 pointPtr_ = Teuchos::rcp(new vec_dbl_Type(dim,0.));
373
374 Teuchos::ArrayRCP<SC> valuesRHS = outBlockMV->getBlock(block)->getDataNonConst(0);
375 Teuchos::ArrayRCP<SC> valuesSubstract = substractBlockMV->getBlock(block)->getDataNonConst(0);
376 for (int i = 0 ; i < bcFlags->size(); i++) {
377 int flag = bcFlags->at(i);
378 if(findFlag(flag,block, loc)){ // loc changed to postion corresponding to bcFlag
379 if (!vecBCType_.at(loc).compare("Dirichlet") ||
380 !vecBCType_.at(loc).compare("Dirichlet_X") ||
381 !vecBCType_.at(loc).compare("Dirichlet_Y") ||
382 !vecBCType_.at(loc).compare("Dirichlet_Z") ||
383 !vecBCType_.at(loc).compare("Dirichlet_X_Y") ||
384 !vecBCType_.at(loc).compare("Dirichlet_X_Z") ||
385 !vecBCType_.at(loc).compare("Dirichlet_Y_Z")) {
386 if (!vecExternalSol_[loc].is_null()) { // we use the external vector here
387 std::string type = "VecMinusBC";
388 this->setDirichletBoundaryFromExternal(valuesRHS, i, loc, t, type, valuesSubstract);
389 }
390 else{
391 for (int d=0; d < dim; d++) {
392 point.at(d) = vecPoints->at(i).at(d);
393 }
394 for (int d=0; d < dofs; d++) {
395 result.at(d) = vecPoints->at(i).at(d);
396 }
397 vecBC_func_.at(loc)(&point.at(0), &result.at(0), t, &(vecBC_Parameters_.at(loc).at(0)));
398
399 for (UN j=0; j<outBlockMV->getNumVectors(); j++) {
400 Teuchos::ArrayRCP<SC> values = outBlockMV->getBlock(block)->getDataNonConst(j);
401 Teuchos::ArrayRCP<const SC> substractValues = substractBlockMV->getBlock(block)->getData(j);
402 if ( !vecBCType_.at(loc).compare("Dirichlet") ) {
403 for (int dd=0; dd < dofs; dd++)
404 values[ dofs * i + dd ] = substractValues[ dofs * i + dd ] - result.at(dd);
405 }
406 else if ( !vecBCType_.at(loc).compare("Dirichlet_X") ) {
407 values[ dofs * i + 0 ] = substractValues[ dofs * i + 0 ] - result.at(0);
408 }
409 else if ( !vecBCType_.at(loc).compare("Dirichlet_Y") ) {
410 values[ dofs * i + 1 ] = substractValues[ dofs * i + 1 ] - result.at(1);
411 }
412 else if ( !vecBCType_.at(loc).compare("Dirichlet_Z") ) {
413 values[ dofs * i + 2 ] = substractValues[ dofs * i + 2 ] - result.at(2);
414 }
415 else if ( !vecBCType_.at(loc).compare("Dirichlet_X_Y") ) {
416 values[ dofs * i + 0 ] = substractValues[ dofs * i + 0 ] - result.at(0);
417 values[ dofs * i + 1 ] = substractValues[ dofs * i + 1 ] - result.at(1);
418 }
419 else if ( !vecBCType_.at(loc).compare("Dirichlet_X_Z") ) {
420 values[ dofs * i + 0 ] = substractValues[ dofs * i + 0 ] - result.at(0);
421 values[ dofs * i + 2 ] = substractValues[ dofs * i + 2 ] - result.at(2);
422 }
423 else if ( !vecBCType_.at(loc).compare("Dirichlet_Y_Z") ) {
424 values[ dofs * i + 1 ] = substractValues[ dofs * i + 1 ] - result.at(1);
425 values[ dofs * i + 2 ] = substractValues[ dofs * i + 2 ] - result.at(2);
426 }
427 }
428 }
429 }
430 }
431 }
432 }
433 //Code below is experimental
434 for (int i=0; i<vecBCType_.size(); i++) {
435 if( vecBCType_.at(i) == "Neumann" && vecBlockID_.at(i)==block){
436 DomainPtr_Type domain = vecDomain_.at(i);
437 FEFacPtr_Type feFactory = Teuchos::rcp( new FEFac_Type() );
438 feFactory->addFE(domain);
439
440 std::vector<SC> funcParameter(3 , 0.);//0: order, 1:time, 2:surface flag (no specific flag is used in function above)
441 funcParameter[1] = t;
442 funcParameter[2] = vecFlag_.at(i);
443 int dim = domain->getDimension();
444 int dofs = vecDofs_.at(i);
445 MultiVectorPtr_Type a;
446 MultiVectorPtr_Type aUnique;
447 if (dofs>1){
448 a = Teuchos::rcp(new MultiVector_Type ( domain->getMapVecFieldRepeated() ) );
449 aUnique = Teuchos::rcp(new MultiVector_Type ( domain->getMapVecFieldUnique() ) );
450 feFactory->assemblySurfaceIntegralFlag( dim, domain->getFEType(), a, "Vector", vecBC_func_.at(i), funcParameter );
451 }
452 else{
453 a = Teuchos::rcp(new MultiVector_Type ( domain->getMapRepeated() ) );
454 aUnique = Teuchos::rcp(new MultiVector_Type ( domain->getMapUnique() ) );
455 feFactory->assemblySurfaceIntegralFlag( dim, domain->getFEType(), a, "Scalar", vecBC_func_.at(i), funcParameter );
456 }
457 aUnique->exportFromVector( a, false, "Add" );
458 MultiVectorPtr_Type mv = outBlockMV->getBlockNonConst(block);
459 mv->update(1.,*aUnique,1.);
460 MultiVectorPtr_Type mvMinus = substractBlockMV->getBlockNonConst(block);
461 mv->update(1.,*mvMinus,-1.);
462 }
463 }
464 }
465}
466
467template<class SC,class LO,class GO,class NO>
468void BCBuilder<SC,LO,GO,NO>::setAllDirichletZero(const BlockMultiVectorPtr_Type &blockMV) const{
469 vec_dbl_Type result;
470 int loc = 0;
471 for (int block = 0; block < blockMV->size(); block++) { // blocks of RHS vector
472 if(blockHasDirichletBC(block, loc)){
473 vec_int_ptr_Type bcFlags = vecDomain_.at(loc)->getBCFlagUnique();
474 vec2D_dbl_ptr_Type vecPoints = vecDomain_.at(loc)->getPointsUnique();
475 int dim = vecDomain_.at(loc)->getDimension();
476 int dofs = vecDofs_.at(loc);
477
478 result.resize(dofs,0.);
479
480 for (int i = 0 ; i < bcFlags->size(); i++) {
481 int flag = bcFlags->at(i);
482 if(findFlag(flag,block, loc)){ // loc changed to postion corresponding to bcFlag
483 if ( !vecBCType_.at(loc).compare("Dirichlet") ||
484 !vecBCType_.at(loc).compare("Dirichlet_X") ||
485 !vecBCType_.at(loc).compare("Dirichlet_Y") ||
486 !vecBCType_.at(loc).compare("Dirichlet_Z") ||
487 !vecBCType_.at(loc).compare("Dirichlet_X_Y") ||
488 !vecBCType_.at(loc).compare("Dirichlet_X_Z") ||
489 !vecBCType_.at(loc).compare("Dirichlet_Y_Z")
490 ){
491
492 for (UN j=0; j<blockMV->getNumVectors(); j++) {
493 Teuchos::ArrayRCP<SC> values = blockMV->getBlock(block)->getDataNonConst(j);
494 if ( !vecBCType_.at(loc).compare("Dirichlet") ) {
495 for (int dd=0; dd < dofs; dd++)
496 values[ dofs * i + dd ] = result[dd];
497 }
498 else if ( !vecBCType_.at(loc).compare("Dirichlet_X") ) {
499 values[ dofs * i + 0 ] = result[0];
500 }
501 else if ( !vecBCType_.at(loc).compare("Dirichlet_Y") ) {
502 values[ dofs * i + 1 ] = result[1];
503 }
504 else if ( !vecBCType_.at(loc).compare("Dirichlet_Z") ) {
505 values[ dofs * i + 2 ] = result[2];
506 }
507 else if ( !vecBCType_.at(loc).compare("Dirichlet_X_Y") ) {
508 values[ dofs * i + 0 ] = result[0];
509 values[ dofs * i + 1 ] = result[1];
510 }
511 else if ( !vecBCType_.at(loc).compare("Dirichlet_X_Z") ) {
512 values[ dofs * i + 0 ] = result[0];
513 values[ dofs * i + 2 ] = result[2];
514 }
515 else if ( !vecBCType_.at(loc).compare("Dirichlet_Y_Z") ) {
516 values[ dofs * i + 1 ] = result[1];
517 values[ dofs * i + 2 ] = result[2];
518 }
519 }
520 }
521 }
522 }
523 }
524 }
525}
526
527
528
529template<class SC,class LO,class GO,class NO>
531 int dummyloc;
532
533 return blockHasDirichletBC(block, dummyloc);
534}
535
536template<class SC,class LO,class GO,class NO>
537bool BCBuilder<SC,LO,GO,NO>::blockHasDirichletBC(int block, int &loc) const{
538
539 bool hasBC = false;
540 bool found = false;
541 loc = -1;
542 std::vector<int>::const_iterator it = vecBlockID_.begin();
543 while (it!=vecBlockID_.end() && !found) {
544 it = std::find(it,vecBlockID_.end(),block);
545 if (it!=vecBlockID_.end()) {
546 loc = distance(vecBlockID_.begin(),it);
547 it++;
548 if (!vecBCType_.at(loc).compare("Dirichlet") ||
549 !vecBCType_.at(loc).compare("Dirichlet_X") ||
550 !vecBCType_.at(loc).compare("Dirichlet_Y") ||
551 !vecBCType_.at(loc).compare("Dirichlet_Z") ||
552 !vecBCType_.at(loc).compare("Dirichlet_X_Y") ||
553 !vecBCType_.at(loc).compare("Dirichlet_X_Z") ||
554 !vecBCType_.at(loc).compare("Dirichlet_Y_Z")
555 ) {
556 found = true;
557 hasBC = true;
558 }
559 }
560 }
561
562 return hasBC;
563}
564
565template<class SC,class LO,class GO,class NO>
566bool BCBuilder<SC,LO,GO,NO>::findFlag(LO flag, int block, int &loc) const{
567
568#ifdef BCBuilder_TIMER
569 Teuchos::TimeMonitor FindFlagMonitor(*FindFlagTimer_);
570#endif
571
572 bool hasFlag = false;
573 bool found = false;
574 loc = -1;
575 std::vector<int>::const_iterator it = vecFlag_.begin();
576 while (it!=vecFlag_.end() && !found) {
577 it = std::find(it,vecFlag_.end(),flag);
578 if (it!=vecFlag_.end()) {
579 loc = distance(vecFlag_.begin(),it);
580 if (vecBlockID_.at(loc)==block) {
581 hasFlag = true;
582 found = true;
583 }
584 else{
585 it++;
586 }
587 }
588 }
589
590 return hasFlag;
591}
592
593
594template<class SC,class LO,class GO,class NO>
595void BCBuilder<SC,LO,GO,NO>::setSystemScaled(const BlockMatrixPtr_Type &blockMatrix,double eps) const{
596
597 UN numBlocks = blockMatrix->size();
598 int loc;
599 for (UN blockRow = 0; blockRow < numBlocks; blockRow++) {
600
601#ifdef BCBuilder_TIMER
602 Teuchos::TimeMonitor SetSystemRowMonitor(*SetSystemRowTimer_);
603#endif
604 bool boolBlockHasDirichlet = false;
605
606 {
607#ifdef BCBuilder_TIMER
608 Teuchos::TimeMonitor BlockRowHasDirichletMonitor(*BlockRowHasDirichletTimer_);
609#endif
610 boolBlockHasDirichlet = blockHasDirichletBC(blockRow,loc);
611 }
612
613 if (boolBlockHasDirichlet){
614 for (UN blockCol = 0; blockCol < numBlocks ; blockCol++) {
615 if ( blockMatrix->blockExists( blockRow, blockCol ) ) {
616 MatrixPtr_Type matrix = blockMatrix->getBlock( blockRow, blockCol );
617 setDirichletBCScaled( matrix, loc, blockRow, blockRow==blockCol,eps );
618 }
619 }
620 }
621 }
622}
623
624
625template<class SC,class LO,class GO,class NO>
626void BCBuilder<SC,LO,GO,NO>::setSystem(const BlockMatrixPtr_Type &blockMatrix) const{
627
628 UN numBlocks = blockMatrix->size();
629 int loc;
630 for (UN blockRow = 0; blockRow < numBlocks; blockRow++) {
631
632#ifdef BCBuilder_TIMER
633 Teuchos::TimeMonitor SetSystemRowMonitor(*SetSystemRowTimer_);
634#endif
635 bool boolBlockHasDirichlet = false;
636
637 {
638#ifdef BCBuilder_TIMER
639 Teuchos::TimeMonitor BlockRowHasDirichletMonitor(*BlockRowHasDirichletTimer_);
640#endif
641
642 boolBlockHasDirichlet = blockHasDirichletBC(blockRow,loc);
643 }
644
645 if (boolBlockHasDirichlet){
646 for (UN blockCol = 0; blockCol < numBlocks ; blockCol++) {
647 if (blockMatrix->blockExists(blockRow, blockCol)) {
648 MatrixPtr_Type matrix = blockMatrix->getBlock(blockRow, blockCol);
649 setDirichletBC(matrix, loc, blockRow, blockRow == blockCol);
650 } else if (blockRow == blockCol) {
651 // If the current block is a diagonal block but does not exist we need to create it and insert the
652 // value 1 in all diagonal entries of the block corresponding to a Dirichlet boundary condition. We
653 // only need max. one entry per row.
654
655 // To get the correct domain from vecDomain_ we need to know the index of any (in this case the
656 // first) entry that was done with addBC for this block
657 auto matrixMap = this->vecDomain_.at(loc)->getMapUnique();
658 // // Use Xpetra::MatrixFactory to build a matrix with known row and column maps
659 auto tpetraMatrix = Teuchos::rcp(new Tpetra::CrsMatrix<SC,LO,GO,NO>(matrixMap->getTpetraMap(), matrixMap->getTpetraMap(), 1)); // Tpetra::MatrixFactory<SC, LO, GO, NO>::Build(matrixMap->getTpetraMap(), matrixMap->getTpetraMap(), 1);
660
661 MatrixPtr_Type matrix = Teuchos::rcp(new Matrix_Type(tpetraMatrix));
662
663 // Fill the matrix with zeros on the diagonal
664 Teuchos::Array<LO> colIndex(1, 0);
665 Teuchos::Array<SC> val(1, Teuchos::ScalarTraits<SC>::zero());
666 for (auto i = 0; i < matrixMap->getNodeNumElements(); i++) {
667 colIndex[0] = i;
668 // Since the matrix has a column map we can use insertLocalValues(). This is more efficient!
669 matrix->insertLocalValues(i, colIndex, val);
670 }
671 matrix->fillComplete(matrixMap, matrixMap);
672 setDirichletBC(matrix, loc, blockRow, true);
673 blockMatrix->addBlock(matrix, blockRow, blockCol);
674 }
675 }
676 }
677 }
678}
679
680
681template<class SC,class LO,class GO,class NO>
682void BCBuilder<SC,LO,GO,NO>::setDirichletBCScaled(const MatrixPtr_Type &matrix, int loc, int blockRow, bool isDiagonalBlock, double eps) const{
683
684 matrix->resumeFill();
685 bool isDirichlet;
686 UN dofsPerNode = vecDofs_.at(loc);
687 vec_int_ptr_Type bcFlags = vecDomain_.at(loc)->getBCFlagUnique();
688 MapConstPtr_Type nodeMap = vecDomain_.at(loc)->getMapUnique();
689
690 for (LO i=0; i<bcFlags->size(); i++) { // flags are for nodes.
691 int flag = bcFlags->at(i);
692 isDirichlet = false;
693 int locThisFlag;
694
695 if( findFlag(flag, blockRow, locThisFlag) ){
696
697 if( !vecBCType_.at(locThisFlag).compare("Dirichlet") ||
698 !vecBCType_.at(locThisFlag).compare("Dirichlet_X") ||
699 !vecBCType_.at(locThisFlag).compare("Dirichlet_Y") ||
700 !vecBCType_.at(locThisFlag).compare("Dirichlet_Z") ||
701 !vecBCType_.at(locThisFlag).compare("Dirichlet_X_Y") ||
702 !vecBCType_.at(locThisFlag).compare("Dirichlet_X_Z") ||
703 !vecBCType_.at(locThisFlag).compare("Dirichlet_Y_Z") )
704 isDirichlet = true;
705 if (isDirichlet) {
706
707 if (isDiagonalBlock)
708 setLocalRowEntry(matrix, i, dofsPerNode, locThisFlag,eps );
709 else
710 setLocalRowZero(matrix, i, dofsPerNode, locThisFlag );
711 }
712 }
713 }
714 matrix->fillComplete( matrix->getMap("domain"), matrix->getMap("range") );
715}
716
717
718template<class SC,class LO,class GO,class NO>
719void BCBuilder<SC,LO,GO,NO>::setLocalRowEntry(const MatrixPtr_Type &matrix, LO localNode, UN dofsPerNode, int loc,double eps) const{
720
721 Teuchos::ArrayView<const SC> valuesOld;
722 Teuchos::ArrayView<const LO> indices;
723
724 MapConstPtr_Type colMap = matrix->getMap("col");
725 LO localDof = (LO) (dofsPerNode * localNode);
726 for (UN dof=0; dof<dofsPerNode; dof++) {
727 if ( vecBCType_.at(loc) == "Dirichlet" ||
728 (vecBCType_.at(loc) == "Dirichlet_X" && dof==0 ) ||
729 (vecBCType_.at(loc) == "Dirichlet_Y" && dof==1 ) ||
730 (vecBCType_.at(loc) == "Dirichlet_Z" && dof==2 ) ||
731 (vecBCType_.at(loc) == "Dirichlet_X_Y" && dof!=2 )||
732 (vecBCType_.at(loc) == "Dirichlet_X_Z" && dof!=1 )||
733 (vecBCType_.at(loc) == "Dirichlet_Y_Z" && dof!=0 )
734 ) {
735 // cout << " Setting Dirichlet Row 1 for node " << localDof << " of type " << vecBCType_.at(loc) <<endl;
736 GO globalDof = matrix->getMap()->getGlobalElement( localDof );
737 matrix->getLocalRowView(localDof, indices, valuesOld);
738 double rowSum = 0.;
739 for (UN j=0; j<indices.size(); j++) {
740 rowSum += abs(valuesOld[j]);
741 }
742 Teuchos::Array<SC> values( valuesOld.size(), Teuchos::ScalarTraits<SC>::zero() );
743 bool setOne = false;
744 for (UN j=0; j<indices.size() && !setOne; j++) {
745 if ( colMap->getGlobalElement( indices[j] ) == globalDof ){
746 values[j] = valuesOld[j]*eps;
747
748 setOne = true;
749 }
750 }
751 matrix->replaceLocalValues(localDof, indices(), values());
752 }
753 localDof++;
754 }
755
756}
757
758
759template<class SC,class LO,class GO,class NO>
760void BCBuilder<SC,LO,GO,NO>::setDirichletBC(const MatrixPtr_Type &matrix, int loc, int blockRow, bool isDiagonalBlock) const{
761
762 matrix->resumeFill();
763 bool isDirichlet;
764 UN dofsPerNode = vecDofs_.at(loc);
765 vec_int_ptr_Type bcFlags = vecDomain_.at(loc)->getBCFlagUnique();
766
767 for (LO i=0; i<bcFlags->size(); i++) { // flags are for nodes.
768 int flag = bcFlags->at(i);
769 isDirichlet = false;
770 int locThisFlag;
771
772 if( findFlag(flag, blockRow, locThisFlag) ){
773
774 if( !vecBCType_.at(locThisFlag).compare("Dirichlet") ||
775 !vecBCType_.at(locThisFlag).compare("Dirichlet_X") ||
776 !vecBCType_.at(locThisFlag).compare("Dirichlet_Y") ||
777 !vecBCType_.at(locThisFlag).compare("Dirichlet_Z") ||
778 !vecBCType_.at(locThisFlag).compare("Dirichlet_X_Y") ||
779 !vecBCType_.at(locThisFlag).compare("Dirichlet_X_Z") ||
780 !vecBCType_.at(locThisFlag).compare("Dirichlet_Y_Z") )
781 isDirichlet = true;
782 if (isDirichlet) {
783
784 if (isDiagonalBlock)
785 setLocalRowOne(matrix, i, dofsPerNode, locThisFlag );
786 else
787 setLocalRowZero(matrix, i, dofsPerNode, locThisFlag );
788 }
789 }
790 }
791 matrix->fillComplete( matrix->getMap("domain"), matrix->getMap("range") );
792}
793
794template<class SC,class LO,class GO,class NO>
795void BCBuilder<SC,LO,GO,NO>::setLocalRowOne(const MatrixPtr_Type &matrix, LO localNode, UN dofsPerNode, int loc) const{
796
797 Teuchos::ArrayView<const SC> valuesOld;
798 Teuchos::ArrayView<const LO> indices;
799
800 MapConstPtr_Type colMap = matrix->getMap("col");
801 LO localDof = (LO) (dofsPerNode * localNode);
802 for (UN dof=0; dof<dofsPerNode; dof++) {
803 if ( vecBCType_.at(loc) == "Dirichlet" ||
804 (vecBCType_.at(loc) == "Dirichlet_X" && dof==0 ) ||
805 (vecBCType_.at(loc) == "Dirichlet_Y" && dof==1 ) ||
806 (vecBCType_.at(loc) == "Dirichlet_Z" && dof==2 ) ||
807 (vecBCType_.at(loc) == "Dirichlet_X_Y" && dof!=2 )||
808 (vecBCType_.at(loc) == "Dirichlet_X_Z" && dof!=1 )||
809 (vecBCType_.at(loc) == "Dirichlet_Y_Z" && dof!=0 )
810 ) {
811 // std::cout << " Setting Dirichlet Row 1 for node " << localDof << " of type " << vecBCType_.at(loc) <<endl;
812 GO globalDof = matrix->getMap()->getGlobalElement( localDof );
813 matrix->getLocalRowView(localDof, indices, valuesOld);
814 Teuchos::Array<SC> values( valuesOld.size(), Teuchos::ScalarTraits<SC>::zero() );
815 bool setOne = false;
816 for (UN j=0; j<indices.size() && !setOne; j++) {
817 if ( colMap->getGlobalElement( indices[j] ) == globalDof ){
818 values[j] = Teuchos::ScalarTraits<SC>::one();
819 setOne = true;
820 }
821 }
822 matrix->replaceLocalValues(localDof, indices(), values());
823 }
824 localDof++;
825 }
826
827}
828
829template<class SC,class LO,class GO,class NO>
830void BCBuilder<SC,LO,GO,NO>::setLocalRowZero(const MatrixPtr_Type &matrix, LO localNode, UN dofsPerNode, int loc) const{
831
832 Teuchos::ArrayView<const SC> valuesOld;
833 Teuchos::ArrayView<const LO> indices;
834 LO localDof = (LO) (dofsPerNode * localNode);
835 for (UN dof=0; dof<dofsPerNode; dof++) {
836 if ( vecBCType_.at(loc) == "Dirichlet" ||
837 (vecBCType_.at(loc) == "Dirichlet_X" && dof==0 ) ||
838 (vecBCType_.at(loc) == "Dirichlet_Y" && dof==1 ) ||
839 (vecBCType_.at(loc) == "Dirichlet_Z" && dof==2 ) ||
840 (vecBCType_.at(loc) == "Dirichlet_X_Y" && dof!=2 )||
841 (vecBCType_.at(loc) == "Dirichlet_X_Z" && dof!=1 )||
842 (vecBCType_.at(loc) == "Dirichlet_Y_Z" && dof!=0 )
843 ) {
844 matrix->getLocalRowView(localDof, indices, valuesOld);
845 Teuchos::Array<SC> values( valuesOld.size(), Teuchos::ScalarTraits<SC>::zero() );
846 matrix->replaceLocalValues(localDof, indices(), values());
847 }
848 localDof++;
849 }
850}
851
852
853
854template<class SC,class LO,class GO,class NO>
856 std::vector<int>::iterator it;
857 it = find(vecBlockID_.begin(),vecBlockID_.end(),block);
858#ifdef ASSERTS_WARNINGS
859 MYASSERT(it!=vecBlockID_.end(),"Requested dof information is not known to BCBuilder class.");
860#endif
861
862 return vecDofs_.at(distance(vecBlockID_.begin(),it));
863}
864}
865
866#endif
void setBCMinusVector(const BlockMultiVectorPtr_Type &outBlockMV, const BlockMultiVectorPtr_Type &substractBlockMV, double t=0.) const
Definition BCBuilder_def.hpp:238
void setVectorMinusBC(const BlockMultiVectorPtr_Type &outBlockMV, const BlockMultiVectorPtr_Type &substractBlockMV, double t=0.) const
Definition BCBuilder_def.hpp:354
void setRHS(const BlockMultiVectorPtr_Type &blockMV, double t=0.) const
Setting boundary conditions to (block)vector.
Definition BCBuilder_def.hpp:97
void set(const BlockMatrixPtr_Type &blockMatrix, const BlockMultiVectorPtr_Type &b, double t=0.) const
Setting bundary condtions to problem.
Definition BCBuilder_def.hpp:88
void setSystem(const BlockMatrixPtr_Type &blockMatrix) const
Set boundary conditions to system.
Definition BCBuilder_def.hpp:626
void addBC(BC_func_Type funcBC, int flag, int block, const DomainPtr_Type &domain, std::string type, int dofs)
Adding Boundary Condition.
Definition BCBuilder_def.hpp:42
void setLocalRowZero(const MatrixPtr_Type &matrix, LO localNode, UN dofsPerNode, int loc) const
Definition BCBuilder_def.hpp:830
void setDirichletBoundaryFromExternal(Teuchos::ArrayRCP< SC > &values, LO index, int loc, double time, std::string type, Teuchos::ArrayRCP< SC > valuesSubstract=Teuchos::null) const
void setSystemScaled(const BlockMatrixPtr_Type &blockMatrix, double eps=1.0) const
Set boundary conditions to system.
Definition BCBuilder_def.hpp:595
void setDirichletBC(const MatrixPtr_Type &matrix, int loc, int blockRow, bool isDiagonalBlock) const
Definition BCBuilder_def.hpp:760
void setAllDirichletZero(const BlockMultiVectorPtr_Type &blockMV) const
Definition BCBuilder_def.hpp:468
bool blockHasDirichletBC(int block) const
void setDirichletBCScaled(const MatrixPtr_Type &matrix, int loc, int blockRow, bool isDiagonalBlock, double eps=1.0) const
Definition BCBuilder_def.hpp:682
bool findFlag(LO flag, int block, int &loc) const
void setLocalRowEntry(const MatrixPtr_Type &matrix, LO localNode, UN dofsPerNode, int loc, double eps) const
Definition BCBuilder_def.hpp:719
void setLocalRowOne(const MatrixPtr_Type &matrix, LO localNode, UN dofsPerNode, int loc) const
Definition BCBuilder_def.hpp:795
int dofsPerNodeAtBlock(int block)
Definition BCBuilder_def.hpp:855
Adaptive Mesh Refinement.
Definition AdaptiveMeshRefinement_decl.hpp:33