MaMiCo 1.2
Loading...
Searching...
No Matches
LBCouetteSolver.h
1// Copyright (C) 2015 Technische Universitaet Muenchen
2// This file is part of the Mamico project. For conditions of distribution
3// and use, please see the copyright notice in Mamico's main folder
4#ifndef _MOLECULARDYNAMICS_COUPLING_SOLVERS_LBCOUETTESOLVER_H_
5#define _MOLECULARDYNAMICS_COUPLING_SOLVERS_LBCOUETTESOLVER_H_
6
7namespace coupling {
8namespace solvers {
11} // namespace solvers
12} // namespace coupling
13
14#if defined(_OPENMP)
15#include <omp.h>
16#endif
17#include "coupling/interface/PintableMacroSolver.h"
18#include "coupling/solvers/NumericalSolver.h"
19#include <cmath>
20
22public:
23 LBCouetteSolverState(int size) : _pdf(size, 0) {}
24
25 LBCouetteSolverState(int size, double* pdf) : LBCouetteSolverState(size) { std::copy(pdf, pdf + size, _pdf.data()); }
26
27 std::unique_ptr<State> clone() const override { return std::make_unique<LBCouetteSolverState>(*this); }
28
29 ~LBCouetteSolverState() {}
30
31 int getSizeBytes() const override { return sizeof(double) * _pdf.size(); }
32
33 std::unique_ptr<State> operator+(const State& rhs) override;
34 std::unique_ptr<State> operator-(const State& rhs) override;
35
36 bool operator==(const State& rhs) const override {
37 const LBCouetteSolverState* other = dynamic_cast<const LBCouetteSolverState*>(&rhs);
38 if (other == nullptr)
39 return false;
40 return _pdf == other->_pdf;
41 }
42
43 double* getData() override { return _pdf.data(); }
44 const double* getData() const override { return _pdf.data(); }
45
46 void print(std::ostream& os) const override { os << "<LBCouetteSolverState instance with size " << getSizeBytes() << ">"; }
47
48private:
49 std::vector<double> _pdf;
50};
51
57public:
73 LBCouetteSolver(const double channelheight, tarch::la::Vector<3, double> wallVelocity, const double kinVisc, const double dx, const double dt,
74 const int plotEveryTimestep, const bool plotAverageVelocity, const std::string filestem, const tarch::la::Vector<3, unsigned int> processes,
75 const unsigned int numThreads = 1, const Scenario* scen = nullptr)
76 : coupling::solvers::NumericalSolver(channelheight, dx, dt, kinVisc, plotEveryTimestep, filestem, processes, scen), _mode(Mode::coupling), _dt_pint(dt),
77 _omega(1.0 / (3.0 * (kinVisc * dt / (dx * dx)) + 0.5)), _wallVelocity((dt / dx) * wallVelocity), _plotAverageVelocity(plotAverageVelocity) {
78 // return if required
79 if (skipRank()) {
80 return;
81 }
82 _pdfsize = 19 * (_domainSizeX + 2) * (_domainSizeY + 2) * (_domainSizeZ + 2);
83 _pdf1 = new double[_pdfsize];
84 _pdf2 = new double[_pdfsize];
85#if defined(_OPENMP)
86 omp_set_num_threads(numThreads);
87#endif
88#if (COUPLING_MD_DEBUG == COUPLING_MD_YES)
89 std::cout << "Domain size=" << _domainSizeX << "," << _domainSizeY << "," << _domainSizeZ << std::endl;
90 std::cout << "tau=" << 1.0 / _omega << std::endl;
91 std::cout << "wallVelocity=" << _wallVelocity << std::endl;
92 for (int z = 0; z < _domainSizeZ + 2; z++) {
93 for (int y = 0; y < _domainSizeY + 2; y++) {
94 for (int x = 0; x < _domainSizeX + 2; x++) {
95 std::cout << x << "," << y << "," << z << "FLAG=" << _flag[get(x, y, z)] << std::endl;
96 }
97 }
98 }
99#endif
100 // check pointers
101 if ((_pdf1 == NULL) || (_pdf2 == NULL) || (_vel == NULL) || (_density == NULL) || (_flag == NULL)) {
102 std::cout << "ERROR LBCouetteSolver: NULL ptr!" << std::endl;
103 exit(EXIT_FAILURE);
104 }
105#if (COUPLING_MD_PARALLEL == COUPLING_MD_YES)
106 if ((_sendBufferX == NULL) || (_recvBufferX == NULL) || (_sendBufferY == NULL) || (_recvBufferY == NULL) || (_sendBufferZ == NULL) ||
107 (_recvBufferZ == NULL)) {
108 std::cout << "ERROR LBCouetteSolver: NULL ptr in send/recv!" << std::endl;
109 exit(EXIT_FAILURE);
110 }
111#endif
112// init everything with lattice weights
113#pragma omp parallel for
114 for (int i = 0; i < (_domainSizeX + 2) * (_domainSizeY + 2) * (_domainSizeZ + 2); i++) {
115 for (int q = 0; q < 19; q++) {
116 _pdf1[get(i) * 19 + q] = _W[q];
117 _pdf2[get(i) * 19 + q] = _W[q];
118 }
119 }
120 computeDensityAndVelocityEverywhere();
121 }
122
125 if (_pdf1 != NULL) {
126 delete[] _pdf1;
127 _pdf1 = NULL;
128 }
129 if (_pdf2 != NULL) {
130 delete[] _pdf2;
131 _pdf2 = NULL;
132 }
133 if (_vel != NULL) {
134 delete[] _vel;
135 _vel = NULL;
136 }
137 if (_density != NULL) {
138 delete[] _density;
139 _density = NULL;
140 }
141 if (_flag != NULL) {
142 delete[] _flag;
143 _flag = NULL;
144 }
145#if (COUPLING_MD_PARALLEL == COUPLING_MD_YES)
146 if (_sendBufferX != NULL) {
147 delete[] _sendBufferX;
148 _sendBufferX = NULL;
149 }
150 if (_sendBufferY != NULL) {
151 delete[] _sendBufferY;
152 _sendBufferY = NULL;
153 }
154 if (_sendBufferZ != NULL) {
155 delete[] _sendBufferZ;
156 _sendBufferZ = NULL;
157 }
158 if (_recvBufferX != NULL) {
159 delete[] _recvBufferX;
160 _recvBufferX = NULL;
161 }
162 if (_recvBufferY != NULL) {
163 delete[] _recvBufferY;
164 _recvBufferY = NULL;
165 }
166 if (_recvBufferZ != NULL) {
167 delete[] _recvBufferZ;
168 _recvBufferZ = NULL;
169 }
170#endif
171 }
172
175 void advance(double dt) override {
176 if (skipRank()) {
177 return;
178 }
179 const int timesteps = floor(dt / _dt + 0.5);
180 if (fabs(timesteps * _dt - dt) / _dt > 1.0e-8) {
181 std::cout << "ERROR LBCouetteSolver::advance(): time steps and dt do not match!" << std::endl;
182 exit(EXIT_FAILURE);
183 }
184 for (int i = 0; i < timesteps; i++) {
186 computeDensityAndVelocityEverywhere();
187 plot();
188 plot_avg_vel();
190 communicate(); // exchange between neighbouring MPI subdomains
191 _counter++;
192 }
193 }
194
200 if (skipRank()) {
201 return;
202 }
203#if (COUPLING_MD_ERROR == COUPLING_MD_YES)
204 if (_mode == Mode::supervising) {
205 std::cout << "ERROR LBCouetteSolver setMDBoundaryValues() called in supervising mode" << std::endl;
206 exit(EXIT_FAILURE);
207 }
208#endif
209 computeDensityAndVelocityEverywhere();
210
211 // loop over all received cells
212 for (auto pair : md2macroBuffer) {
213 I01 idx;
215 std::tie(couplingCell, idx) = pair;
216 // determine cell index of this cell in LB domain
217 tarch::la::Vector<3, unsigned int> globalCellCoords{idx.get()};
218 globalCellCoords[0] = (globalCellCoords[0] + _offset[0]) - _coords[0] * _avgDomainSizeX;
219 globalCellCoords[1] = (globalCellCoords[1] + _offset[1]) - _coords[1] * _avgDomainSizeY;
220 globalCellCoords[2] = (globalCellCoords[2] + _offset[2]) - _coords[2] * _avgDomainSizeZ;
221#if (COUPLING_MD_DEBUG == COUPLING_MD_YES)
222 std::cout << "Process coords: " << _coords << ": GlobalCellCoords for index " << idx << ": " << globalCellCoords << std::endl;
223#endif
224 const int index = get(globalCellCoords[0], globalCellCoords[1], globalCellCoords[2]);
225#if (COUPLING_MD_ERROR == COUPLING_MD_YES)
226 if (_flag[index] != MD_BOUNDARY) {
227 std::cout << "ERROR LBCouetteSolver::setMDBoundaryValues(): Cell " << index << " is no MD boundary cell!" << std::endl;
228 exit(EXIT_FAILURE);
229 }
230#endif
231 // set velocity value and pdfs in MD boundary cell (before streaming); the
232 // boundary velocities are interpolated between the neighbouring and this
233 // cell. This interpolation is valid for FLUID-MD_BOUNDARY neighbouring
234 // relations only. determine local velocity received from MaMiCo and
235 // convert it to LB units; store the velocity in _vel
236 // massFactor is used to ensure conservation of energy
237 tarch::la::Vector<3, double> localVel((1.0 / couplingCell->getMacroscopicMass()) * (_dt / _dx) * couplingCell->getMacroscopicMomentum());
238 for (unsigned int d = 0; d < 3; d++) {
239 _vel[3 * index + d] = localVel[d];
240 }
241 // loop over all pdfs and set them according to interpolated moving-wall
242 // conditions
243 for (unsigned int q = 0; q < 19; q++) {
244 // index of neighbour cell; only if cell is located inside local domain
245 if (((int)globalCellCoords[0] + _C[q][0] > 0) && ((int)globalCellCoords[0] + _C[q][0] < _domainSizeX + 1) &&
246 ((int)globalCellCoords[1] + _C[q][1] > 0) && ((int)globalCellCoords[1] + _C[q][1] < _domainSizeY + 1) &&
247 ((int)globalCellCoords[2] + _C[q][2] > 0) && ((int)globalCellCoords[2] + _C[q][2] < _domainSizeZ + 1)) {
248 const int nbIndex = get((_C[q][0] + globalCellCoords[0]), (_C[q][1] + globalCellCoords[1]), (_C[q][2] + globalCellCoords[2]));
249 const tarch::la::Vector<3, double> interpolVel(0.5 * (_vel[3 * index] + _vel[3 * nbIndex]), 0.5 * (_vel[3 * index + 1] + _vel[3 * nbIndex + 1]),
250 0.5 * (_vel[3 * index + 2] + _vel[3 * nbIndex + 2]));
251 _pdf1[19 * index + q] =
252 _pdf1[19 * nbIndex + 18 - q] -
253 6.0 * _W[q] * _density[nbIndex] * (_C[18 - q][0] * interpolVel[0] + _C[18 - q][1] * interpolVel[1] + _C[18 - q][2] * interpolVel[2]);
254 }
255 }
256 }
257 }
258
265 // check pos-data for process locality (todo: put this in debug mode in
266 // future releases)
267 if ((pos[0] < domainOffset[0]) || (pos[0] > domainOffset[0] + _domainSizeX * _dx) || (pos[1] < domainOffset[1]) ||
268 (pos[1] > domainOffset[1] + _domainSizeY * _dx) || (pos[2] < domainOffset[2]) || (pos[2] > domainOffset[2] + _domainSizeZ * _dx)) {
269 std::cout << "ERROR LBCouetteSolver::getVelocity(): Position " << pos << " out of range!" << std::endl;
270 std::cout << "domainOffset = " << domainOffset << std::endl;
271 std::cout << "_domainSizeX = " << _domainSizeX << std::endl;
272 std::cout << "_domainSizeY = " << _domainSizeY << std::endl;
273 std::cout << "_domainSizeZ = " << _domainSizeZ << std::endl;
274 std::cout << "_dx = " << _dx << std::endl;
275 exit(EXIT_FAILURE);
276 }
277 // compute index for respective cell (_dx+... for ghost cells); use coords
278 // to store local cell coordinates
279 for (unsigned int d = 0; d < 3; d++) {
280 coords[d] = (unsigned int)((_dx + pos[d] - domainOffset[d]) / _dx);
281 }
282 const int index = get(coords[0], coords[1], coords[2]);
284 // extract and scale velocity to "real"=MD units
285 for (int d = 0; d < 3; d++) {
286 vel[d] = _dx / _dt * _vel[3 * index + d];
287 }
288#if (COUPLING_MD_DEBUG == COUPLING_MD_YES)
289 std::cout << "Position " << pos << " corresponds to cell: " << coords << "; vel=" << vel << std::endl;
290#endif
291 return vel;
292 }
293
297 double getDensity(tarch::la::Vector<3, double> pos) const override {
300 // check pos-data for process locality (todo: put this in debug mode in
301 // future releases)
302 if ((pos[0] < domainOffset[0]) || (pos[0] > domainOffset[0] + _domainSizeX * _dx) || (pos[1] < domainOffset[1]) ||
303 (pos[1] > domainOffset[1] + _domainSizeY * _dx) || (pos[2] < domainOffset[2]) || (pos[2] > domainOffset[2] + _domainSizeZ * _dx)) {
304 std::cout << "ERROR LBCouetteSolver::getDensity(): Position " << pos << " out of range!" << std::endl;
305 exit(EXIT_FAILURE);
306 }
307 // compute index for respective cell (_dx+... for ghost cells); use coords
308 // to store local cell coordinates
309 for (unsigned int d = 0; d < 3; d++) {
310 coords[d] = (unsigned int)((_dx + pos[d] - domainOffset[d]) / _dx);
311 }
312 const int index = get(coords[0], coords[1], coords[2]);
313 return _density[index];
314 }
315
318 virtual void setWallVelocity(const tarch::la::Vector<3, double> wallVelocity) override { _wallVelocity = (_dt / _dx) * wallVelocity; }
319
323
324 std::unique_ptr<State> getState() override {
325 computeDensityAndVelocityEverywhere();
326 if (skipRank())
327 return std::make_unique<LBCouetteSolverState>(0);
328 return std::make_unique<LBCouetteSolverState>(_pdfsize, _pdf1);
329 }
330
331 void setState(const std::unique_ptr<State>& input, int cycle) override {
332 if (skipRank())
333 return;
334
335 const LBCouetteSolverState* state = dynamic_cast<const LBCouetteSolverState*>(input.get());
336
337#if (COUPLING_MD_ERROR == COUPLING_MD_YES)
338 if (state == nullptr) {
339 std::cout << "ERROR LBCouetteSolver setState() wrong state type" << std::endl;
340 exit(EXIT_FAILURE);
341 }
342#endif
343
344 std::copy(state->getData(), state->getData() + _pdfsize, _pdf1);
345 computeDensityAndVelocityEverywhere();
346
347 _counter = cycle;
348 }
349
350 std::unique_ptr<State> operator()(const std::unique_ptr<State>& input, int cycle) override {
351 setState(input, cycle);
352
353#if (COUPLING_MD_ERROR == COUPLING_MD_YES)
354 if (_mode != Mode::supervising) {
355 std::cout << "ERROR LBCouetteSolver operator() called but not in supervising mode" << std::endl;
356 exit(EXIT_FAILURE);
357 }
358#endif
359
360 advance(_dt_pint);
361 return getState();
362 }
363
364 Mode getMode() const override { return _mode; }
365
372 std::unique_ptr<PintableMacroSolver> getSupervisor(int num_cycles, double visc_multiplier) const override {
373#if (COUPLING_MD_ERROR == COUPLING_MD_YES)
374 if (_mode == Mode::supervising) {
375 std::cout << "ERROR LBCouetteSolver getSupervisor(): already in supervising mode" << std::endl;
376 exit(EXIT_FAILURE);
377 }
378#endif
379
380 int numThreads = 1;
381#if defined(_OPENMP)
382 numThreads = omp_get_num_threads();
383#endif
384
385 auto res = std::make_unique<LBCouetteSolver>(_channelheight, _wallVelocity * _dx / _dt, _kinVisc * visc_multiplier, _dx, _dt, _plotEveryTimestep,
386 _plotAverageVelocity, _filestem + std::string("_supervising"), _processes, numThreads, _scen);
387
388 res->_mode = Mode::supervising;
389 res->_dt_pint = _dt * num_cycles;
390
391 return res;
392 }
393
394 void print(std::ostream& os) const override {
395 if (_mode == Mode::supervising)
396 os << "<LBCouetteSolver instance in supervising mode >";
397 if (_mode == Mode::coupling)
398 os << "<LBCouetteSolver instance in coupling mode >";
399 }
400
401 double get_avg_vel(const std::unique_ptr<State>& state) const override {
402 if (skipRank())
403 return 0;
404 double vel[3];
405 double density;
406 double res[3]{0, 0, 0};
407 for (int i = 0; i < _pdfsize; i += 19) {
408 LBCouetteSolver::computeDensityAndVelocity(vel, density, state->getData() + i);
409 res[0] += vel[0];
410 res[1] += vel[1];
411 res[2] += vel[2];
412 }
413 if (_pdfsize > 0) {
414 res[0] /= (_pdfsize / 19);
415 res[1] /= (_pdfsize / 19);
416 res[2] /= (_pdfsize / 19);
417 }
418 return std::sqrt(res[0] * res[0] + res[1] * res[1] + res[2] * res[2]);
419 }
420
421 double get_avg_velX(const std::unique_ptr<State>& state) const {
422 if (skipRank())
423 return 0;
424 double vel[3];
425 double density;
426 double res{0};
427 for (int i = 0; i < _pdfsize; i += 19) {
428 LBCouetteSolver::computeDensityAndVelocity(vel, density, state->getData() + i);
429 res += vel[0];
430 }
431 if (_pdfsize > 0) {
432 res /= (_pdfsize / 19);
433 }
434 return res;
435 }
436
437private:
438 Mode _mode;
439 double _dt_pint;
440
441 void computeDensityAndVelocityEverywhere() {
442 if (skipRank())
443 return;
444 for (int z = 1; z < _domainSizeZ + 1; z++) {
445 for (int y = 1; y < _domainSizeY + 1; y++) {
446 for (int x = 1; x < _domainSizeX + 1; x++) {
447 const int index = get(x, y, z);
448 const int pI = 19 * index;
449 double* vel = &_vel[3 * index];
450 computeDensityAndVelocity(vel, _density[index], &_pdf1[pI]);
451 }
452 }
453 }
454 }
455
456 void plot_avg_vel() {
458 return;
459
460 int rank = 0;
461#if (COUPLING_MD_PARALLEL == COUPLING_MD_YES)
462 MPI_Comm_rank(coupling::indexing::IndexingService<3>::getInstance().getComm(), &rank);
463#endif
464 std::stringstream ss;
465 ss << _filestem << "_r" << rank;
466 if (_scen != nullptr) {
467 auto ts = _scen->getTimeIntegrationService();
468 if (ts != nullptr) {
469 if (ts->isPintEnabled())
470 ss << "_i" << ts->getIteration();
471 }
472 }
473 ss << ".csv";
474 std::string filename = ss.str();
475 std::ofstream file(filename.c_str(), _counter == 0 ? std::ofstream::out : std::ofstream::app);
476 if (!file.is_open()) {
477 std::cout << "ERROR LBCouetteSolver::plot_avg_vel(): Could not open file " << filename << "!" << std::endl;
478 exit(EXIT_FAILURE);
479 }
480
481 if (_counter == 0) {
482 file << "coupling_cycle ; avg_vel ; avg_velX" << std::endl;
483 }
484
485 std::unique_ptr<State> s = std::make_unique<LBCouetteSolverState>(_pdfsize, _pdf1);
486 double vel = get_avg_vel(s);
487 double velX = get_avg_velX(s);
488 file << _counter << " ; " << vel << " ; " << velX << std::endl;
489 file.close();
490 }
491
495
499#pragma omp parallel for
500 for (int z = 1; z < _domainSizeZ + 1; z++) {
501 for (int y = 1; y < _domainSizeY + 1; y++) {
502 for (int x = 1; x < _domainSizeX + 1; x++) {
503 const int index = get(x, y, z);
504 if (_flag[index] == FLUID) {
505 stream(index);
506 collide(index, x, y, z);
507 }
508 }
509 }
510 }
511 // swap fields
512 double* swap = _pdf1;
513 _pdf1 = _pdf2;
514 _pdf2 = swap;
515 }
516
518 void stream(int index) {
519 const int pI = 19 * index;
520 for (int q = 0; q < 9; q++) {
521 const int nb = 19 * (_C[q][0] + _C[q][1] * _xO + _C[q][2] * _yO);
522 _pdf2[pI + q] = _pdf1[pI + q - nb];
523 _pdf2[pI + 18 - q] = _pdf1[pI + 18 - q + nb];
524 }
525 _pdf2[pI + 9] = _pdf1[pI + 9];
526 }
527
529 void collide(int index, int x, int y, int z) {
530 // index of start of cell-local pdfs in AoS
531 const int pI = 19 * index;
532 // compute and store density, velocity
533 double* vel = &_vel[3 * index];
534 computeDensityAndVelocity(vel, _density[index], &_pdf2[pI]);
535 // collide (BGK); always handle pdfs no. q and inv(q)=18-q in one step
536 const double u2 = 1.0 - 1.5 * (vel[0] * vel[0] + vel[1] * vel[1] + vel[2] * vel[2]);
537 // pdf 0,18
538 double cu = -vel[1] - vel[2];
539 int nb = -_xO - _yO;
540 double feq = _W[0] * _density[index] * (u2 + 3.0 * cu + 4.5 * cu * cu);
541 _pdf2[pI] -= _omega * (_pdf2[pI] - feq);
542 boundary(_pdf2, pI, x, y, z, 0, _flag[index + nb], pI + 19 * nb);
543 feq -= 6.0 * _W[0] * _density[index] * cu;
544 _pdf2[pI + 18] -= _omega * (_pdf2[pI + 18] - feq);
545 boundary(_pdf2, pI, x, y, z, 18, _flag[index - nb], pI - 19 * nb);
546 // pdf 1,17
547 cu = -vel[0] - vel[2];
548 nb = -1 - _yO;
549 feq = _W[1] * _density[index] * (u2 + 3.0 * cu + 4.5 * cu * cu);
550 _pdf2[pI + 1] -= _omega * (_pdf2[pI + 1] - feq);
551 boundary(_pdf2, pI, x, y, z, 1, _flag[index + nb], pI + 19 * nb);
552 feq -= 6.0 * _W[1] * _density[index] * cu;
553 _pdf2[pI + 17] -= _omega * (_pdf2[pI + 17] - feq);
554 boundary(_pdf2, pI, x, y, z, 17, _flag[index - nb], pI - 19 * nb);
555 // pdf 2,16
556 cu = -vel[2];
557 nb = -_yO;
558 feq = _W[2] * _density[index] * (u2 + 3.0 * cu + 4.5 * cu * cu);
559 _pdf2[pI + 2] -= _omega * (_pdf2[pI + 2] - feq);
560 boundary(_pdf2, pI, x, y, z, 2, _flag[index + nb], pI + 19 * nb);
561 feq -= 6.0 * _W[2] * _density[index] * cu;
562 _pdf2[pI + 16] -= _omega * (_pdf2[pI + 16] - feq);
563 boundary(_pdf2, pI, x, y, z, 16, _flag[index - nb], pI - 19 * nb);
564 // pdf 3,15
565 cu = vel[0] - vel[2];
566 nb = 1 - _yO;
567 feq = _W[3] * _density[index] * (u2 + 3.0 * cu + 4.5 * cu * cu);
568 _pdf2[pI + 3] -= _omega * (_pdf2[pI + 3] - feq);
569 boundary(_pdf2, pI, x, y, z, 3, _flag[index + nb], pI + 19 * nb);
570 feq -= 6.0 * _W[3] * _density[index] * cu;
571 _pdf2[pI + 15] -= _omega * (_pdf2[pI + 15] - feq);
572 boundary(_pdf2, pI, x, y, z, 15, _flag[index - nb], pI - 19 * nb);
573 // pdf 4,14
574 cu = vel[1] - vel[2];
575 nb = _xO - _yO;
576 feq = _W[4] * _density[index] * (u2 + 3.0 * cu + 4.5 * cu * cu);
577 _pdf2[pI + 4] -= _omega * (_pdf2[pI + 4] - feq);
578 boundary(_pdf2, pI, x, y, z, 4, _flag[index + nb], pI + 19 * nb);
579 feq -= 6.0 * _W[4] * _density[index] * cu;
580 _pdf2[pI + 14] -= _omega * (_pdf2[pI + 14] - feq);
581 boundary(_pdf2, pI, x, y, z, 14, _flag[index - nb], pI - 19 * nb);
582 // pdf 5,13
583 cu = -vel[0] - vel[1];
584 nb = -1 - _xO;
585 feq = _W[5] * _density[index] * (u2 + 3.0 * cu + 4.5 * cu * cu);
586 _pdf2[pI + 5] -= _omega * (_pdf2[pI + 5] - feq);
587 boundary(_pdf2, pI, x, y, z, 5, _flag[index + nb], pI + 19 * nb);
588 feq -= 6.0 * _W[5] * _density[index] * cu;
589 _pdf2[pI + 13] -= _omega * (_pdf2[pI + 13] - feq);
590 boundary(_pdf2, pI, x, y, z, 13, _flag[index - nb], pI - 19 * nb);
591 // pdf 6,12
592 cu = -vel[1];
593 nb = -_xO;
594 feq = _W[6] * _density[index] * (u2 + 3.0 * cu + 4.5 * cu * cu);
595 _pdf2[pI + 6] -= _omega * (_pdf2[pI + 6] - feq);
596 boundary(_pdf2, pI, x, y, z, 6, _flag[index + nb], pI + 19 * nb);
597 feq -= 6.0 * _W[6] * _density[index] * cu;
598 _pdf2[pI + 12] -= _omega * (_pdf2[pI + 12] - feq);
599 boundary(_pdf2, pI, x, y, z, 12, _flag[index - nb], pI - 19 * nb);
600 // pdf 7,11
601 cu = vel[0] - vel[1];
602 nb = 1 - _xO;
603 feq = _W[7] * _density[index] * (u2 + 3.0 * cu + 4.5 * cu * cu);
604 _pdf2[pI + 7] -= _omega * (_pdf2[pI + 7] - feq);
605 boundary(_pdf2, pI, x, y, z, 7, _flag[index + nb], pI + 19 * nb);
606 feq -= 6.0 * _W[7] * _density[index] * cu;
607 _pdf2[pI + 11] -= _omega * (_pdf2[pI + 11] - feq);
608 boundary(_pdf2, pI, x, y, z, 11, _flag[index - nb], pI - 19 * nb);
609 // pdf 8,10
610 cu = -vel[0];
611 nb = -1;
612 feq = _W[8] * _density[index] * (u2 + 3.0 * cu + 4.5 * cu * cu);
613 _pdf2[pI + 8] -= _omega * (_pdf2[pI + 8] - feq);
614 boundary(_pdf2, pI, x, y, z, 8, _flag[index + nb], pI + 19 * nb);
615 feq -= 6.0 * _W[8] * _density[index] * cu;
616 _pdf2[pI + 10] -= _omega * (_pdf2[pI + 10] - feq);
617 boundary(_pdf2, pI, x, y, z, 10, _flag[index - nb], pI - 19 * nb);
618 // pdf 9
619 _pdf2[pI + 9] -= _omega * (_pdf2[pI + 9] - _W[9] * _density[index] * u2);
620 }
621
631 void boundary(double* const pdf, int index, int x, int y, int z, int q, const Flag& flag, int nbIndex) {
632 if (flag != FLUID) {
633 if (flag == NO_SLIP) {
634 // half-way bounce back
635 pdf[nbIndex + 18 - q] = pdf[index + q];
636 } else if (flag == MOVING_WALL) {
637 // half-way bounce back + moving wall acceleration (only x-direction for
638 // wall supported at the moment)
639 pdf[nbIndex + 18 - q] =
640 pdf[index + q] - 6.0 * _W[q] * _density[index / 19] * (_C[q][0] * _wallVelocity[0] + _C[q][1] * _wallVelocity[1] + _C[q][2] * _wallVelocity[2]);
641 } else if (flag == PERIODIC) {
642 // periodic treatment
643 int target[3] = {x, y, z};
644 if (target[0] + _C[q][0] == 0) {
645 target[0] = _domainSizeX + 1;
646 } else if (target[0] + _C[q][0] == _domainSizeX + 1) {
647 target[0] = 0;
648 }
649 if (target[1] + _C[q][1] == 0) {
650 target[1] = _domainSizeY + 1;
651 } else if (target[1] + _C[q][1] == _domainSizeY + 1) {
652 target[1] = 0;
653 }
654 if (target[2] + _C[q][2] == 0) {
655 target[2] = _domainSizeZ + 1;
656 } else if (target[2] + _C[q][2] == _domainSizeZ + 1) {
657 target[2] = 0;
658 }
659 const int periodicNb = target[0] + (_domainSizeX + 2) * (target[1] + (_domainSizeY + 2) * target[2]);
660 pdf[19 * periodicNb + q] = pdf[index + q];
661 }
662 }
663 }
664
669 static void computeDensityAndVelocity(double* const vel, double& density, const double* const pdf) {
670 vel[0] = -(pdf[1] + pdf[5] + pdf[8] + pdf[11] + pdf[15]);
671 density = pdf[3] + pdf[7] + pdf[10] + pdf[13] + pdf[17];
672 vel[1] = (pdf[4] + pdf[11] + pdf[12] + pdf[13] + pdf[18]) - (pdf[0] + pdf[5] + pdf[6] + pdf[7] + pdf[14]);
673 vel[0] = density + vel[0];
674 density = density + pdf[0] + pdf[1] + pdf[2] + pdf[4] + pdf[5] + pdf[6] + pdf[8] + pdf[9] + pdf[11] + pdf[12] + pdf[14] + pdf[15] + pdf[16] + pdf[18];
675 vel[2] = (pdf[14] + pdf[15] + pdf[16] + pdf[17] + pdf[18]) - (pdf[0] + pdf[1] + pdf[2] + pdf[3] + pdf[4]);
676 vel[0] = vel[0] / density;
677 vel[1] = vel[1] / density;
678 vel[2] = vel[2] / density;
679 }
680
695 void communicatePart(double* pdf, double* sendBuffer, double* recvBuffer, NbFlag nbFlagTo, NbFlag nbFlagFrom, tarch::la::Vector<3, int> startSend,
697#if (COUPLING_MD_PARALLEL == COUPLING_MD_YES)
698 // directions that point to LEFT/RIGHT,... -> same ordering as enums!
699 const int directions[6][5] = {{1, 5, 8, 11, 15}, {3, 7, 10, 13, 17}, {4, 11, 12, 13, 18}, {0, 5, 6, 7, 14}, {0, 1, 2, 3, 4}, {14, 15, 16, 17, 18}};
700 MPI_Request requests[2];
701 MPI_Status status[2];
703 tarch::la::Vector<2, int> domainSize;
704 // find out plane coordinates
705 if (nbFlagTo == LEFT || nbFlagTo == RIGHT) {
706 plane[0] = 1;
707 plane[1] = 2;
708 domainSize[0] = _domainSizeY;
709 domainSize[1] = _domainSizeZ;
710 } else if (nbFlagTo == FRONT || nbFlagTo == BACK) {
711 plane[0] = 0;
712 plane[1] = 2;
713 domainSize[0] = _domainSizeX;
714 domainSize[1] = _domainSizeZ;
715 } else if (nbFlagTo == TOP || nbFlagTo == BOTTOM) {
716 plane[0] = 0;
717 plane[1] = 1;
718 domainSize[0] = _domainSizeX;
719 domainSize[1] = _domainSizeY;
720 } else {
721 std::cout << "ERROR LBCouetteSolver::communicatePart: d >2 or d < 0!" << std::endl;
722 exit(EXIT_FAILURE);
723 }
724 // extract data and write to send buffer
726 for (coords[2] = startSend[2]; coords[2] < endSend[2]; coords[2]++) {
727 for (coords[1] = startSend[1]; coords[1] < endSend[1]; coords[1]++) {
728 for (coords[0] = startSend[0]; coords[0] < endSend[0]; coords[0]++) {
729 for (int q = 0; q < 5; q++) {
730 sendBuffer[q + 5 * getParBuf(coords[plane[0]], coords[plane[1]], domainSize[0], domainSize[1])] =
731 pdf[directions[nbFlagTo][q] + 19 * get(coords[0], coords[1], coords[2])];
732 }
733 }
734 }
735 }
736 // send and receive data
737 MPI_Irecv(recvBuffer, (domainSize[0] + 2) * (domainSize[1] + 2) * 5, MPI_DOUBLE, _parallelNeighbours[nbFlagFrom], 1000,
738 coupling::indexing::IndexingService<3>::getInstance().getComm(), &requests[0]);
739 MPI_Isend(sendBuffer, (domainSize[0] + 2) * (domainSize[1] + 2) * 5, MPI_DOUBLE, _parallelNeighbours[nbFlagTo], 1000,
740 coupling::indexing::IndexingService<3>::getInstance().getComm(), &requests[1]);
741 MPI_Waitall(2, requests, status);
742 // write data back to pdf field
743 if (_parallelNeighbours[nbFlagFrom] != MPI_PROC_NULL) {
744 for (coords[2] = startRecv[2]; coords[2] < endRecv[2]; coords[2]++) {
745 for (coords[1] = startRecv[1]; coords[1] < endRecv[1]; coords[1]++) {
746 for (coords[0] = startRecv[0]; coords[0] < endRecv[0]; coords[0]++) {
747 for (int q = 0; q < 5; q++) {
748 if (_flag[get(coords[0], coords[1], coords[2])] == PARALLEL_BOUNDARY) {
749 pdf[directions[nbFlagTo][q] + 19 * get(coords[0], coords[1], coords[2])] =
750 recvBuffer[q + 5 * getParBuf(coords[plane[0]], coords[plane[1]], domainSize[0], domainSize[1])];
751 }
752 }
753 }
754 }
755 }
756 }
757#endif
758 }
759
762 void communicate() {
763#if (COUPLING_MD_PARALLEL == COUPLING_MD_YES)
764 // send from right to left
768 // send from left to right
772 // send from back to front
776 // send from front to back
780 // send from top to bottom
784 // send from bottom to top
788#endif
789 }
790
792 const double _omega;
795 int _pdfsize{0};
797 double* _pdf1{NULL};
799 double* _pdf2{NULL};
801 const int _C[19][3]{{0, -1, -1}, {-1, 0, -1}, {0, 0, -1}, {1, 0, -1}, {0, 1, -1}, {-1, -1, 0}, {0, -1, 0}, {1, -1, 0}, {-1, 0, 0}, {0, 0, 0},
802 {1, 0, 0}, {-1, 1, 0}, {0, 1, 0}, {1, 1, 0}, {0, -1, 1}, {-1, 0, 1}, {0, 0, 1}, {1, 0, 1}, {0, 1, 1}};
803
804 const double _W[19]{1.0 / 36.0, 1.0 / 36.0, 1.0 / 18.0, 1.0 / 36.0, 1.0 / 36.0, 1.0 / 36.0, 1.0 / 18.0, 1.0 / 36.0, 1.0 / 18.0, 1.0 / 3.0,
805 1.0 / 18.0, 1.0 / 36.0, 1.0 / 18.0, 1.0 / 36.0, 1.0 / 36.0, 1.0 / 36.0, 1.0 / 18.0, 1.0 / 36.0, 1.0 / 36.0};
806
808};
809
810#endif // _MOLECULARDYNAMICS_COUPLING_SOLVERS_LBCOUETTESOLVER_H_
Definition Scenario.h:19
defines the cell type with cell-averaged quantities only (no linked cells).
Definition CouplingCell.h:29
const tarch::la::Vector< dim, double > & getMacroscopicMomentum() const
Definition CouplingCell.h:64
const double & getMacroscopicMass() const
Definition CouplingCell.h:58
provides access to coupling cells, which may belong to different indexing domains
Definition FlexibleCellContainer.h:30
value_T get() const
Definition CellIndex.h:138
Definition PintableMacroSolver.h:67
Definition PintableMacroSolver.h:30
Definition LBCouetteSolver.h:21
int getSizeBytes() const override
Definition LBCouetteSolver.h:31
std::unique_ptr< State > operator+(const State &rhs) override
const double * getData() const override
Definition LBCouetteSolver.h:44
void print(std::ostream &os) const override
Definition LBCouetteSolver.h:46
std::unique_ptr< State > operator-(const State &rhs) override
bool operator==(const State &rhs) const override
Definition LBCouetteSolver.h:36
double * getData() override
Definition LBCouetteSolver.h:43
implements a three-dimensional Lattice-Boltzmann Couette flow solver.
Definition LBCouetteSolver.h:56
static void computeDensityAndVelocity(double *const vel, double &density, const double *const pdf)
refers to the LB method; computes density and velocity on pdf
Definition LBCouetteSolver.h:669
std::unique_ptr< State > operator()(const std::unique_ptr< State > &input, int cycle) override
Definition LBCouetteSolver.h:350
std::unique_ptr< PintableMacroSolver > getSupervisor(int num_cycles, double visc_multiplier) const override
Definition LBCouetteSolver.h:372
void communicatePart(double *pdf, double *sendBuffer, double *recvBuffer, NbFlag nbFlagTo, NbFlag nbFlagFrom, tarch::la::Vector< 3, int > startSend, tarch::la::Vector< 3, int > endSend, tarch::la::Vector< 3, int > startRecv, tarch::la::Vector< 3, int > endRecv)
Definition LBCouetteSolver.h:695
std::unique_ptr< State > getState() override
Definition LBCouetteSolver.h:324
double get_avg_vel(const std::unique_ptr< State > &state) const override
Definition LBCouetteSolver.h:401
const double _omega
relaxation frequency
Definition LBCouetteSolver.h:792
const int _C[19][3]
lattice velocities
Definition LBCouetteSolver.h:801
void setMDBoundaryValues(coupling::datastructures::FlexibleCellContainer< 3 > &md2macroBuffer) override
applies the values received from the MD-solver within the conntinuum solver
Definition LBCouetteSolver.h:199
double * _pdf1
partical distribution function field
Definition LBCouetteSolver.h:797
void collide(int index, int x, int y, int z)
Definition LBCouetteSolver.h:529
void boundary(double *const pdf, int index, int x, int y, int z, int q, const Flag &flag, int nbIndex)
takes care of the correct boundary treatment for the LB method
Definition LBCouetteSolver.h:631
Mode getMode() const override
Definition LBCouetteSolver.h:364
const double _W[19]
lattice weights
Definition LBCouetteSolver.h:804
void communicate()
comunicates the boundary field data between the different processes
Definition LBCouetteSolver.h:762
tarch::la::Vector< 3, double > getVelocity(tarch::la::Vector< 3, double > pos) const override
returns velocity at a certain position
Definition LBCouetteSolver.h:262
void advance(double dt) override
advances one time step dt in time and triggers vtk plot if required
Definition LBCouetteSolver.h:175
virtual ~LBCouetteSolver()
a simple destructor
Definition LBCouetteSolver.h:124
virtual void setWallVelocity(const tarch::la::Vector< 3, double > wallVelocity) override
changes the velocity at the moving wall (z=0)
Definition LBCouetteSolver.h:318
tarch::la::Vector< 3, double > _wallVelocity
velocity of moving wall of Couette flow
Definition LBCouetteSolver.h:794
const bool _plotAverageVelocity
enables avg_vel CSV output
Definition LBCouetteSolver.h:807
void collidestream()
collide-stream algorithm for the Lattice-Boltzmann method
Definition LBCouetteSolver.h:498
void setState(const std::unique_ptr< State > &input, int cycle) override
Definition LBCouetteSolver.h:331
void print(std::ostream &os) const override
Definition LBCouetteSolver.h:394
double * _pdf2
partial distribution function field (stores the old time step)
Definition LBCouetteSolver.h:799
LBCouetteSolver(const double channelheight, tarch::la::Vector< 3, double > wallVelocity, const double kinVisc, const double dx, const double dt, const int plotEveryTimestep, const bool plotAverageVelocity, const std::string filestem, const tarch::la::Vector< 3, unsigned int > processes, const unsigned int numThreads=1, const Scenario *scen=nullptr)
a simple constructor
Definition LBCouetteSolver.h:73
void stream(int index)
the stream part of the LB algorithm (from pdf1 to pdf2)
Definition LBCouetteSolver.h:518
double getDensity(tarch::la::Vector< 3, double > pos) const override
returns density at a certain position
Definition LBCouetteSolver.h:297
is a virtual base class for the interface for a numerical fluid solver for the Couette scenario
Definition NumericalSolver.h:33
const int _domainSizeX
domain size in x-direction
Definition NumericalSolver.h:554
double * _recvBufferY
buffer to receive data from from front/back neighbour
Definition NumericalSolver.h:584
double * _sendBufferX
buffer to send data from left/right to right/left neighbour
Definition NumericalSolver.h:578
const double _channelheight
the height and width of the channel in z and y direction
Definition NumericalSolver.h:539
const double _kinVisc
kinematic viscosity of the fluid
Definition NumericalSolver.h:545
int _counter
time step counter
Definition NumericalSolver.h:569
int get(int i) const
returns i and performs checks in debug mode
Definition NumericalSolver.h:360
const int _yO
offset for z-direction
Definition NumericalSolver.h:593
double * _recvBufferZ
buffer to receive data from from top/buttom neighbour
Definition NumericalSolver.h:588
const int _avgDomainSizeZ
avg. domain size in MPI-parallel simulation in z-direction
Definition NumericalSolver.h:564
int getParBuf(int x, int y, int lengthx, int lengthy) const
returns index in 2D parallel buffer with buffer dimensions lengthx+2,lengthy+2. Performs checks in de...
Definition NumericalSolver.h:409
double * _sendBufferY
buffer to send data from front/back to front/back neighbour
Definition NumericalSolver.h:582
const int _xO
offset for y-direction (lexicographic grid ordering)
Definition NumericalSolver.h:591
double * _sendBufferZ
buffer to send data from top/buttom to top/buttom neighbour
Definition NumericalSolver.h:586
NumericalSolver(const double channelheight, const double dx, const double dt, const double kinVisc, const int plotEveryTimestep, const std::string filestem, const tarch::la::Vector< 3, unsigned int > processes, const Scenario *scen=nullptr)
a simple constructor
Definition NumericalSolver.h:46
Flag * _flag
flag field
Definition NumericalSolver.h:575
const int _avgDomainSizeY
avg. domain size in MPI-parallel simulation in y-direction
Definition NumericalSolver.h:562
const tarch::la::Vector< 3, unsigned int > _coords
coordinates of this process (=1,1,1, unless parallel run of the solver )
Definition NumericalSolver.h:567
const int _domainSizeY
domain size in y-direction
Definition NumericalSolver.h:556
const double _dt
time step
Definition NumericalSolver.h:543
tarch::la::Vector< 3, unsigned int > _processes
domain decomposition on MPI rank basis; total number is given by multipling all entries
Definition NumericalSolver.h:548
double * _density
density field
Definition NumericalSolver.h:573
tarch::la::Vector< 6, int > _parallelNeighbours
neighbour ranks
Definition NumericalSolver.h:595
const std::string _filestem
file stem for vtk plot
Definition NumericalSolver.h:552
NbFlag
The flags are used on parallel boundaries to define in which direction the boundary goes.
Definition NumericalSolver.h:530
@ LEFT
a parallel boundary to the left
Definition NumericalSolver.h:531
@ RIGHT
a parallel boundary to the right
Definition NumericalSolver.h:532
@ BOTTOM
a parallel boundary to the bottom
Definition NumericalSolver.h:535
@ FRONT
a parallel boundary to the front
Definition NumericalSolver.h:534
@ TOP
a parallel boundary to the top
Definition NumericalSolver.h:536
@ BACK
a parallel boundary to the back
Definition NumericalSolver.h:533
Flag
for every cell exists a flag entry, upon this is defined how the cell is handled
Definition NumericalSolver.h:519
@ MD_BOUNDARY
a cell on the boundary to md
Definition NumericalSolver.h:524
@ PERIODIC
a cell on a periodic boundary
Definition NumericalSolver.h:523
@ PARALLEL_BOUNDARY
a cell on a inner boundary of a splitted domain in a parallel run
Definition NumericalSolver.h:525
@ NO_SLIP
a cell on the no slip (non-moving) wall
Definition NumericalSolver.h:521
@ FLUID
a normal fluid cell
Definition NumericalSolver.h:520
@ MOVING_WALL
a cell on the moving wall
Definition NumericalSolver.h:522
const int _plotEveryTimestep
number of time steps between vtk plots
Definition NumericalSolver.h:550
double * _vel
velocity field
Definition NumericalSolver.h:571
const int _avgDomainSizeX
avg. domain size in MPI-parallel simulation in x-direction
Definition NumericalSolver.h:560
bool skipRank() const
returns true, if this rank is not of relevance for the LB simulation
Definition NumericalSolver.h:509
double * _recvBufferX
buffer to receive data from from left/right neighbour
Definition NumericalSolver.h:580
tarch::la::Vector< 3, int > _offset
offset of the md domain
Definition NumericalSolver.h:597
const double _dx
mesh size, dx=dy=dz
Definition NumericalSolver.h:541
const int _domainSizeZ
domain size in z-direction
Definition NumericalSolver.h:558
Definition Vector.h:25
all numerical solvers are defined in the namespace, and their interfaces
Definition CouetteSolver.h:14
everything necessary for coupling operations, is defined in here
Definition AdditiveMomentumInsertion.h:15