mirror of
https://git.proxmox.com/git/mirror_iproute2
synced 2025-10-05 10:57:12 +00:00

Enrich rdmatool with an option to query rdma subsystem parameter whether rdma devices are shared among multiple network namespaces or exclusive to single network namespace. rdma tool command examples and output. $ rdma system show netns shared $ rdma system set netns exclusive $ rdma system show netns exclusive Reviewed-by: Leon Romanovsky <leonro@mellanox.com> Signed-off-by: Parav Pandit <parav@mellanox.com> Signed-off-by: David Ahern <dsahern@gmail.com>
200 lines
4.0 KiB
C
200 lines
4.0 KiB
C
// SPDX-License-Identifier: GPL-2.0 OR Linux-OpenIB
|
|
/*
|
|
* rdma.c RDMA tool
|
|
* Authors: Leon Romanovsky <leonro@mellanox.com>
|
|
*/
|
|
|
|
#include "rdma.h"
|
|
#include "SNAPSHOT.h"
|
|
|
|
static void help(char *name)
|
|
{
|
|
pr_out("Usage: %s [ OPTIONS ] OBJECT { COMMAND | help }\n"
|
|
" %s [ -f[orce] ] -b[atch] filename\n"
|
|
"where OBJECT := { dev | link | resource | system | help }\n"
|
|
" OPTIONS := { -V[ersion] | -d[etails] | -j[son] | -p[retty]}\n", name, name);
|
|
}
|
|
|
|
static int cmd_help(struct rd *rd)
|
|
{
|
|
help(rd->filename);
|
|
return 0;
|
|
}
|
|
|
|
static int rd_cmd(struct rd *rd, int argc, char **argv)
|
|
{
|
|
const struct rd_cmd cmds[] = {
|
|
{ NULL, cmd_help },
|
|
{ "help", cmd_help },
|
|
{ "dev", cmd_dev },
|
|
{ "link", cmd_link },
|
|
{ "resource", cmd_res },
|
|
{ "system", cmd_sys },
|
|
{ 0 }
|
|
};
|
|
|
|
rd->argc = argc;
|
|
rd->argv = argv;
|
|
|
|
return rd_exec_cmd(rd, cmds, "object");
|
|
}
|
|
|
|
static int rd_batch(struct rd *rd, const char *name, bool force)
|
|
{
|
|
char *line = NULL;
|
|
size_t len = 0;
|
|
int ret = 0;
|
|
|
|
if (name && strcmp(name, "-") != 0) {
|
|
if (!freopen(name, "r", stdin)) {
|
|
pr_err("Cannot open file \"%s\" for reading: %s\n",
|
|
name, strerror(errno));
|
|
return errno;
|
|
}
|
|
}
|
|
|
|
cmdlineno = 0;
|
|
while (getcmdline(&line, &len, stdin) != -1) {
|
|
char *largv[512];
|
|
int largc;
|
|
|
|
largc = makeargs(line, largv, ARRAY_SIZE(largv));
|
|
if (!largc)
|
|
continue; /* blank line */
|
|
|
|
ret = rd_cmd(rd, largc, largv);
|
|
if (ret) {
|
|
pr_err("Command failed %s:%d\n", name, cmdlineno);
|
|
if (!force)
|
|
break;
|
|
}
|
|
}
|
|
|
|
free(line);
|
|
|
|
return ret;
|
|
}
|
|
|
|
static int rd_init(struct rd *rd, char *filename)
|
|
{
|
|
uint32_t seq;
|
|
int ret;
|
|
|
|
rd->filename = filename;
|
|
INIT_LIST_HEAD(&rd->dev_map_list);
|
|
INIT_LIST_HEAD(&rd->filter_list);
|
|
|
|
if (rd->json_output) {
|
|
rd->jw = jsonw_new(stdout);
|
|
if (!rd->jw) {
|
|
pr_err("Failed to create JSON writer\n");
|
|
return -ENOMEM;
|
|
}
|
|
jsonw_pretty(rd->jw, rd->pretty_output);
|
|
}
|
|
|
|
rd->buff = malloc(MNL_SOCKET_BUFFER_SIZE);
|
|
if (!rd->buff)
|
|
return -ENOMEM;
|
|
|
|
rd_prepare_msg(rd, RDMA_NLDEV_CMD_GET,
|
|
&seq, (NLM_F_REQUEST | NLM_F_ACK | NLM_F_DUMP));
|
|
ret = rd_send_msg(rd);
|
|
if (ret)
|
|
return ret;
|
|
|
|
return rd_recv_msg(rd, rd_dev_init_cb, rd, seq);
|
|
}
|
|
|
|
static void rd_cleanup(struct rd *rd)
|
|
{
|
|
if (rd->json_output)
|
|
jsonw_destroy(&rd->jw);
|
|
rd_free(rd);
|
|
}
|
|
|
|
int main(int argc, char **argv)
|
|
{
|
|
static const struct option long_options[] = {
|
|
{ "version", no_argument, NULL, 'V' },
|
|
{ "help", no_argument, NULL, 'h' },
|
|
{ "json", no_argument, NULL, 'j' },
|
|
{ "pretty", no_argument, NULL, 'p' },
|
|
{ "details", no_argument, NULL, 'd' },
|
|
{ "force", no_argument, NULL, 'f' },
|
|
{ "batch", required_argument, NULL, 'b' },
|
|
{ NULL, 0, NULL, 0 }
|
|
};
|
|
bool show_driver_details = false;
|
|
const char *batch_file = NULL;
|
|
bool pretty_output = false;
|
|
bool show_details = false;
|
|
bool json_output = false;
|
|
bool force = false;
|
|
struct rd rd = {};
|
|
char *filename;
|
|
int opt;
|
|
int err;
|
|
|
|
filename = basename(argv[0]);
|
|
|
|
while ((opt = getopt_long(argc, argv, ":Vhdpjfb:",
|
|
long_options, NULL)) >= 0) {
|
|
switch (opt) {
|
|
case 'V':
|
|
printf("%s utility, iproute2-ss%s\n",
|
|
filename, SNAPSHOT);
|
|
return EXIT_SUCCESS;
|
|
case 'p':
|
|
pretty_output = true;
|
|
break;
|
|
case 'd':
|
|
if (show_details)
|
|
show_driver_details = true;
|
|
else
|
|
show_details = true;
|
|
break;
|
|
case 'j':
|
|
json_output = true;
|
|
break;
|
|
case 'f':
|
|
force = true;
|
|
break;
|
|
case 'b':
|
|
batch_file = optarg;
|
|
break;
|
|
case 'h':
|
|
help(filename);
|
|
return EXIT_SUCCESS;
|
|
case ':':
|
|
pr_err("-%c option requires an argument\n", optopt);
|
|
return EXIT_FAILURE;
|
|
default:
|
|
pr_err("Unknown option.\n");
|
|
help(filename);
|
|
return EXIT_FAILURE;
|
|
}
|
|
}
|
|
|
|
argc -= optind;
|
|
argv += optind;
|
|
|
|
rd.show_details = show_details;
|
|
rd.show_driver_details = show_driver_details;
|
|
rd.json_output = json_output;
|
|
rd.pretty_output = pretty_output;
|
|
|
|
err = rd_init(&rd, filename);
|
|
if (err)
|
|
goto out;
|
|
|
|
if (batch_file)
|
|
err = rd_batch(&rd, batch_file, force);
|
|
else
|
|
err = rd_cmd(&rd, argc, argv);
|
|
out:
|
|
/* Always cleanup */
|
|
rd_cleanup(&rd);
|
|
return err ? EXIT_FAILURE : EXIT_SUCCESS;
|
|
}
|