Page Menu
Home
c4science
Search
Configure Global Search
Log In
Files
F94935689
pi_1sided_get.cc
No One
Temporary
Actions
Download File
Edit File
Delete File
View Transforms
Subscribe
Mute Notifications
Award Token
Subscribers
None
File Metadata
Details
File Info
Storage
Attached
Created
Wed, Dec 11, 13:03
Size
2 KB
Mime Type
text/x-c
Expires
Fri, Dec 13, 13:03 (1 d, 23 h)
Engine
blob
Format
Raw Data
Handle
22902510
Attached To
R7871 phys-743-exercises
pi_1sided_get.cc
View Options
/*
This exercise is taken from the class Parallel Programming Workshop (MPI,
OpenMP and Advanced Topics) at HLRS given by Rolf Rabenseifner
*/
#include <chrono>
#include <cstdio>
#include <cmath>
#include <mpi.h>
using
clk
=
std
::
chrono
::
high_resolution_clock
;
using
second
=
std
::
chrono
::
duration
<
double
>
;
using
time_point
=
std
::
chrono
::
time_point
<
clk
>
;
inline
int
digit
(
double
x
,
int
n
)
{
return
std
::
trunc
(
x
*
std
::
pow
(
10.
,
n
))
-
std
::
trunc
(
x
*
std
::
pow
(
10.
,
n
-
1
))
*
10.
;
}
inline
double
f
(
double
a
)
{
return
(
4.
/
(
1.
+
a
*
a
));
}
const
int
n
=
10000000
;
int
main
(
int
/* argc */
,
char
**
/* argv */
)
{
int
i
;
double
dx
,
x
,
sum
,
pi
;
MPI_Init
(
NULL
,
NULL
);
int
prank
,
psize
;
MPI_Comm_rank
(
MPI_COMM_WORLD
,
&
prank
);
MPI_Comm_size
(
MPI_COMM_WORLD
,
&
psize
);
auto
t1
=
clk
::
now
();
auto
ln
=
n
/
psize
+
(
prank
<
n
%
psize
?
1
:
0
);
auto
i_start
=
prank
*
ln
+
(
prank
<
n
%
psize
?
0
:
n
%
psize
);
auto
i_end
=
i_start
+
ln
;
/* calculate pi = integral [0..1] 4 / (1 + x**2) dx */
dx
=
1.
/
n
;
double
lsum
=
0.0
;
for
(
i
=
i_start
;
i
<
i_end
;
i
++
)
{
x
=
1.
*
i
*
dx
;
lsum
=
lsum
+
f
(
x
);
}
double
send
,
recv
;
//auto next = (prank + 1) % psize;
auto
prev
=
(
prank
-
1
+
psize
)
%
psize
;
MPI_Win
win
;
MPI_Win_create
(
&
send
,
MPI_Aint
(
sizeof
(
double
)),
sizeof
(
double
),
MPI_INFO_NULL
,
MPI_COMM_WORLD
,
&
win
);
send
=
sum
=
lsum
;
for
(
int
p
=
0
;
p
<
psize
-
1
;
++
p
)
{
MPI_Win_fence
(
MPI_MODE_NOPUT
|
MPI_MODE_NOPRECEDE
,
win
);
MPI_Get
(
&
recv
,
1
,
MPI_DOUBLE
,
prev
,
(
MPI_Aint
)
0
,
1
,
MPI_DOUBLE
,
win
);
MPI_Win_fence
(
MPI_MODE_NOSTORE
|
MPI_MODE_NOPUT
|
MPI_MODE_NOSUCCEED
,
win
);
sum
+=
recv
;
send
=
recv
;
}
pi
=
dx
*
sum
;
MPI_Win_free
(
&
win
);
second
elapsed
=
clk
::
now
()
-
t1
;
if
(
prank
==
0
)
{
std
::
printf
(
"computed pi = %.16g
\n
"
,
pi
);
std
::
printf
(
"wall clock time (chrono) = %.4gs
\n
"
,
elapsed
.
count
());
std
::
printf
(
"n mpi process = %.d
\n
"
,
psize
);
for
(
int
d
=
1
;
d
<=
15
;
++
d
)
{
std
::
printf
(
"%d"
,
digit
(
pi
,
d
));
}
}
MPI_Finalize
();
return
0
;
}
Event Timeline
Log In to Comment