LCOV - code coverage report
Current view: top level - tools - Communicator.cpp (source / functions) Hit Total Coverage
Test: plumed test coverage Lines: 124 147 84.4 %
Date: 2025-03-25 09:33:27 Functions: 30 36 83.3 %

          Line data    Source code
       1             : /* +++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
       2             :    Copyright (c) 2012-2023 The plumed team
       3             :    (see the PEOPLE file at the root of the distribution for a list of names)
       4             : 
       5             :    See http://www.plumed.org for more information.
       6             : 
       7             :    This file is part of plumed, version 2.
       8             : 
       9             :    plumed is free software: you can redistribute it and/or modify
      10             :    it under the terms of the GNU Lesser General Public License as published by
      11             :    the Free Software Foundation, either version 3 of the License, or
      12             :    (at your option) any later version.
      13             : 
      14             :    plumed is distributed in the hope that it will be useful,
      15             :    but WITHOUT ANY WARRANTY; without even the implied warranty of
      16             :    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
      17             :    GNU Lesser General Public License for more details.
      18             : 
      19             :    You should have received a copy of the GNU Lesser General Public License
      20             :    along with plumed.  If not, see <http://www.gnu.org/licenses/>.
      21             : +++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++ */
      22             : #include "Communicator.h"
      23             : #include "Exception.h"
      24             : #include "AtomNumber.h"
      25             : #include <cstdlib>
      26             : #include <cstring>
      27             : #include <cstdio>
      28             : 
      29             : namespace PLMD {
      30             : 
      31          54 : bool Communicator::plumedHasMPI() {
      32             : #ifdef __PLUMED_HAS_MPI
      33          54 :   return true;
      34             : #else
      35             :   return false;
      36             : #endif
      37             : }
      38             : 
      39     1622824 : Communicator::Communicator()
      40             : #ifdef __PLUMED_HAS_MPI
      41     1622824 :   : communicator(MPI_COMM_SELF)
      42             : #endif
      43             : {
      44     1622824 : }
      45             : 
      46           0 : Communicator::Communicator(const Communicator&pc) {
      47           0 :   Set_comm(pc.communicator);
      48           0 : }
      49             : 
      50             : Communicator::Status Communicator::StatusIgnore;
      51             : 
      52           0 : Communicator& Communicator::operator=(const Communicator&pc) {
      53           0 :   if (this != &pc) {
      54           0 :     Set_comm(pc.communicator);
      55             :   }
      56           0 :   return *this;
      57             : }
      58             : 
      59     7786665 : int Communicator::Get_rank()const {
      60     7786665 :   int r=0;
      61             : #ifdef __PLUMED_HAS_MPI
      62     7786665 :   if(initialized()) {
      63     2784828 :     MPI_Comm_rank(communicator,&r);
      64             :   }
      65             : #endif
      66     7786665 :   return r;
      67             : }
      68             : 
      69     2984087 : int Communicator::Get_size()const {
      70     2984087 :   int s=1;
      71             : #ifdef __PLUMED_HAS_MPI
      72     2984087 :   if(initialized()) {
      73     1130465 :     MPI_Comm_size(communicator,&s);
      74             :   }
      75             : #endif
      76     2984087 :   return s;
      77             : }
      78             : 
      79        2448 : void Communicator::Set_comm(MPI_Comm c) {
      80             : #ifdef __PLUMED_HAS_MPI
      81        2448 :   if(initialized()) {
      82        1813 :     if(communicator!=MPI_COMM_SELF && communicator!=MPI_COMM_WORLD) {
      83           0 :       MPI_Comm_free(&communicator);
      84             :     }
      85        1813 :     if(c!=MPI_COMM_SELF) {
      86        1770 :       MPI_Comm_dup(c,&communicator);
      87             :     }
      88             :   }
      89             : #else
      90             :   (void) c;
      91             : #endif
      92        2448 : }
      93             : 
      94     3242344 : Communicator::~Communicator() {
      95             : #ifdef __PLUMED_HAS_MPI
      96     1622824 :   if(initialized() && communicator!=MPI_COMM_SELF && communicator!=MPI_COMM_WORLD) {
      97        2184 :     MPI_Comm_free(&communicator);
      98             :   }
      99             : #endif
     100     3242344 : }
     101             : 
     102        1257 : void Communicator::Set_comm(const TypesafePtr & val) {
     103             : #ifdef __PLUMED_HAS_MPI
     104        1257 :   plumed_massert(initialized(),"you are trying to use an MPI function, but MPI is not initialized");
     105        1257 :   if(val) {
     106        1257 :     Set_comm(*(const MPI_Comm*)val.get<const void*>());
     107             :   }
     108             : #else
     109             :   (void) val;
     110             :   plumed_merror("you are trying to use an MPI function, but PLUMED has been compiled without MPI support");
     111             : #endif
     112        1257 : }
     113             : 
     114           0 : void Communicator::Set_fcomm(const TypesafePtr & val) {
     115             : #ifdef __PLUMED_HAS_MPI
     116           0 :   plumed_massert(initialized(),"you are trying to use an MPI function, but MPI is not initialized");
     117           0 :   if(val) {
     118           0 :     MPI_Comm comm=MPI_Comm_f2c(*(const MPI_Fint*)val.get<const void*>());
     119           0 :     Set_comm(comm);
     120             :   }
     121             : #else
     122             :   (void) val;
     123             :   plumed_merror("you are trying to use an MPI function, but PLUMED has been compiled without MPI support");
     124             : #endif
     125           0 : }
     126             : 
     127           0 : void Communicator::Abort(int errorcode) {
     128             : #ifdef __PLUMED_HAS_MPI
     129           0 :   if(initialized()) {
     130           0 :     MPI_Abort(communicator,errorcode);
     131             :   }
     132             : #endif
     133           0 :   std::fprintf(stderr,"aborting with error code %d\n",errorcode);
     134           0 :   std::abort();
     135             : }
     136             : 
     137     5554050 : void Communicator::Bcast(Data data,int root) {
     138             : #if defined(__PLUMED_HAS_MPI)
     139     5554050 :   if(initialized()) {
     140     1851298 :     MPI_Bcast(data.pointer,data.size,data.type,root,communicator);
     141             :   }
     142             : #else
     143             :   (void) data;
     144             :   (void) root;
     145             : #endif
     146     5554050 : }
     147             : 
     148     4558840 : void Communicator::Sum(Data data) {
     149             : #if defined(__PLUMED_HAS_MPI)
     150     4558840 :   if(initialized()) {
     151     1982890 :     MPI_Allreduce(MPI_IN_PLACE,data.pointer,data.size,data.type,MPI_SUM,communicator);
     152             :   }
     153             : #else
     154             :   (void) data;
     155             : #endif
     156     4558840 : }
     157             : 
     158           3 : void Communicator::Prod(Data data) {
     159             : #if defined(__PLUMED_HAS_MPI)
     160           3 :   if(initialized()) {
     161           3 :     MPI_Allreduce(MPI_IN_PLACE,data.pointer,data.size,data.type,MPI_PROD,communicator);
     162             :   }
     163             : #else
     164             :   (void) data;
     165             : #endif
     166           3 : }
     167             : 
     168         155 : void Communicator::Max(Data data) {
     169             : #if defined(__PLUMED_HAS_MPI)
     170         155 :   if(initialized()) {
     171         155 :     MPI_Allreduce(MPI_IN_PLACE,data.pointer,data.size,data.type,MPI_MAX,communicator);
     172             :   }
     173             : #else
     174             :   (void) data;
     175             : #endif
     176         155 : }
     177             : 
     178           3 : void Communicator::Min(Data data) {
     179             : #if defined(__PLUMED_HAS_MPI)
     180           3 :   if(initialized()) {
     181           3 :     MPI_Allreduce(MPI_IN_PLACE,data.pointer,data.size,data.type,MPI_MIN,communicator);
     182             :   }
     183             : #else
     184             :   (void) data;
     185             : #endif
     186           3 : }
     187             : 
     188       15394 : Communicator::Request Communicator::Isend(ConstData data,int source,int tag) {
     189             :   Request req;
     190             : #ifdef __PLUMED_HAS_MPI
     191       15394 :   plumed_massert(initialized(),"you are trying to use an MPI function, but MPI is not initialized");
     192       15394 :   void*s=const_cast<void*>((const void*)data.pointer);
     193       15394 :   MPI_Isend(s,data.size,data.type,source,tag,communicator,&req.r);
     194             : #else
     195             :   (void) data;
     196             :   (void) source;
     197             :   (void) tag;
     198             :   plumed_merror("you are trying to use an MPI function, but PLUMED has been compiled without MPI support");
     199             : #endif
     200       15394 :   return req;
     201             : }
     202             : 
     203       20888 : void Communicator::Allgatherv(ConstData in,Data out,const int*recvcounts,const int*displs) {
     204       20888 :   void*s=const_cast<void*>((const void*)in.pointer);
     205       20888 :   void*r=const_cast<void*>((const void*)out.pointer);
     206             :   int*rc=const_cast<int*>(recvcounts);
     207             :   int*di=const_cast<int*>(displs);
     208             : #if defined(__PLUMED_HAS_MPI)
     209       20888 :   if(initialized()) {
     210       20887 :     if(s==NULL) {
     211             :       s=MPI_IN_PLACE;
     212             :     }
     213       20887 :     MPI_Allgatherv(s,in.size,in.type,r,rc,di,out.type,communicator);
     214             :   } else {
     215           1 :     plumed_assert(in.nbytes==out.nbytes);
     216           1 :     plumed_assert(in.size==out.size);
     217           1 :     plumed_assert(rc);
     218           1 :     plumed_assert(rc[0]==in.size);
     219           1 :     plumed_assert(di);
     220           1 :     if(s) {
     221           1 :       std::memcpy(static_cast<char*>(r)+displs[0]*in.nbytes,s,size_t(in.size)*in.nbytes);
     222             :     }
     223             :   }
     224             : #else
     225             :   plumed_assert(in.nbytes==out.nbytes);
     226             :   plumed_assert(in.size==out.size);
     227             :   plumed_assert(rc);
     228             :   plumed_assert(rc[0]==in.size);
     229             :   plumed_assert(di);
     230             :   if(s) {
     231             :     std::memcpy(static_cast<char*>(r)+displs[0]*in.nbytes,s,size_t(in.size)*in.nbytes);
     232             :   }
     233             : #endif
     234       20888 : }
     235             : 
     236        5850 : void Communicator::Allgather(ConstData in,Data out) {
     237        5850 :   void*s=const_cast<void*>((const void*)in.pointer);
     238        5850 :   void*r=const_cast<void*>((const void*)out.pointer);
     239             : #if defined(__PLUMED_HAS_MPI)
     240        5850 :   if(initialized()) {
     241        5849 :     if(s==NULL) {
     242             :       s=MPI_IN_PLACE;
     243             :     }
     244        5849 :     MPI_Allgather(s,in.size,in.type,r,out.size/Get_size(),out.type,communicator);
     245             :   } else {
     246           1 :     plumed_assert(in.nbytes==out.nbytes);
     247           1 :     plumed_assert(in.size==out.size);
     248           1 :     if(s) {
     249           1 :       std::memcpy(r,s,size_t(in.size)*in.nbytes);
     250             :     }
     251             :   }
     252             : #else
     253             :   plumed_assert(in.nbytes==out.nbytes);
     254             :   plumed_assert(in.size==out.size);
     255             :   if(s) {
     256             :     std::memcpy(r,s,size_t(in.size)*in.nbytes);
     257             :   }
     258             : #endif
     259        5850 : }
     260             : 
     261       15394 : void Communicator::Recv(Data data,int source,int tag,Status&status) {
     262             : #ifdef __PLUMED_HAS_MPI
     263       15394 :   plumed_massert(initialized(),"you are trying to use an MPI function, but MPI is not initialized");
     264       15394 :   if(&status==&StatusIgnore) {
     265        7759 :     MPI_Recv(data.pointer,data.size,data.type,source,tag,communicator,MPI_STATUS_IGNORE);
     266             :   } else {
     267        7635 :     MPI_Recv(data.pointer,data.size,data.type,source,tag,communicator,&status.s);
     268             :   }
     269             : #else
     270             :   (void) data;
     271             :   (void) source;
     272             :   (void) tag;
     273             :   (void) status;
     274             :   plumed_merror("you are trying to use an MPI function, but PLUMED has been compiled without MPI support");
     275             : #endif
     276       15394 : }
     277             : 
     278     5222715 : void Communicator::Barrier()const {
     279             : #ifdef __PLUMED_HAS_MPI
     280     5222715 :   if(initialized()) {
     281     1785320 :     MPI_Barrier(communicator);
     282             :   }
     283             : #endif
     284     5222715 : }
     285             : 
     286        1923 : MPI_Comm & Communicator::Get_comm() {
     287        1923 :   return communicator;
     288             : }
     289             : 
     290    27819624 : bool Communicator::initialized() {
     291             : #if defined(__PLUMED_HAS_MPI)
     292    27819624 :   int flag=0;
     293    27819624 :   MPI_Initialized(&flag);
     294    27819624 :   if(flag) {
     295             :     return true;
     296             :   } else {
     297    18194654 :     return false;
     298             :   }
     299             : #endif
     300             :   return false;
     301             : }
     302             : 
     303       14986 : void Communicator::Request::wait(Status&s) {
     304             : #ifdef __PLUMED_HAS_MPI
     305       14986 :   plumed_massert(initialized(),"you are trying to use an MPI function, but MPI is not initialized");
     306       14986 :   if(&s==&StatusIgnore) {
     307       14985 :     MPI_Wait(&r,MPI_STATUS_IGNORE);
     308             :   } else {
     309           1 :     MPI_Wait(&r,&s.s);
     310             :   }
     311             : #else
     312             :   (void) s;
     313             :   plumed_merror("you are trying to use an MPI function, but PLUMED has been compiled without MPI support");
     314             : #endif
     315       14986 : }
     316             : 
     317             : #ifdef __PLUMED_HAS_MPI
     318           0 : template<> MPI_Datatype Communicator::getMPIType<float>() {
     319           0 :   return MPI_FLOAT;
     320             : }
     321     4804553 : template<> MPI_Datatype Communicator::getMPIType<double>() {
     322     4804553 :   return MPI_DOUBLE;
     323             : }
     324       99734 : template<> MPI_Datatype Communicator::getMPIType<int>()   {
     325       99734 :   return MPI_INT;
     326             : }
     327         295 : template<> MPI_Datatype Communicator::getMPIType<char>()   {
     328         295 :   return MPI_CHAR;
     329             : }
     330       83699 : template<> MPI_Datatype Communicator::getMPIType<unsigned>()   {
     331       83699 :   return MPI_UNSIGNED;
     332             : }
     333          24 : template<> MPI_Datatype Communicator::getMPIType<AtomNumber>()   {
     334          24 :   return MPI_UNSIGNED;
     335             : }
     336     5216639 : template<> MPI_Datatype Communicator::getMPIType<long unsigned>()   {
     337     5216639 :   return MPI_UNSIGNED_LONG;
     338             : }
     339           6 : template<> MPI_Datatype Communicator::getMPIType<long long unsigned>() {
     340           6 :   return MPI_UNSIGNED_LONG_LONG;
     341             : }
     342           0 : template<> MPI_Datatype Communicator::getMPIType<long double>()   {
     343           0 :   return MPI_LONG_DOUBLE;
     344             : }
     345             : #else
     346             : template<> MPI_Datatype Communicator::getMPIType<float>() {
     347             :   return MPI_Datatype();
     348             : }
     349             : template<> MPI_Datatype Communicator::getMPIType<double>() {
     350             :   return MPI_Datatype();
     351             : }
     352             : template<> MPI_Datatype Communicator::getMPIType<int>() {
     353             :   return MPI_Datatype();
     354             : }
     355             : template<> MPI_Datatype Communicator::getMPIType<char>() {
     356             :   return MPI_Datatype();
     357             : }
     358             : template<> MPI_Datatype Communicator::getMPIType<unsigned>() {
     359             :   return MPI_Datatype();
     360             : }
     361             : template<> MPI_Datatype Communicator::getMPIType<AtomNumber>()   {
     362             :   return MPI_Datatype();
     363             : }
     364             : template<> MPI_Datatype Communicator::getMPIType<long unsigned>() {
     365             :   return MPI_Datatype();
     366             : }
     367             : template<> MPI_Datatype Communicator::getMPIType<long long unsigned>() {
     368             :   return MPI_Datatype();
     369             : }
     370             : template<> MPI_Datatype Communicator::getMPIType<long double>() {
     371             :   return MPI_Datatype();
     372             : }
     373             : #endif
     374             : 
     375         414 : void Communicator::Split(int color,int key,Communicator&pc)const {
     376             : #ifdef __PLUMED_HAS_MPI
     377         414 :   MPI_Comm_split(communicator,color,key,&pc.communicator);
     378             : #else
     379             :   (void) color;
     380             :   (void) key;
     381             :   (void) pc;
     382             :   plumed_merror("you are trying to use an MPI function, but PLUMED has been compiled without MPI support");
     383             : #endif
     384         414 : }
     385             : 
     386        7635 : int Communicator::Status::Get_count(MPI_Datatype type)const {
     387             :   int i;
     388             : #ifdef __PLUMED_HAS_MPI
     389        7635 :   plumed_massert(initialized(),"you are trying to use an MPI function, but MPI is not initialized");
     390        7635 :   MPI_Get_count(const_cast<MPI_Status*>(&s),type,&i);
     391             : #else
     392             :   i=0;
     393             :   plumed_merror("you are trying to use an MPI function, but PLUMED has been compiled without MPI support");
     394             : #endif
     395        7635 :   return i;
     396             : }
     397             : 
     398             : }
     399             : 

Generated by: LCOV version 1.16