2018-08-02 16:07:50 +00:00
|
|
|
/*-
|
|
|
|
* BSD LICENSE
|
|
|
|
*
|
|
|
|
* Copyright (c) Intel Corporation.
|
|
|
|
* All rights reserved.
|
|
|
|
*
|
|
|
|
* Redistribution and use in source and binary forms, with or without
|
|
|
|
* modification, are permitted provided that the following conditions
|
|
|
|
* are met:
|
|
|
|
*
|
|
|
|
* * Redistributions of source code must retain the above copyright
|
|
|
|
* notice, this list of conditions and the following disclaimer.
|
|
|
|
* * Redistributions in binary form must reproduce the above copyright
|
|
|
|
* notice, this list of conditions and the following disclaimer in
|
|
|
|
* the documentation and/or other materials provided with the
|
|
|
|
* distribution.
|
|
|
|
* * Neither the name of Intel Corporation nor the names of its
|
|
|
|
* contributors may be used to endorse or promote products derived
|
|
|
|
* from this software without specific prior written permission.
|
|
|
|
*
|
|
|
|
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
|
|
|
|
* "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
|
|
|
|
* LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
|
|
|
|
* A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
|
|
|
|
* OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
|
|
|
|
* SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
|
|
|
|
* LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
|
|
|
|
* DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
|
|
|
|
* THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
|
|
|
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
|
|
|
|
* OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "spdk/stdinc.h"
|
|
|
|
#include "spdk/thread.h"
|
|
|
|
#include "spdk/env.h"
|
|
|
|
#include "spdk/event.h"
|
|
|
|
#include "spdk/log.h"
|
|
|
|
#include "spdk/string.h"
|
|
|
|
|
|
|
|
#include "spdk/sock.h"
|
|
|
|
#include "spdk/net.h"
|
|
|
|
|
|
|
|
#define ACCEPT_TIMEOUT_US 1000
|
|
|
|
#define CLOSE_TIMEOUT_US 1000000
|
|
|
|
#define BUFFER_SIZE 1024
|
|
|
|
#define ADDR_STR_LEN INET6_ADDRSTRLEN
|
|
|
|
|
|
|
|
static bool g_is_running;
|
|
|
|
|
|
|
|
static char *g_host;
|
|
|
|
static int g_port;
|
|
|
|
static bool g_is_server;
|
|
|
|
static bool g_verbose;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* We'll use this struct to gather housekeeping hello_context to pass between
|
|
|
|
* our events and callbacks.
|
|
|
|
*/
|
|
|
|
struct hello_context_t {
|
|
|
|
bool is_server;
|
|
|
|
char *host;
|
|
|
|
int port;
|
|
|
|
|
|
|
|
bool verbose;
|
|
|
|
int bytes_in;
|
|
|
|
int bytes_out;
|
|
|
|
|
|
|
|
struct spdk_sock *sock;
|
|
|
|
|
|
|
|
struct spdk_sock_group *group;
|
|
|
|
struct spdk_poller *poller_in;
|
|
|
|
struct spdk_poller *poller_out;
|
|
|
|
struct spdk_poller *time_out;
|
2018-09-26 07:55:28 +00:00
|
|
|
|
|
|
|
int rc;
|
2018-08-02 16:07:50 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Usage function for printing parameters that are specific to this application
|
|
|
|
*/
|
|
|
|
static void
|
|
|
|
hello_sock_usage(void)
|
|
|
|
{
|
|
|
|
printf(" -H host_addr host address\n");
|
|
|
|
printf(" -P port port number\n");
|
|
|
|
printf(" -S start in server mode\n");
|
|
|
|
printf(" -V print out additional informations");
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* This function is called to parse the parameters that are specific to this application
|
|
|
|
*/
|
2019-01-08 19:40:06 +00:00
|
|
|
static int hello_sock_parse_arg(int ch, char *arg)
|
2018-08-02 16:07:50 +00:00
|
|
|
{
|
|
|
|
switch (ch) {
|
|
|
|
case 'H':
|
|
|
|
g_host = arg;
|
|
|
|
break;
|
|
|
|
case 'P':
|
2019-01-23 01:01:26 +00:00
|
|
|
g_port = spdk_strtol(arg, 10);
|
|
|
|
if (g_port < 0) {
|
|
|
|
fprintf(stderr, "Invalid port ID\n");
|
|
|
|
return g_port;
|
|
|
|
}
|
2018-08-02 16:07:50 +00:00
|
|
|
break;
|
|
|
|
case 'S':
|
|
|
|
g_is_server = 1;
|
|
|
|
break;
|
|
|
|
case 'V':
|
|
|
|
g_verbose = true;
|
2019-01-21 14:56:47 +00:00
|
|
|
break;
|
|
|
|
default:
|
|
|
|
return -EINVAL;
|
2018-08-02 16:07:50 +00:00
|
|
|
}
|
2019-01-08 19:40:06 +00:00
|
|
|
return 0;
|
2018-08-02 16:07:50 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
|
|
|
hello_sock_close_timeout_poll(void *arg)
|
|
|
|
{
|
|
|
|
struct hello_context_t *ctx = arg;
|
|
|
|
SPDK_NOTICELOG("Connection closed\n");
|
|
|
|
|
|
|
|
spdk_poller_unregister(&ctx->time_out);
|
|
|
|
spdk_poller_unregister(&ctx->poller_in);
|
|
|
|
spdk_sock_close(&ctx->sock);
|
2018-09-26 07:55:28 +00:00
|
|
|
spdk_sock_group_close(&ctx->group);
|
2018-08-02 16:07:50 +00:00
|
|
|
|
2019-08-30 10:10:21 +00:00
|
|
|
spdk_app_stop(ctx->rc);
|
2018-09-26 07:55:28 +00:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
|
|
|
hello_sock_quit(struct hello_context_t *ctx, int rc)
|
|
|
|
{
|
|
|
|
ctx->rc = rc;
|
|
|
|
spdk_poller_unregister(&ctx->poller_out);
|
2018-12-14 11:17:46 +00:00
|
|
|
if (!ctx->time_out) {
|
|
|
|
ctx->time_out = spdk_poller_register(hello_sock_close_timeout_poll, ctx,
|
|
|
|
CLOSE_TIMEOUT_US);
|
|
|
|
}
|
2018-08-02 16:07:50 +00:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
|
|
|
hello_sock_recv_poll(void *arg)
|
|
|
|
{
|
|
|
|
struct hello_context_t *ctx = arg;
|
|
|
|
int rc;
|
|
|
|
char buf_in[BUFFER_SIZE];
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Get response
|
|
|
|
*/
|
|
|
|
rc = spdk_sock_recv(ctx->sock, buf_in, sizeof(buf_in) - 1);
|
|
|
|
|
|
|
|
if (rc <= 0) {
|
|
|
|
if (errno == EAGAIN || errno == EWOULDBLOCK) {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
SPDK_ERRLOG("spdk_sock_recv() failed, errno %d: %s\n",
|
|
|
|
errno, spdk_strerror(errno));
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (rc > 0) {
|
|
|
|
ctx->bytes_in += rc;
|
|
|
|
buf_in[rc] = '\0';
|
|
|
|
printf("%s", buf_in);
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
|
|
|
hello_sock_writev_poll(void *arg)
|
|
|
|
{
|
|
|
|
struct hello_context_t *ctx = arg;
|
|
|
|
int rc = 0;
|
|
|
|
char buf_out[BUFFER_SIZE];
|
|
|
|
struct iovec iov;
|
|
|
|
ssize_t n;
|
|
|
|
|
|
|
|
n = read(STDIN_FILENO, buf_out, sizeof(buf_out));
|
|
|
|
if (n == 0 || !g_is_running) {
|
|
|
|
/* EOF */
|
|
|
|
SPDK_NOTICELOG("Closing connection...\n");
|
2018-09-26 07:55:28 +00:00
|
|
|
hello_sock_quit(ctx, 0);
|
2018-08-02 16:07:50 +00:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
if (n > 0) {
|
|
|
|
/*
|
|
|
|
* Send message to the server
|
|
|
|
*/
|
|
|
|
iov.iov_base = buf_out;
|
|
|
|
iov.iov_len = n;
|
|
|
|
rc = spdk_sock_writev(ctx->sock, &iov, 1);
|
|
|
|
if (rc > 0) {
|
|
|
|
ctx->bytes_out += rc;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
|
|
|
hello_sock_connect(struct hello_context_t *ctx)
|
|
|
|
{
|
|
|
|
int rc;
|
|
|
|
char saddr[ADDR_STR_LEN], caddr[ADDR_STR_LEN];
|
2018-10-08 01:51:03 +00:00
|
|
|
uint16_t cport, sport;
|
2018-08-02 16:07:50 +00:00
|
|
|
|
|
|
|
SPDK_NOTICELOG("Connecting to the server on %s:%d\n", ctx->host, ctx->port);
|
|
|
|
|
sock: Add impl_name parameter in spdk_sock_listen/connect.
Purpose: With this patch,
(1)We can support using different sock implementations in
one application together.
(2)For one IP address managed by kernel, we can use different method
to listen/connect, e.g., posix, or uring. With this patch, we can
designate the specified sock implementation if impl_name is not NULL
and valid. Otherwise, spdk_sock_listen/connect will try to use the sock
implementations in the list by order if impl_name is NULL.
Without this patch, the app will always use the same type of sock implementation
if the order is fixed. For example, if we have posix and uring together,
the first one will always be uring.
Signed-off-by: Ziye Yang <ziye.yang@intel.com>
Change-Id: Ic49563f5025085471d356798e522ff7ab748f586
Reviewed-on: https://review.gerrithub.io/c/spdk/spdk/+/478140
Community-CI: Broadcom SPDK FC-NVMe CI <spdk-ci.pdl@broadcom.com>
Community-CI: SPDK CI Jenkins <sys_sgci@intel.com>
Tested-by: SPDK CI Jenkins <sys_sgci@intel.com>
Reviewed-by: Ben Walker <benjamin.walker@intel.com>
Reviewed-by: Shuhei Matsumoto <shuhei.matsumoto.xt@hitachi.com>
Reviewed-by: Jim Harris <james.r.harris@intel.com>
2019-12-17 13:57:10 +00:00
|
|
|
ctx->sock = spdk_sock_connect(ctx->host, ctx->port, NULL);
|
2018-08-02 16:07:50 +00:00
|
|
|
if (ctx->sock == NULL) {
|
|
|
|
SPDK_ERRLOG("connect error(%d): %s\n", errno, spdk_strerror(errno));
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2018-10-08 01:51:03 +00:00
|
|
|
rc = spdk_sock_getaddr(ctx->sock, saddr, sizeof(saddr), &sport, caddr, sizeof(caddr), &cport);
|
2018-08-02 16:07:50 +00:00
|
|
|
if (rc < 0) {
|
|
|
|
SPDK_ERRLOG("Cannot get connection addresses\n");
|
|
|
|
spdk_sock_close(&ctx->sock);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2018-10-08 01:51:03 +00:00
|
|
|
SPDK_NOTICELOG("Connection accepted from (%s, %hu) to (%s, %hu)\n", caddr, cport, saddr, sport);
|
2018-08-02 16:07:50 +00:00
|
|
|
|
|
|
|
fcntl(STDIN_FILENO, F_SETFL, fcntl(STDIN_FILENO, F_GETFL) | O_NONBLOCK);
|
|
|
|
|
|
|
|
g_is_running = true;
|
|
|
|
ctx->poller_in = spdk_poller_register(hello_sock_recv_poll, ctx, 0);
|
|
|
|
ctx->poller_out = spdk_poller_register(hello_sock_writev_poll, ctx, 0);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
hello_sock_cb(void *arg, struct spdk_sock_group *group, struct spdk_sock *sock)
|
|
|
|
{
|
|
|
|
ssize_t n;
|
|
|
|
char buf[BUFFER_SIZE];
|
|
|
|
struct iovec iov;
|
|
|
|
struct hello_context_t *ctx = arg;
|
|
|
|
|
|
|
|
n = spdk_sock_recv(sock, buf, sizeof(buf));
|
|
|
|
if (n < 0) {
|
|
|
|
if (errno == EAGAIN || errno == EWOULDBLOCK) {
|
|
|
|
SPDK_ERRLOG("spdk_sock_recv() failed, errno %d: %s\n",
|
|
|
|
errno, spdk_strerror(errno));
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
SPDK_ERRLOG("spdk_sock_recv() failed, errno %d: %s\n",
|
|
|
|
errno, spdk_strerror(errno));
|
|
|
|
}
|
|
|
|
|
|
|
|
if (n > 0) {
|
|
|
|
ctx->bytes_in += n;
|
|
|
|
iov.iov_base = buf;
|
|
|
|
iov.iov_len = n;
|
|
|
|
n = spdk_sock_writev(sock, &iov, 1);
|
|
|
|
if (n > 0) {
|
|
|
|
ctx->bytes_out += n;
|
|
|
|
}
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Connection closed */
|
|
|
|
SPDK_NOTICELOG("Connection closed\n");
|
|
|
|
spdk_sock_group_remove_sock(group, sock);
|
|
|
|
spdk_sock_close(&sock);
|
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
|
|
|
hello_sock_accept_poll(void *arg)
|
|
|
|
{
|
|
|
|
struct hello_context_t *ctx = arg;
|
|
|
|
struct spdk_sock *sock;
|
|
|
|
int rc;
|
|
|
|
int count = 0;
|
|
|
|
char saddr[ADDR_STR_LEN], caddr[ADDR_STR_LEN];
|
2018-10-08 01:51:03 +00:00
|
|
|
uint16_t cport, sport;
|
2018-08-02 16:07:50 +00:00
|
|
|
|
|
|
|
if (!g_is_running) {
|
2018-09-26 07:55:28 +00:00
|
|
|
hello_sock_quit(ctx, 0);
|
2018-08-02 16:07:50 +00:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
while (1) {
|
|
|
|
sock = spdk_sock_accept(ctx->sock);
|
|
|
|
if (sock != NULL) {
|
2018-09-26 07:55:28 +00:00
|
|
|
rc = spdk_sock_getaddr(sock, saddr, sizeof(saddr), &sport, caddr, sizeof(caddr), &cport);
|
|
|
|
if (rc < 0) {
|
|
|
|
SPDK_ERRLOG("Cannot get connection addresses\n");
|
|
|
|
spdk_sock_close(&ctx->sock);
|
|
|
|
return -1;
|
|
|
|
}
|
2018-08-02 16:07:50 +00:00
|
|
|
|
2018-10-08 01:51:03 +00:00
|
|
|
SPDK_NOTICELOG("Accepting a new connection from (%s, %hu) to (%s, %hu)\n",
|
|
|
|
caddr, cport, saddr, sport);
|
2018-08-02 16:07:50 +00:00
|
|
|
|
|
|
|
rc = spdk_sock_group_add_sock(ctx->group, sock,
|
|
|
|
hello_sock_cb, ctx);
|
|
|
|
|
|
|
|
if (rc < 0) {
|
|
|
|
spdk_sock_close(&sock);
|
|
|
|
SPDK_ERRLOG("failed\n");
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
count++;
|
|
|
|
} else {
|
|
|
|
if (errno != EAGAIN && errno != EWOULDBLOCK) {
|
|
|
|
SPDK_ERRLOG("accept error(%d): %s\n", errno, spdk_strerror(errno));
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return count;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
|
|
|
hello_sock_group_poll(void *arg)
|
|
|
|
{
|
|
|
|
struct hello_context_t *ctx = arg;
|
|
|
|
int rc;
|
|
|
|
|
|
|
|
rc = spdk_sock_group_poll(ctx->group);
|
|
|
|
if (rc < 0) {
|
|
|
|
SPDK_ERRLOG("Failed to poll sock_group=%p\n", ctx->group);
|
|
|
|
}
|
|
|
|
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
|
|
|
hello_sock_listen(struct hello_context_t *ctx)
|
|
|
|
{
|
sock: Add impl_name parameter in spdk_sock_listen/connect.
Purpose: With this patch,
(1)We can support using different sock implementations in
one application together.
(2)For one IP address managed by kernel, we can use different method
to listen/connect, e.g., posix, or uring. With this patch, we can
designate the specified sock implementation if impl_name is not NULL
and valid. Otherwise, spdk_sock_listen/connect will try to use the sock
implementations in the list by order if impl_name is NULL.
Without this patch, the app will always use the same type of sock implementation
if the order is fixed. For example, if we have posix and uring together,
the first one will always be uring.
Signed-off-by: Ziye Yang <ziye.yang@intel.com>
Change-Id: Ic49563f5025085471d356798e522ff7ab748f586
Reviewed-on: https://review.gerrithub.io/c/spdk/spdk/+/478140
Community-CI: Broadcom SPDK FC-NVMe CI <spdk-ci.pdl@broadcom.com>
Community-CI: SPDK CI Jenkins <sys_sgci@intel.com>
Tested-by: SPDK CI Jenkins <sys_sgci@intel.com>
Reviewed-by: Ben Walker <benjamin.walker@intel.com>
Reviewed-by: Shuhei Matsumoto <shuhei.matsumoto.xt@hitachi.com>
Reviewed-by: Jim Harris <james.r.harris@intel.com>
2019-12-17 13:57:10 +00:00
|
|
|
ctx->sock = spdk_sock_listen(ctx->host, ctx->port, NULL);
|
2018-08-02 16:07:50 +00:00
|
|
|
if (ctx->sock == NULL) {
|
|
|
|
SPDK_ERRLOG("Cannot create server socket\n");
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
SPDK_NOTICELOG("Listening connection on %s:%d\n", ctx->host, ctx->port);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Create sock group for server socket
|
|
|
|
*/
|
2019-05-14 18:40:20 +00:00
|
|
|
ctx->group = spdk_sock_group_create(NULL);
|
2018-08-02 16:07:50 +00:00
|
|
|
|
|
|
|
g_is_running = true;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Start acceptor and group poller
|
|
|
|
*/
|
|
|
|
ctx->poller_in = spdk_poller_register(hello_sock_accept_poll, ctx,
|
|
|
|
ACCEPT_TIMEOUT_US);
|
|
|
|
ctx->poller_out = spdk_poller_register(hello_sock_group_poll, ctx, 0);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
hello_sock_shutdown_cb(void)
|
|
|
|
{
|
|
|
|
g_is_running = false;
|
|
|
|
}
|
2018-09-26 07:55:28 +00:00
|
|
|
|
2018-08-02 16:07:50 +00:00
|
|
|
/*
|
|
|
|
* Our initial event that kicks off everything from main().
|
|
|
|
*/
|
|
|
|
static void
|
2019-08-30 10:10:21 +00:00
|
|
|
hello_start(void *arg1)
|
2018-08-02 16:07:50 +00:00
|
|
|
{
|
|
|
|
struct hello_context_t *ctx = arg1;
|
2019-08-30 10:10:21 +00:00
|
|
|
int rc;
|
2018-08-02 16:07:50 +00:00
|
|
|
|
|
|
|
SPDK_NOTICELOG("Successfully started the application\n");
|
|
|
|
|
|
|
|
if (ctx->is_server) {
|
|
|
|
rc = hello_sock_listen(ctx);
|
|
|
|
} else {
|
|
|
|
rc = hello_sock_connect(ctx);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (rc) {
|
|
|
|
spdk_app_stop(-1);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
int
|
|
|
|
main(int argc, char **argv)
|
|
|
|
{
|
|
|
|
struct spdk_app_opts opts = {};
|
|
|
|
int rc = 0;
|
|
|
|
struct hello_context_t hello_context = {};
|
|
|
|
|
|
|
|
/* Set default values in opts structure. */
|
|
|
|
spdk_app_opts_init(&opts);
|
|
|
|
opts.name = "hello_sock";
|
|
|
|
opts.shutdown_cb = hello_sock_shutdown_cb;
|
|
|
|
|
2018-08-14 15:29:11 +00:00
|
|
|
if ((rc = spdk_app_parse_args(argc, argv, &opts, "H:P:SV", NULL, hello_sock_parse_arg,
|
2018-08-02 16:07:50 +00:00
|
|
|
hello_sock_usage)) != SPDK_APP_PARSE_ARGS_SUCCESS) {
|
|
|
|
exit(rc);
|
|
|
|
}
|
|
|
|
hello_context.is_server = g_is_server;
|
|
|
|
hello_context.host = g_host;
|
|
|
|
hello_context.port = g_port;
|
|
|
|
hello_context.verbose = g_verbose;
|
|
|
|
|
2019-08-30 10:10:21 +00:00
|
|
|
rc = spdk_app_start(&opts, hello_start, &hello_context);
|
2018-08-02 16:07:50 +00:00
|
|
|
if (rc) {
|
|
|
|
SPDK_ERRLOG("ERROR starting application\n");
|
|
|
|
}
|
|
|
|
|
|
|
|
SPDK_NOTICELOG("Exiting from application\n");
|
|
|
|
|
|
|
|
if (hello_context.verbose) {
|
|
|
|
printf("** %d bytes received, %d bytes sent **\n",
|
|
|
|
hello_context.bytes_in, hello_context.bytes_out);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Gracefully close out all of the SPDK subsystems. */
|
|
|
|
spdk_app_fini();
|
|
|
|
return rc;
|
|
|
|
}
|