Skip to content
Snippets Groups Projects
HeatTransfer.cpp 6.01 KiB
Newer Older
/*
 * heatTransfer.cpp
 *
 * Recreates heat_transfer.f90 (Fortran) in C++
 *
 * Created on: Feb 2017
 *     Author: Norbert Podhorszki
 *
 */
#include <mpi.h>

#include <stdexcept>
#include <memory>
#include <iostream>
#include <iomanip>
#include <string>
#include <math.h>


#include "HeatTransfer.h"


HeatTransfer::HeatTransfer( std::shared_ptr<Settings> settings )
: m_s{settings}
{
    m_T1 = new double*[ m_s->ndx+2 ];
    m_T1[0] = new double[ (m_s->ndx+2) * (m_s->ndy+2) ];
    m_T2 = new double*[ m_s->ndx+2 ];
    m_T2[0] = new double[ (m_s->ndx+2) * (m_s->ndy+2) ];
    for (int i = 1; i < m_s->ndx+2; i++)
    {
        m_T1[i] = m_T1[i-1] + m_s->ndy+2;
        m_T2[i] = m_T2[i-1] + m_s->ndy+2;
    }
    m_TCurrent = m_T1;
    m_TNext = m_T2;
}

HeatTransfer::~HeatTransfer()
{
    delete[] m_T1[0];
    delete[] m_T1;
    delete[] m_T2[0];
    delete[] m_T2;
}


void HeatTransfer::init(bool init_with_rank)
{
    if (init_with_rank)
    {
        for (int i = 0; i < m_s->ndx+2; i++)
            for (int j = 0; j < m_s->ndy+2; j++)
                m_T1[i][j] = m_s->rank;
    }
    else
    {
        const double hx = 2.0 * 4.0*atan(1.0)/m_s->ndx;
        const double hy = 2.0 * 4.0*atan(1.0)/m_s->ndy;

        double x, y;
        for (int i = 0; i < m_s->ndx+2; i++)
        {
            x = 0.0 + hx*(i-1);
            for (int j = 0; j < m_s->ndy+2; j++)
            {
                y = 0.0 + hy*(j-1);
                m_T1[i][j] = cos(8*x) + cos(6*x) - cos(4*x) + cos(2*x) - cos(x) +
                        sin(8*y) - sin(6*y) + sin(4*y) - sin(2*y) + sin(y);
            }
        }
    }
    m_TCurrent = m_T1;
    m_TNext = m_T2;
}

void HeatTransfer::printT(std::string message, MPI_Comm comm)
{
    int rank, size;
    int tag = 1;
    int token;
    MPI_Status status;
    MPI_Comm_rank( comm, &rank );
    MPI_Comm_size( comm, &size );
    if( rank > 0)
    {
        MPI_Recv(&token, 1, MPI_INT, rank-1, tag, comm, &status);
    }

    std::cout << "Rank " << rank << " " << message << std::endl;
    for (int i = 0; i < m_s->ndx+2; i++)
    {
        std::cout << "  T[" << i << "][] = ";
        for (int j = 0; j < m_s->ndy+2; j++)
        {
            std::cout << std::setw(6) << m_TCurrent[i][j];
        }
        std::cout << std::endl;
    }
    std::cout << std::flush << std::endl;

    if( rank < size-1)
    {
        MPI_Send(&token, 1, MPI_INT, rank+1, tag, comm);
    }
}


void HeatTransfer::switchCurrentNext()
{
    double **tmp = m_TCurrent;
    m_TCurrent = m_TNext;
    m_TNext = tmp;
}

void HeatTransfer::iterate()
{
    for( int i = 1; i <= m_s->ndx; ++i)
    {
        for( int j = 1; j <= m_s->ndy; ++j)
        {
            m_TNext[i][j] =
                    omega/4*(m_TCurrent[i-1][j] +
                             m_TCurrent[i+1][j] +
                             m_TCurrent[i][j-1] +
                             m_TCurrent[i][j+1]) +
                    (1.0-omega)*m_TCurrent[i][j];
        }
    }
    switchCurrentNext();
}

void HeatTransfer::heatEdges()
{
    // Heat the whole global edges
    if( m_s->posx==0 )
        for( int j = 0; j < m_s->ndy+2; ++j)
            m_TCurrent[0][j]= edgetemp;

    if( m_s->posx==m_s->npx-1 )
        for( int j = 0; j < m_s->ndy+2; ++j)
            m_TCurrent[m_s->ndx+1][j]= edgetemp;

    if (m_s->posy==0)
        for( int i = 0; i < m_s->ndx+2; ++i)
            m_TCurrent[i][0]= edgetemp;

    if (m_s->posy==m_s->npy-1)
        for( int i = 0; i < m_s->ndx+2; ++i)
            m_TCurrent[i][m_s->ndy+1]= edgetemp;
}

void HeatTransfer::exchange( MPI_Comm comm )
{
    // Exchange ghost cells, in the order left-right-up-down

    double * send_x = new double[m_s->ndx+2];
    double * recv_x = new double[m_s->ndx+2];

    // send to left + receive from right
    int tag = 1;
    MPI_Status status;
    if( m_s->rank_left >= 0 )
    {
        std::cout << "Rank " << m_s->rank << " send left to rank " <<  m_s->rank_left;
        for( int i = 0; i < m_s->ndx+2; ++i)
            send_x[i] = m_TCurrent[i][1];
         MPI_Send(send_x, m_s->ndx+2, MPI_REAL8, m_s->rank_left, tag, comm);
    }
    if( m_s->rank_right >= 0 )
    {
        std::cout << "Rank " << m_s->rank << " receive from right from rank " <<  m_s->rank_right;
        MPI_Recv(recv_x, m_s->ndx+2, MPI_REAL8, m_s->rank_right, tag, comm, &status);
        for( int i = 0; i < m_s->ndx+2; ++i)
            m_TCurrent[i][m_s->ndy+1] = recv_x[i];
    }

    // send to right + receive from left
    tag = 2;
    if( m_s->rank_right >= 0 )
    {
        std::cout << "Rank " << m_s->rank << " send right to rank " <<  m_s->rank_right;
        for( int i = 0; i < m_s->ndx+2; ++i)
            send_x[i] = m_TCurrent[i][m_s->ndy];
        MPI_Send(send_x, m_s->ndx+2, MPI_REAL8, m_s->rank_right, tag, comm);
    }
    if( m_s->rank_left >= 0 )
    {
        std::cout << "Rank " << m_s->rank << " receive from left from rank " <<  m_s->rank_left;
        MPI_Recv(recv_x, m_s->ndx+2, MPI_REAL8, m_s->rank_left, tag, comm, &status);
        for( int i = 0; i < m_s->ndx+2; ++i)
            m_TCurrent[i][0] = recv_x[i];
    }

    // send down + receive from above
    tag = 3;
    if( m_s->rank_down >= 0 )
    {
        std::cout << "Rank " << m_s->rank << " send down to rank " <<  m_s->rank_down;
        MPI_Send(m_TCurrent[m_s->ndx], m_s->ndy+2, MPI_REAL8, m_s->rank_down, tag, comm);
    }
    if ( m_s->rank_up >= 0 )
    {
        std::cout << "Rank " << m_s->rank << " receive from above from rank " <<  m_s->rank_up;
        MPI_Recv(m_TCurrent[0], m_s->ndy+2, MPI_REAL8, m_s->rank_up, tag, comm, &status);
    }

    // send up + receive from below
    tag = 4;
    if( m_s->rank_up >= 0 )
    {
        std::cout << "Rank " << m_s->rank << " send up to rank " <<  m_s->rank_up;
        MPI_Send(m_TCurrent[1], m_s->ndy+2, MPI_REAL8, m_s->rank_up, tag, comm);
    }
    if ( m_s->rank_down >= 0 )
    {
        std::cout << "Rank " << m_s->rank << " receive from below from rank " <<  m_s->rank_down;
        MPI_Recv(m_TCurrent[m_s->ndx+1], m_s->ndy+2, MPI_REAL8, m_s->rank_down, tag, comm, &status);
    }

    delete[] send_x;
    delete[] recv_x;
}