Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Support
Keyboard shortcuts
?
Submit feedback
Contribute to GitLab
Sign in / Register
Toggle navigation
R
rdma-mwe
Project overview
Project overview
Details
Activity
Releases
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Issues
0
Issues
0
List
Boards
Labels
Milestones
Merge Requests
0
Merge Requests
0
CI / CD
CI / CD
Pipelines
Jobs
Schedules
Analytics
Analytics
CI / CD
Repository
Value Stream
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Create a new issue
Jobs
Commits
Issue Boards
Open sidebar
Titouan Soulard
rdma-mwe
Commits
c6b05feb
Commit
c6b05feb
authored
Feb 13, 2024
by
Titouan Soulard
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
example: add a stress test
parent
c7de92cc
Changes
1
Hide whitespace changes
Inline
Side-by-side
Showing
1 changed file
with
129 additions
and
67 deletions
+129
-67
example/rdma_test.c
example/rdma_test.c
+129
-67
No files found.
example/rdma_test.c
View file @
c6b05feb
...
...
@@ -10,6 +10,9 @@
#include "libcapulet/net_udp.h"
#include "libcapulet/rdma_ib.h"
#define RDMA_TEST_FILE 1
#define RDMA_TEST_STRESS 2
struct
ServeMrThreadContext
{
struct
CommonHashtableTable
*
mr_mgr
;
int
server_socket
;
...
...
@@ -28,29 +31,34 @@ int main(int argc, char *argv[]) {
struct
CapuletNetUdpMrInfoPacket
out_info_packet
;
struct
CommonHashtableTable
mr_mgr
;
struct
ServeMrThreadContext
serve_mr_td_ctx
;
struct
ibv_mr
*
in_mr
;
struct
ibv_mr
*
out_mr
;
struct
ibv_wc
poll_wc
;
struct
addrinfo
server_hints
;
struct
addrinfo
*
server_infos
;
struct
timespec
timestamps
[
2
];
pthread_t
serve_mr_td
;
struct
ServeMrThreadContext
serve_mr_td_ctx
;
size_t
remote_host_size
;
size_t
device_name_size
;
long
elapsed_time
;
long
max_time
;
int
allocated_size
;
int
server_socket
;
int
opt
;
char
*
remote_host
;
char
*
device_name
;
char
mode
;
char
in_mr_fc
;
bool
is_client
;
bool
result
;
// Preallocate some variables
allocated_size
=
16384
*
sizeof
(
char
);
max_time
=
0
;
remote_host
=
NULL
;
device_name
=
NULL
;
is_client
=
false
;
...
...
@@ -65,6 +73,15 @@ int main(int argc, char *argv[]) {
/******************************
**** Arguments processing ****
******************************/
if
(
strcmp
(
argv
[
1
],
"file"
)
==
0
)
{
mode
=
RDMA_TEST_FILE
;
}
else
if
(
strcmp
(
argv
[
1
],
"stress"
)
==
0
)
{
mode
=
RDMA_TEST_STRESS
;
}
else
{
fprintf
(
stderr
,
"Invalid command, allowed modes are: file and stress
\n
"
);
return
-
1
;
}
while
((
opt
=
getopt
(
argc
,
argv
,
"c:d:"
))
!=
-
1
)
{
// Detect if a server was given; if any, act as a client
if
(
opt
==
'c'
)
{
...
...
@@ -106,7 +123,6 @@ int main(int argc, char *argv[]) {
return
-
1
;
}
// Allocate memory and get user data from STDIN
in_mr
=
capulet_rdma_ib_create_mr
(
&
rdma_ctx
,
allocated_size
,
IBV_ACCESS_LOCAL_WRITE
|
IBV_ACCESS_REMOTE_WRITE
);
out_mr
=
capulet_rdma_ib_create_mr
(
&
rdma_ctx
,
allocated_size
,
IBV_ACCESS_LOCAL_WRITE
|
IBV_ACCESS_REMOTE_READ
);
...
...
@@ -127,29 +143,12 @@ int main(int argc, char *argv[]) {
return
-
1
;
}
if
(
is_client
)
{
read
(
STDIN_FILENO
,
(
char
*
)
in_mr
->
addr
,
allocated_size
);
}
else
{
read
(
STDIN_FILENO
,
(
char
*
)
out_mr
->
addr
,
allocated_size
);
}
result
=
capulet_rdma_ib_initialize_qp
(
&
rdma_ctx
,
IBV_ACCESS_LOCAL_WRITE
|
IBV_ACCESS_REMOTE_READ
|
IBV_ACCESS_REMOTE_WRITE
);
if
(
!
result
)
{
fprintf
(
stderr
,
"Queue Pair initialization failed
\n
"
);
return
-
1
;
}
result
=
capulet_rdma_ib_post_recv
(
&
rdma_ctx
,
in_mr
,
allocated_size
);
if
(
!
result
)
{
fprintf
(
stderr
,
"Posting Recv (in) failed
\n
"
);
return
-
1
;
}
result
=
capulet_rdma_ib_post_recv
(
&
rdma_ctx
,
out_mr
,
allocated_size
);
if
(
!
result
)
{
fprintf
(
stderr
,
"Posting Recv (out) failed
\n
"
);
return
-
1
;
}
/******************************
**** Exchange informations ***
...
...
@@ -181,73 +180,136 @@ int main(int argc, char *argv[]) {
capulet_net_udp_dump_qp_packet
(
udp_ctx
->
remote
);
/******************************
******** Read or wait ********
******************************/
result
=
capulet_rdma_ib_set_peer_from_udp
(
&
rdma_ctx
,
udp_ctx
);
if
(
!
result
)
{
fprintf
(
stderr
,
"Setting state to RTS failed
\n
"
);
return
-
1
;
}
if
(
!
is_client
)
{
// XXX: check ownership
serve_mr_td_ctx
.
mr_mgr
=
&
mr_mgr
;
serve_mr_td_ctx
.
server_socket
=
server_socket
;
pthread_create
(
&
serve_mr_td
,
NULL
,
*
serve_mr_td_fn
,
(
void
*
)
&
serve_mr_td_ctx
);
}
else
{
result
=
capulet_net_udp_query_mr
(
server_socket
,
server_infos
,
"in"
,
&
in_info_packet
);
if
(
!
result
||
(
strcmp
(
in_info_packet
.
name
,
"in"
)
!=
0
))
{
fprintf
(
stderr
,
"Querying MR (in) from server failed
\n
"
);
return
-
1
;
if
(
mode
==
RDMA_TEST_FILE
)
{
/******************************
******** Read or wait ********
******************************/
if
(
is_client
)
{
read
(
STDIN_FILENO
,
(
char
*
)
in_mr
->
addr
,
allocated_size
);
}
else
{
read
(
STDIN_FILENO
,
(
char
*
)
out_mr
->
addr
,
allocated_size
);
}
result
=
capulet_
net_udp_query_mr
(
server_socket
,
server_infos
,
"out"
,
&
out_info_packet
);
if
(
!
result
||
(
strcmp
(
out_info_packet
.
name
,
"out"
)
!=
0
)
)
{
fprintf
(
stderr
,
"
Querying MR (out) from server
failed
\n
"
);
result
=
capulet_
rdma_ib_post_recv
(
&
rdma_ctx
,
in_mr
,
allocated_size
);
if
(
!
result
)
{
fprintf
(
stderr
,
"
Posting Recv (in)
failed
\n
"
);
return
-
1
;
}
result
=
capulet_rdma_ib_post_
send
(
&
rdma_ctx
,
IBV_WR_RDMA_READ
,
out_mr
,
allocated_size
,
&
out_info_packet
);
result
=
capulet_rdma_ib_post_
recv
(
&
rdma_ctx
,
out_mr
,
allocated_size
);
if
(
!
result
)
{
fprintf
(
stderr
,
"
Sending Read
failed
\n
"
);
fprintf
(
stderr
,
"
Posting Recv (out)
failed
\n
"
);
return
-
1
;
}
result
=
capulet_rdma_ib_post_send
(
&
rdma_ctx
,
IBV_WR_RDMA_WRITE
,
in_mr
,
allocated_size
,
&
in_info_packet
);
if
(
!
result
)
{
fprintf
(
stderr
,
"Sending Write failed
\n
"
);
return
-
1
;
if
(
!
is_client
)
{
serve_mr_td_ctx
.
mr_mgr
=
&
mr_mgr
;
serve_mr_td_ctx
.
server_socket
=
server_socket
;
pthread_create
(
&
serve_mr_td
,
NULL
,
*
serve_mr_td_fn
,
(
void
*
)
&
serve_mr_td_ctx
);
}
else
{
result
=
capulet_net_udp_query_mr
(
server_socket
,
server_infos
,
"in"
,
&
in_info_packet
);
if
(
!
result
||
(
strcmp
(
in_info_packet
.
name
,
"in"
)
!=
0
))
{
fprintf
(
stderr
,
"Querying MR (in) from server failed
\n
"
);
return
-
1
;
}
result
=
capulet_net_udp_query_mr
(
server_socket
,
server_infos
,
"out"
,
&
out_info_packet
);
if
(
!
result
||
(
strcmp
(
out_info_packet
.
name
,
"out"
)
!=
0
))
{
fprintf
(
stderr
,
"Querying MR (out) from server failed
\n
"
);
return
-
1
;
}
result
=
capulet_rdma_ib_post_send
(
&
rdma_ctx
,
IBV_WR_RDMA_READ
,
out_mr
,
allocated_size
,
&
out_info_packet
);
if
(
!
result
)
{
fprintf
(
stderr
,
"Sending Read failed
\n
"
);
return
-
1
;
}
result
=
capulet_rdma_ib_post_send
(
&
rdma_ctx
,
IBV_WR_RDMA_WRITE
,
in_mr
,
allocated_size
,
&
in_info_packet
);
if
(
!
result
)
{
fprintf
(
stderr
,
"Sending Write failed
\n
"
);
return
-
1
;
}
}
}
/******************************
****** Poll completion *******
******************************/
if
(
is_client
)
{
do
{
result
=
ibv_poll_cq
(
rdma_ctx
.
recv_cq
,
1
,
&
poll_wc
);
usleep
(
100
);
}
while
(
result
==
0
);
if
(
result
>
0
&&
poll_wc
.
status
==
IBV_WC_SUCCESS
)
{
if
(
is_client
)
{
printf
(
"%s
\n
"
,
(
char
*
)
out_mr
->
addr
);
/******************************
****** Poll completion *******
******************************/
if
(
is_client
)
{
do
{
result
=
ibv_poll_cq
(
rdma_ctx
.
send_cq
,
1
,
&
poll_wc
);
usleep
(
100
);
}
while
(
result
==
0
);
if
(
result
>
0
&&
poll_wc
.
status
==
IBV_WC_SUCCESS
)
{
if
(
is_client
)
{
printf
(
"%s
\n
"
,
(
char
*
)
out_mr
->
addr
);
}
}
else
{
printf
(
"Failed: %s (WR %lu)
\n
"
,
ibv_wc_status_str
(
poll_wc
.
status
),
poll_wc
.
wr_id
);
}
}
else
{
printf
(
"Failed: %s (WR %lu)
\n
"
,
ibv_wc_status_str
(
poll_wc
.
status
),
poll_wc
.
wr_id
);
// RDMA Read and Write do not give any information on completion
// on the server side, so we wait for the first char of the string
// to be non-zero.
do
{
in_mr_fc
=
*
((
char
*
)
in_mr
->
addr
);
usleep
(
100
);
}
while
(
in_mr_fc
==
0
);
printf
(
"%s
\n
"
,
(
char
*
)
in_mr
->
addr
);
}
}
else
if
(
mode
==
RDMA_TEST_STRESS
)
{
// Fill the Recv queue
for
(
int
i
=
0
;
i
<
16
;
i
++
)
{
capulet_rdma_ib_post_recv
(
&
rdma_ctx
,
in_mr
,
allocated_size
);
}
// A small delay is needed to avoid sending anything before Recv requests
// are posted on the other side.
usleep
(
1000
);
while
(
1
)
{
clock_gettime
(
CLOCK_MONOTONIC
,
&
timestamps
[
0
]);
// Wait for Send
capulet_rdma_ib_post_send
(
&
rdma_ctx
,
IBV_WR_SEND
,
out_mr
,
allocated_size
,
NULL
);
do
{
result
=
ibv_poll_cq
(
rdma_ctx
.
send_cq
,
1
,
&
poll_wc
);
}
while
(
result
==
0
);
if
(
result
<
0
||
poll_wc
.
status
!=
IBV_WC_SUCCESS
)
{
printf
(
"WR %lu failed: %s (%d)
\n
"
,
poll_wc
.
wr_id
,
ibv_wc_status_str
(
poll_wc
.
status
),
poll_wc
.
vendor_err
);
return
-
1
;
}
// Wait for Recv
do
{
result
=
ibv_poll_cq
(
rdma_ctx
.
recv_cq
,
1
,
&
poll_wc
);
}
while
(
result
==
0
);
if
(
result
<
0
||
poll_wc
.
status
!=
IBV_WC_SUCCESS
)
{
printf
(
"WR %lu failed: %s (%d)
\n
"
,
poll_wc
.
wr_id
,
ibv_wc_status_str
(
poll_wc
.
status
),
poll_wc
.
vendor_err
);
return
-
1
;
}
// The Recv queue should always be full for optimal performances: once
// an element is consumed, push back to it.
capulet_rdma_ib_post_recv
(
&
rdma_ctx
,
in_mr
,
allocated_size
);
clock_gettime
(
CLOCK_MONOTONIC
,
&
timestamps
[
1
]);
elapsed_time
=
(
timestamps
[
1
].
tv_sec
-
timestamps
[
0
].
tv_sec
)
*
1000000000
+
timestamps
[
1
].
tv_nsec
-
timestamps
[
0
].
tv_nsec
;
if
(
elapsed_time
>
max_time
)
max_time
=
elapsed_time
;
printf
(
"
\r
%8lu / %8lu"
,
elapsed_time
,
max_time
);
fflush
(
stdout
);
}
}
else
{
// RDMA Read and Write do not give any information on completion
// on the server side, so we wait for the first char of the string
// to be non-zero.
do
{
in_mr_fc
=
*
((
char
*
)
in_mr
->
addr
);
usleep
(
100
);
}
while
(
in_mr_fc
==
0
);
printf
(
"%s
\n
"
,
(
char
*
)
in_mr
->
addr
);
}
/******************************
...
...
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment