-
Notifications
You must be signed in to change notification settings - Fork 2
/
Copy pathex2.c
217 lines (177 loc) · 7.44 KB
/
ex2.c
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
/*******************************************************************************
* Copyright (C) 2015-2019 Commissariat a l'energie atomique et aux energies alternatives (CEA)
*
* Permission is hereby granted, free of charge, to any person obtaining a copy
* of this software and associated documentation files (the "Software"), to deal
* in the Software without restriction, including without limitation the rights
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
* copies of the Software, and to permit persons to whom the Software is
* furnished to do so, subject to the following conditions:
*
* The above copyright notice and this permission notice shall be included in all
* copies or substantial portions of the Software.
*
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
* SOFTWARE.
******************************************************************************/
#include <mpi.h>
#include <assert.h>
#include <math.h>
#include <stdio.h>
#include <stdlib.h>
#include <time.h>
#include <paraconf.h>
// load the PDI header
#include <pdi.h>
/// size of the local data as [HEIGHT, WIDTH] including ghosts & boundary constants
int dsize[2];
/// 2D size of the process grid as [HEIGHT, WIDTH]
int psize[2];
/// 2D rank of the local process in the process grid as [YY, XX]
int pcoord[2];
/// the alpha coefficient used in the computation
double alpha;
/** Initialize the data all to 0 except for the left border (XX==0) initialized to 1 million
* \param[out] dat the local data to initialize
*/
void init(double dat[dsize[0]][dsize[1]])
{
for (int yy=0; yy<dsize[0]; ++yy) for (int xx=0; xx<dsize[1]; ++xx) dat[yy][xx] = 0;
if ( pcoord[1] == 0 ) for (int yy=0; yy<dsize[0]; ++yy) dat[yy][0] = 1000000;
}
/** Compute the values at the next time-step based on the values at the current time-step
* \param[in] cur the local data at the current time-step
* \param[out] next the local data at the next time-step
*/
void iter(double cur[dsize[0]][dsize[1]], double next[dsize[0]][dsize[1]])
{
int xx, yy;
for (xx=0; xx<dsize[1]; ++xx) next[0][xx] = cur[0][xx];
for (yy=1; yy<dsize[0]-1; ++yy) {
next[yy][0] = cur[yy][0];
for (xx=1; xx<dsize[1]-1; ++xx) {
next[yy][xx] =
(1.-4.*alpha) * cur[yy][xx]
+ alpha * ( cur[yy][xx-1]
+ cur[yy][xx+1]
+ cur[yy-1][xx]
+ cur[yy+1][xx]
);
}
next[yy][dsize[1]-1] = cur[yy][dsize[1]-1];
}
for (xx=0; xx<dsize[1]; ++xx) next[dsize[0]-1][xx] = cur[dsize[0]-1][xx];
}
/** Exchanges ghost values with neighbours
* \param[in] cart_comm the MPI communicator with all processes organized in a 2D Cartesian grid
* \param[in] cur the local data at the current time-step whose ghosts need exchanging
*/
void exchange(MPI_Comm cart_comm, double cur[dsize[0]][dsize[1]])
{
MPI_Status status;
int rank_source, rank_dest;
static MPI_Datatype column, row;
static int initialized = 0;
if ( !initialized ) {
MPI_Type_vector(dsize[0]-2, 1, dsize[1], MPI_DOUBLE, &column);
MPI_Type_commit(&column);
MPI_Type_contiguous(dsize[1]-2, MPI_DOUBLE, &row);
MPI_Type_commit(&row);
initialized = 1;
}
// send down
MPI_Cart_shift(cart_comm, 0, 1, &rank_source, &rank_dest);
MPI_Sendrecv(&cur[dsize[0]-2][1], 1, row, rank_dest, 100, // send row before ghost
&cur[0][1], 1, row, rank_source, 100, // receive 1st row (ghost)
cart_comm, &status);
// send up
MPI_Cart_shift(cart_comm, 0, -1, &rank_source, &rank_dest);
MPI_Sendrecv(&cur[1][1], 1, row, rank_dest, 100, // send column after ghost
&cur[dsize[0]-1][1], 1, row, rank_source, 100, // receive last column (ghost)
cart_comm, &status);
// send to the right
MPI_Cart_shift(cart_comm, 1, 1, &rank_source, &rank_dest);
MPI_Sendrecv(&cur[1][dsize[1]-2], 1, column, rank_dest, 100, // send column before ghost
&cur[1][0], 1, column, rank_source, 100, // receive 1st column (ghost)
cart_comm, &status);
// send to the left
MPI_Cart_shift(cart_comm, 1, -1, &rank_source, &rank_dest);
MPI_Sendrecv(&cur[1][1], 1, column, rank_dest, 100, // send column after ghost
&cur[1][dsize[1]-1], 1, column, rank_source, 100, // receive last column (ghost)
cart_comm, &status);
}
int main( int argc, char* argv[] )
{
MPI_Init(&argc, &argv);
// load the configuration tree
PC_tree_t conf = PC_parse_path("ex2.yml");
// NEVER USE MPI_COMM_WORLD IN THE CODE, use our own communicator main_comm instead
MPI_Comm main_comm = MPI_COMM_WORLD;
// initialize PDI, it can replace our main communicator by its own
PDI_init(PC_get(conf, ".pdi"));
// load the MPI rank & size
int psize_1d; MPI_Comm_size(main_comm, &psize_1d);
int pcoord_1d; MPI_Comm_rank(main_comm, &pcoord_1d);
long longval;
// load the alpha parameter
PC_double(PC_get(conf, ".alpha"), &alpha);
// load the global data-size
int global_size[2];
PC_int(PC_get(conf, ".global_size.height"), &longval); global_size[0] = longval;
PC_int(PC_get(conf, ".global_size.width"), &longval); global_size[1] = longval;
// load the parallelism configuration
PC_int(PC_get(conf, ".parallelism.height"), &longval); psize[0] = longval;
PC_int(PC_get(conf, ".parallelism.width" ), &longval); psize[1] = longval;
// check the configuration is coherent
assert(global_size[0]%psize[0]==0);
assert(global_size[1]%psize[1]==0);
assert(psize[1]*psize[0] == psize_1d);
// compute the local data-size with space for ghosts and boundary constants
dsize[0] = global_size[0]/psize[0] + 2;
dsize[1] = global_size[1]/psize[1] + 2;
// create a 2D Cartesian MPI communicator & get our coordinate (rank) in it
int cart_period[2] = { 0, 0 };
MPI_Comm cart_comm; MPI_Cart_create(main_comm, 2, psize, cart_period, 1, &cart_comm);
MPI_Cart_coords(cart_comm, pcoord_1d, 2, pcoord);
// allocate memory for the double buffered data
double(*cur)[dsize[1]] = malloc(sizeof(double)*dsize[1]*dsize[0]);
double(*next)[dsize[1]] = malloc(sizeof(double)*dsize[1]*dsize[0]);
// initialize the data content
init(cur);
// our loop counter so as to be able to use it outside the loop
int ii=0;
// share useful configuration bits with PDI
PDI_share("ii", &ii, PDI_OUT);
PDI_reclaim("ii");
// the main loop
for (; ii<10; ++ii) {
// share the loop counter at each iteration
PDI_share("ii", &ii, PDI_OUT);
PDI_reclaim("ii");
// compute the values for the next iteration
iter(cur, next);
// exchange data with the neighbours
exchange(cart_comm, next);
// swap the current and next values
double (*tmp)[dsize[1]] = cur; cur = next; next = tmp;
}
// finally share the main field after the main loop body
PDI_share("main_field", cur, PDI_OUT);
PDI_reclaim("main_field");
// finalize PDI
PDI_finalize();
// destroy the paraconf configuration tree
PC_tree_destroy(&conf);
// free the allocated memory
free(cur);
free(next);
// finalize MPI
MPI_Finalize();
fprintf(stderr, "[%d] SUCCESS\n", pcoord_1d);
return EXIT_SUCCESS;
}