mpi_node.txx 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466
  1. /**
  2. * \file mpi_node.txx
  3. * \author Dhairya Malhotra, dhairya.malhotra@gmail.com
  4. * \date 12-11-2010
  5. * \brief This file contains the implementation of the class MPI_Node.
  6. */
  7. #include <cmath>
  8. #include <matrix.hpp>
  9. #include <mem_mgr.hpp>
  10. namespace pvfmm{
  11. template <class T>
  12. MPI_Node<T>::~MPI_Node(){
  13. }
  14. template <class T>
  15. void MPI_Node<T>::Initialize(TreeNode* parent_,int path2node_, TreeNode::NodeData* data_){
  16. TreeNode::Initialize(parent_,path2node_,data_);
  17. //Set node coordinates.
  18. Real_t coord_offset=((Real_t)1.0)/((Real_t)(((uint64_t)1)<<Depth()));
  19. if(!parent_){
  20. for(int j=0;j<dim;j++) coord[j]=0;
  21. }else if(parent_){
  22. int flag=1;
  23. for(int j=0;j<dim;j++){
  24. coord[j]=((MPI_Node<Real_t>*)parent_)->coord[j]+
  25. ((Path2Node() & flag)?coord_offset:0.0f);
  26. flag=flag<<1;
  27. }
  28. }
  29. //Initialize colleagues array.
  30. int n=(int)pow(3.0,Dim());
  31. for(int i=0;i<n;i++) colleague[i]=NULL;
  32. //Set MPI_Node specific data.
  33. typename MPI_Node<Real_t>::NodeData* mpi_data=dynamic_cast<typename MPI_Node<Real_t>::NodeData*>(data_);
  34. if(data_){
  35. max_pts =mpi_data->max_pts;
  36. pt_coord=mpi_data->pt_coord;
  37. pt_value=mpi_data->pt_value;
  38. }else if(parent){
  39. max_pts =((MPI_Node<T>*)parent)->max_pts;
  40. SetGhost(((MPI_Node<T>*)parent)->IsGhost());
  41. }
  42. }
  43. template <class T>
  44. void MPI_Node<T>::ClearData(){
  45. pt_coord.ReInit(0);
  46. pt_value.ReInit(0);
  47. }
  48. template <class T>
  49. MortonId MPI_Node<T>::GetMortonId(){
  50. assert(coord);
  51. Real_t s=0.25/(1UL<<MAX_DEPTH);
  52. return MortonId(coord[0]+s,coord[1]+s,coord[2]+s, Depth()); // TODO: Use interger coordinates instead of floating point.
  53. }
  54. template <class T>
  55. void MPI_Node<T>::SetCoord(MortonId& mid){
  56. assert(coord);
  57. mid.GetCoord(coord);
  58. depth=mid.GetDepth();
  59. }
  60. template <class T>
  61. TreeNode* MPI_Node<T>::NewNode(TreeNode* n_){
  62. MPI_Node<Real_t>* n=(n_?static_cast<MPI_Node<Real_t>*>(n_):new MPI_Node<Real_t>());
  63. n->max_pts=max_pts;
  64. return TreeNode::NewNode(n);
  65. }
  66. template <class T>
  67. bool MPI_Node<T>::SubdivCond(){
  68. int n=(1UL<<this->Dim());
  69. // Do not subdivide beyond max_depth
  70. if(this->Depth()>=this->max_depth-1) return false;
  71. if(!this->IsLeaf()){ // If has non-leaf children, then return true.
  72. for(int i=0;i<n;i++){
  73. MPI_Node<Real_t>* ch=static_cast<MPI_Node<Real_t>*>(this->Child(i));
  74. assert(ch); //ch==NULL should never happen
  75. if(!ch->IsLeaf() || ch->IsGhost()) return true;
  76. }
  77. }
  78. else{ // Do not refine ghost leaf nodes.
  79. if(this->IsGhost()) return false;
  80. }
  81. if(!this->IsLeaf()){
  82. size_t pt_vec_size=0;
  83. for(int i=0;i<n;i++){
  84. MPI_Node<Real_t>* ch=static_cast<MPI_Node<Real_t>*>(this->Child(i));
  85. pt_vec_size+=ch->pt_coord.Dim();
  86. }
  87. return pt_vec_size/Dim()>max_pts;
  88. }else{
  89. return pt_coord.Dim()/Dim()>max_pts;
  90. }
  91. }
  92. template <class T>
  93. void MPI_Node<T>::Subdivide(){
  94. if(!this->IsLeaf()) return;
  95. TreeNode::Subdivide();
  96. int nchld=(1UL<<this->Dim());
  97. if(!IsGhost()){ // Partition point coordinates and values.
  98. std::vector<Vector<Real_t>*> pt_coord;
  99. std::vector<Vector<Real_t>*> pt_value;
  100. std::vector<Vector<size_t>*> pt_scatter;
  101. this->NodeDataVec(pt_coord, pt_value, pt_scatter);
  102. std::vector<std::vector<Vector<Real_t>*> > chld_pt_coord(nchld);
  103. std::vector<std::vector<Vector<Real_t>*> > chld_pt_value(nchld);
  104. std::vector<std::vector<Vector<size_t>*> > chld_pt_scatter(nchld);
  105. for(size_t i=0;i<nchld;i++){
  106. static_cast<MPI_Node<Real_t>*>((MPI_Node<T>*)this->Child(i))
  107. ->NodeDataVec(chld_pt_coord[i], chld_pt_value[i], chld_pt_scatter[i]);
  108. }
  109. Real_t* c=this->Coord();
  110. Real_t s=pow(0.5,this->Depth()+1);
  111. for(size_t j=0;j<pt_coord.size();j++){
  112. if(!pt_coord[j] || !pt_coord[j]->Dim()) continue;
  113. Vector<Real_t>& coord=*pt_coord[j];
  114. size_t npts=coord.Dim()/this->dim;
  115. Vector<size_t> cdata(nchld+1);
  116. for(size_t i=0;i<nchld+1;i++){
  117. long long pt1=-1, pt2=npts;
  118. while(pt2-pt1>1){ // binary search
  119. long long pt3=(pt1+pt2)/2;
  120. assert(pt3<npts);
  121. if(pt3<0) pt3=0;
  122. int ch_id=(coord[pt3*3+0]>=c[0]+s)*1+
  123. (coord[pt3*3+1]>=c[1]+s)*2+
  124. (coord[pt3*3+2]>=c[2]+s)*4;
  125. if(ch_id< i) pt1=pt3;
  126. if(ch_id>=i) pt2=pt3;
  127. }
  128. cdata[i]=pt2;
  129. }
  130. if(pt_coord[j]){
  131. Vector<Real_t>& vec=*pt_coord[j];
  132. size_t dof=vec.Dim()/npts;
  133. if(dof>0) for(size_t i=0;i<nchld;i++){
  134. Vector<Real_t>& chld_vec=*chld_pt_coord[i][j];
  135. chld_vec.ReInit((cdata[i+1]-cdata[i])*dof, &vec[0]+cdata[i]*dof);
  136. }
  137. vec.ReInit(0);
  138. }
  139. if(pt_value[j]){
  140. Vector<Real_t>& vec=*pt_value[j];
  141. size_t dof=vec.Dim()/npts;
  142. if(dof>0) for(size_t i=0;i<nchld;i++){
  143. Vector<Real_t>& chld_vec=*chld_pt_value[i][j];
  144. chld_vec.ReInit((cdata[i+1]-cdata[i])*dof, &vec[0]+cdata[i]*dof);
  145. }
  146. vec.ReInit(0);
  147. }
  148. if(pt_scatter[j]){
  149. Vector<size_t>& vec=*pt_scatter[j];
  150. size_t dof=vec.Dim()/npts;
  151. if(dof>0) for(size_t i=0;i<nchld;i++){
  152. Vector<size_t>& chld_vec=*chld_pt_scatter[i][j];
  153. chld_vec.ReInit((cdata[i+1]-cdata[i])*dof, &vec[0]+cdata[i]*dof);
  154. }
  155. vec.ReInit(0);
  156. }
  157. }
  158. }
  159. };
  160. template <class T>
  161. void MPI_Node<T>::Truncate(){
  162. if(!this->IsLeaf()){
  163. int nchld=(1UL<<this->Dim());
  164. for(size_t i=0;i<nchld;i++){
  165. if(!this->Child(i)->IsLeaf()){
  166. this->Child(i)->Truncate();
  167. }
  168. }
  169. std::vector<Vector<Real_t>*> pt_coord;
  170. std::vector<Vector<Real_t>*> pt_value;
  171. std::vector<Vector<size_t>*> pt_scatter;
  172. this->NodeDataVec(pt_coord, pt_value, pt_scatter);
  173. std::vector<std::vector<Vector<Real_t>*> > chld_pt_coord(nchld);
  174. std::vector<std::vector<Vector<Real_t>*> > chld_pt_value(nchld);
  175. std::vector<std::vector<Vector<size_t>*> > chld_pt_scatter(nchld);
  176. for(size_t i=0;i<nchld;i++){
  177. static_cast<MPI_Node<Real_t>*>((MPI_Node<T>*)this->Child(i))
  178. ->NodeDataVec(chld_pt_coord[i], chld_pt_value[i], chld_pt_scatter[i]);
  179. }
  180. for(size_t j=0;j<pt_coord.size();j++){
  181. if(!pt_coord[j]) continue;
  182. if(pt_coord[j]){
  183. size_t vec_size=0;
  184. for(size_t i=0;i<nchld;i++){
  185. Vector<Real_t>& chld_vec=*chld_pt_coord[i][j];
  186. vec_size+=chld_vec.Dim();
  187. }
  188. Vector<Real_t> vec=*pt_coord[j];
  189. vec.ReInit(vec_size);
  190. vec_size=0;
  191. for(size_t i=0;i<nchld;i++){
  192. Vector<Real_t>& chld_vec=*chld_pt_coord[i][j];
  193. if(chld_vec.Dim()>0){
  194. mem::memcopy(&vec[vec_size],&chld_vec[0],chld_vec.Dim()*sizeof(Real_t));
  195. vec_size+=chld_vec.Dim();
  196. }
  197. }
  198. }
  199. if(pt_value[j]){
  200. size_t vec_size=0;
  201. for(size_t i=0;i<nchld;i++){
  202. Vector<Real_t>& chld_vec=*chld_pt_value[i][j];
  203. vec_size+=chld_vec.Dim();
  204. }
  205. Vector<Real_t> vec=*pt_value[j];
  206. vec.ReInit(vec_size);
  207. vec_size=0;
  208. for(size_t i=0;i<nchld;i++){
  209. Vector<Real_t>& chld_vec=*chld_pt_value[i][j];
  210. if(chld_vec.Dim()>0){
  211. mem::memcopy(&vec[vec_size],&chld_vec[0],chld_vec.Dim()*sizeof(Real_t));
  212. vec_size+=chld_vec.Dim();
  213. }
  214. }
  215. }
  216. if(pt_scatter[j]){
  217. size_t vec_size=0;
  218. for(size_t i=0;i<nchld;i++){
  219. Vector<size_t>& chld_vec=*chld_pt_scatter[i][j];
  220. vec_size+=chld_vec.Dim();
  221. }
  222. Vector<size_t> vec=*pt_scatter[j];
  223. vec.ReInit(vec_size);
  224. vec_size=0;
  225. for(size_t i=0;i<nchld;i++){
  226. Vector<size_t>& chld_vec=*chld_pt_scatter[i][j];
  227. if(chld_vec.Dim()>0){
  228. mem::memcopy(&vec[vec_size],&chld_vec[0],chld_vec.Dim()*sizeof(Real_t));
  229. vec_size+=chld_vec.Dim();
  230. }
  231. }
  232. }
  233. }
  234. }
  235. TreeNode::Truncate();
  236. }
  237. template <class T>
  238. PackedData MPI_Node<T>::Pack(bool ghost, void* buff_ptr, size_t offset){
  239. std::vector<Vector<Real_t>*> pt_coord;
  240. std::vector<Vector<Real_t>*> pt_value;
  241. std::vector<Vector<size_t>*> pt_scatter;
  242. this->NodeDataVec(pt_coord, pt_value, pt_scatter);
  243. PackedData p0;
  244. // Determine data length.
  245. p0.length =sizeof(size_t)+sizeof(int)+sizeof(long long)+sizeof(MortonId);
  246. for(size_t j=0;j<pt_coord.size();j++){
  247. p0.length+=3*sizeof(size_t);
  248. if(pt_coord [j]) p0.length+=(pt_coord [j]->Dim())*sizeof(Real_t);
  249. if(pt_value [j]) p0.length+=(pt_value [j]->Dim())*sizeof(Real_t);
  250. if(pt_scatter[j]) p0.length+=(pt_scatter[j]->Dim())*sizeof(size_t);
  251. }
  252. // Allocate memory.
  253. p0.data=(char*)buff_ptr;
  254. if(!p0.data){
  255. this->packed_data.Resize(p0.length+offset);
  256. p0.data=&this->packed_data[0];
  257. }
  258. char* data_ptr=(char*)p0.data;
  259. data_ptr+=offset;
  260. // Header
  261. ((size_t*)data_ptr)[0]=p0.length;
  262. data_ptr+=sizeof(size_t);
  263. ((int*)data_ptr)[0]=this->Depth();
  264. data_ptr+=sizeof(int);
  265. ((long long*)data_ptr)[0]=this->NodeCost();
  266. data_ptr+=sizeof(long long);
  267. ((MortonId*)data_ptr)[0]=this->GetMortonId();
  268. data_ptr+=sizeof(MortonId);
  269. // Copy Vector data.
  270. for(size_t j=0;j<pt_coord.size();j++){
  271. if(pt_coord[j]){
  272. Vector<Real_t>& vec=*pt_coord[j];
  273. ((size_t*)data_ptr)[0]=vec.Dim(); data_ptr+=sizeof(size_t);
  274. if(vec.Dim()>0 && data_ptr!=(char*)&vec[0])
  275. mem::memcopy(data_ptr, &vec[0], sizeof(Real_t)*vec.Dim());
  276. data_ptr+=vec.Dim()*sizeof(Real_t);
  277. }else{
  278. ((size_t*)data_ptr)[0]=0; data_ptr+=sizeof(size_t);
  279. }
  280. if(pt_value[j]){
  281. Vector<Real_t>& vec=*pt_value[j];
  282. ((size_t*)data_ptr)[0]=vec.Dim(); data_ptr+=sizeof(size_t);
  283. if(vec.Dim()>0 && data_ptr!=(char*)&vec[0])
  284. mem::memcopy(data_ptr, &vec[0], sizeof(Real_t)*vec.Dim());
  285. data_ptr+=vec.Dim()*sizeof(Real_t);
  286. }else{
  287. ((size_t*)data_ptr)[0]=0; data_ptr+=sizeof(size_t);
  288. }
  289. if(pt_scatter[j] && !ghost){
  290. Vector<size_t>& vec=*pt_scatter[j];
  291. ((size_t*)data_ptr)[0]=vec.Dim(); data_ptr+=sizeof(size_t);
  292. if(vec.Dim()>0 && data_ptr!=(char*)&vec[0])
  293. mem::memcopy(data_ptr, &vec[0], sizeof(size_t)*vec.Dim());
  294. data_ptr+=vec.Dim()*sizeof(size_t);
  295. }else{
  296. ((size_t*)data_ptr)[0]=0; data_ptr+=sizeof(size_t);
  297. }
  298. }
  299. return p0;
  300. }
  301. template <class T>
  302. void MPI_Node<T>::Unpack(PackedData p0, bool own_data){
  303. std::vector<Vector<Real_t>*> pt_coord;
  304. std::vector<Vector<Real_t>*> pt_value;
  305. std::vector<Vector<size_t>*> pt_scatter;
  306. this->NodeDataVec(pt_coord, pt_value, pt_scatter);
  307. char* data_ptr=(char*)p0.data;
  308. // Check header
  309. assert(((size_t*)data_ptr)[0]==p0.length);
  310. data_ptr+=sizeof(size_t);
  311. this->depth=(((int*)data_ptr)[0]);
  312. data_ptr+=sizeof(int);
  313. this->NodeCost()=(((long long*)data_ptr)[0]);
  314. data_ptr+=sizeof(long long);
  315. this->SetCoord(((MortonId*)data_ptr)[0]);
  316. data_ptr+=sizeof(MortonId);
  317. for(size_t j=0;j<pt_coord.size();j++){
  318. if(pt_coord[j]){
  319. Vector<Real_t>& vec=*pt_coord[j];
  320. size_t vec_sz=(((size_t*)data_ptr)[0]); data_ptr+=sizeof(size_t);
  321. if(own_data){
  322. vec=Vector<Real_t>(vec_sz,(Real_t*)data_ptr,false);
  323. }else{
  324. vec.ReInit(vec_sz,(Real_t*)data_ptr,false);
  325. }
  326. data_ptr+=vec_sz*sizeof(Real_t);
  327. }else{
  328. assert(!((size_t*)data_ptr)[0]);
  329. data_ptr+=sizeof(size_t);
  330. }
  331. if(pt_value[j]){
  332. Vector<Real_t>& vec=*pt_value[j];
  333. size_t vec_sz=(((size_t*)data_ptr)[0]); data_ptr+=sizeof(size_t);
  334. if(own_data){
  335. vec=Vector<Real_t>(vec_sz,(Real_t*)data_ptr,false);
  336. }else{
  337. vec.ReInit(vec_sz,(Real_t*)data_ptr,false);
  338. }
  339. data_ptr+=vec_sz*sizeof(Real_t);
  340. }else{
  341. assert(!((size_t*)data_ptr)[0]);
  342. data_ptr+=sizeof(size_t);
  343. }
  344. if(pt_scatter[j]){
  345. Vector<size_t>& vec=*pt_scatter[j];
  346. size_t vec_sz=(((size_t*)data_ptr)[0]); data_ptr+=sizeof(size_t);
  347. if(own_data){
  348. vec=Vector<size_t>(vec_sz,(size_t*)data_ptr,false);
  349. }else{
  350. vec.ReInit(vec_sz,(size_t*)data_ptr,false);
  351. }
  352. data_ptr+=vec_sz*sizeof(size_t);
  353. }else{
  354. assert(!((size_t*)data_ptr)[0]);
  355. data_ptr+=sizeof(size_t);
  356. }
  357. }
  358. }
  359. template <class T>
  360. void MPI_Node<T>::ReadVal(std::vector<Real_t> x,std::vector<Real_t> y, std::vector<Real_t> z, Real_t* val, bool show_ghost){
  361. if(!pt_coord.Dim()) return;
  362. size_t n_pts=pt_coord.Dim()/dim;
  363. size_t data_dof=pt_value.Dim()/n_pts;
  364. std::vector<Real_t> v(data_dof,0);
  365. for(size_t i=0;i<n_pts;i++)
  366. for(int j=0;j<data_dof;j++)
  367. v[j]+=pt_value[i*data_dof+j];
  368. for(int j=0;j<data_dof;j++)
  369. v[j]=v[j]/n_pts;
  370. for(size_t i=0;i<x.size()*y.size()*z.size()*data_dof;i++){
  371. val[i]=v[i%data_dof];
  372. }
  373. }
  374. template <class T>
  375. void MPI_Node<T>::VTU_Data(std::vector<Real_t>& coord, std::vector<Real_t>& value, std::vector<int32_t>& connect, std::vector<int32_t>& offset, std::vector<uint8_t>& types, int lod){
  376. if(!pt_coord.Dim()) return;
  377. size_t point_cnt=coord.size()/COORD_DIM;
  378. size_t connect_cnt=connect.size();
  379. for(size_t i=0;i<pt_coord.Dim();i+=3){
  380. coord.push_back(pt_coord[i+0]);
  381. coord.push_back(pt_coord[i+1]);
  382. coord.push_back(pt_coord[i+2]);
  383. connect_cnt++;
  384. connect.push_back(point_cnt);
  385. offset.push_back(connect_cnt);
  386. types.push_back(1);
  387. point_cnt++;
  388. }
  389. for(size_t i=0;i<pt_value.Dim();i++) value.push_back(pt_value[i]);
  390. Real_t* c=this->Coord();
  391. Real_t s=pow(0.5,this->Depth());
  392. size_t data_dof=pt_value.Dim()/(pt_coord.Dim()/dim);
  393. for(int i=0;i<8;i++){
  394. coord.push_back(c[0]+(i&1?1:0)*s);
  395. coord.push_back(c[1]+(i&2?1:0)*s);
  396. coord.push_back(c[2]+(i&4?1:0)*s);
  397. for(int j=0;j<data_dof;j++) value.push_back(0.0);
  398. connect.push_back(point_cnt+i);
  399. }
  400. offset.push_back(8+connect_cnt);
  401. types.push_back(11);
  402. {// Set point values.
  403. Real_t* val_ptr=&value[point_cnt*data_dof];
  404. std::vector<Real_t> x(2);
  405. std::vector<Real_t> y(2);
  406. std::vector<Real_t> z(2);
  407. x[0]=c[0]; x[1]=c[0]+s;
  408. y[0]=c[1]; y[1]=c[1]+s;
  409. z[0]=c[2]; z[1]=c[2]+s;
  410. this->ReadVal(x, y, z, val_ptr);
  411. //Rearrrange data
  412. //(x1,x2,x3,...,y1,y2,...z1,...) => (x1,y1,z1,x2,y2,z2,...)
  413. Matrix<Real_t> M(data_dof,8,val_ptr,false);
  414. M=M.Transpose();
  415. }
  416. }
  417. }//end namespace