Skip to content

Commit

Permalink
examples/proxy: add wait batch count
Browse files Browse the repository at this point in the history
Use -w to set the desired number of CQEs to wait for before processing
them. This is typical batching behavior in processing. Can be used
with -t for setting the wait timeout as well. This allows tightly
controlling latencies, while still providing efficiency.

Batching can make a substantial difference in effiency. For example,
running proxy as a sink and using the default wait of 1, and sending
it 32 byte packets over a 10G link yields the following performance
based on wait_batch size:

Wait batch		QPS
============================
1			5.7M
4			7.7M
32			8.7M

Even just a small wait batch can drastically improve the performance
at which a single threaded application (like proxy) can receive.

Signed-off-by: Jens Axboe <[email protected]>
  • Loading branch information
axboe committed Feb 18, 2024
1 parent 12118cc commit 282e5dd
Showing 1 changed file with 41 additions and 8 deletions.
49 changes: 41 additions & 8 deletions examples/proxy.c
Original file line number Diff line number Diff line change
Expand Up @@ -76,6 +76,8 @@ static int bidi;
static int ipv6;
static int napi;
static int napi_timeout;
static int wait_batch = 1;
static int wait_usec = 1000000;
static int verbose;

static int nr_bufs = 256;
Expand Down Expand Up @@ -1059,6 +1061,8 @@ static void usage(const char *name)
printf("\t-S:\t\tUse SQPOLL (%d)\n", sqpoll);
printf("\t-b:\t\tSend/receive buf size (%d)\n", buf_size);
printf("\t-n:\t\tNumber of provided buffers (pow2) (%d)\n", nr_bufs);
printf("\t-w:\t\tNumber of CQEs to wait for each loop (%d)\n", wait_batch);
printf("\t-t:\t\tTimeout for waiting on CQEs (usec) (%d)\n", wait_usec);
printf("\t-s:\t\tAct only as a sink (%d)\n", is_sink);
printf("\t-f:\t\tUse only fixed files (%d)\n", fixed_files);
printf("\t-B:\t\tUse bi-directional mode (%d)\n", bidi);
Expand Down Expand Up @@ -1093,6 +1097,7 @@ static void check_for_close(struct io_uring *ring)
*/
static int event_loop(struct io_uring *ring, int fd)
{
struct __kernel_timespec active_ts, idle_ts = { .tv_sec = 1, };
struct io_uring_sqe *sqe;

/*
Expand All @@ -1114,19 +1119,41 @@ static int event_loop(struct io_uring *ring, int fd)
io_uring_prep_multishot_accept(sqe, fd, NULL, NULL, 0);
__encode_userdata(sqe, 0, __ACCEPT, 0, 0, fd);

if (wait_usec > 1000000) {
active_ts.tv_sec = wait_usec / 1000000;
wait_usec -= active_ts.tv_sec * 1000000;
}
active_ts.tv_nsec = wait_usec * 1000;

while (1) {
struct __kernel_timespec ts = {
.tv_sec = 1,
};
struct __kernel_timespec *ts = &idle_ts;
struct io_uring_cqe *cqe;
unsigned int head;
int flags, i, to_wait;
int ret, flags, i, to_wait;

/*
* If wait_batch is set higher than 1, then we'll wait on
* that amount of CQEs to be posted each loop. If used with
* DEFER_TASKRUN, this can provide a substantial reduction
* in context switch rate as the task isn't woken until the
* requested number of events can be returned.
*
* Can be used with -t to set a wait_usec timeout as well.
* For example, if an application can deal with 250 usec
* of wait latencies, it can set -w8 -t250 which will cause
* io_uring to return when either 8 events have been received,
* or if 250 usec of waiting has passed.
*
* If we don't have any open connections, wait on just 1
* always.
*/
to_wait = 1;
if (open_conns)
to_wait = open_conns;
if (open_conns) {
cur_ts = &active_ts;
to_wait = open_conns * wait_batch;
}

io_uring_submit_and_wait_timeout(ring, &cqe, to_wait, &ts, NULL);
io_uring_submit_and_wait_timeout(ring, &cqe, to_wait, ts, NULL);

i = flags = 0;
io_uring_for_each_cqe(ring, head, cqe) {
Expand Down Expand Up @@ -1168,7 +1195,7 @@ int main(int argc, char *argv[])
return 1;
}

while ((opt = getopt(argc, argv, "m:d:S:s:b:f:H:r:p:n:B:N:T:6Vh?")) != -1) {
while ((opt = getopt(argc, argv, "m:d:S:s:b:f:H:r:p:n:B:N:T:w:t:6Vh?")) != -1) {
switch (opt) {
case 'm':
mshot = !!atoi(optarg);
Expand All @@ -1185,6 +1212,12 @@ int main(int argc, char *argv[])
case 'n':
nr_bufs = atoi(optarg);
break;
case 'w':
wait_batch = atoi(optarg);
break;
case 't':
wait_usec = atoi(optarg);
break;
case 's':
is_sink = !!atoi(optarg);
break;
Expand Down

0 comments on commit 282e5dd

Please sign in to comment.