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

This patch presents link, id, task name, lqpn, as well as all sub counters of a QP counter. A QP counter is a dynamically allocated statistic counter that is bound with one or more QPs. It has several sub-counters, each is used for a different purpose. Examples: $ rdma stat qp show link mlx5_2/1 cntn 5 pid 31609 comm client.1 rx_write_requests 0 rx_read_requests 0 rx_atomic_requests 0 out_of_buffer 0 out_of_sequence 0 duplicate_request 0 rnr_nak_retry_err 0 packet_seq_err 0 implied_nak_seq_err 0 local_ack_timeout_err 0 resp_local_length_error 0 resp_cqe_error 0 req_cqe_error 0 req_remote_invalid_request 0 req_remote_access_errors 0 resp_remote_access_errors 0 resp_cqe_flush_error 0 req_cqe_flush_error 0 LQPN: <178> $ rdma stat show link rocep1s0f5/1 link rocep1s0f5/1 rx_write_requests 0 rx_read_requests 0 rx_atomic_requests 0 out_of_buffer 0 duplicate_request 0 rnr_nak_retry_err 0 packet_seq_err 0 implied_nak_seq_err 0 local_ack_timeout_err 0 resp_local_length_error 0 resp_cqe_error 0 req_cqe_error 0 req_remote_invalid_request 0 req_remote_access_errors 0 resp_remote_access_errors 0 resp_cqe_flush_error 0 req_cqe_flush_error 0 rp_cnp_ignored 0 rp_cnp_handled 0 np_ecn_marked_roce_packets 0 np_cnp_sent 0 $ rdma stat show link rocep1s0f5/1 -p link rocep1s0f5/1 rx_write_requests 0 rx_read_requests 0 rx_atomic_requests 0 out_of_buffer 0 duplicate_request 0 rnr_nak_retry_err 0 packet_seq_err 0 implied_nak_seq_err 0 local_ack_timeout_err 0 resp_local_length_error 0 resp_cqe_error 0 req_cqe_error 0 req_remote_invalid_request 0 req_remote_access_errors 0 resp_remote_access_errors 0 resp_cqe_flush_error 0 req_cqe_flush_error 0 rp_cnp_ignored 0 rp_cnp_handled 0 np_ecn_marked_roce_packets 0 np_cnp_sent 0 Signed-off-by: Mark Zhang <markz@mellanox.com> Signed-off-by: Leon Romanovsky <leonro@mellanox.com> Signed-off-by: Stephen Hemminger <stephen@networkplumber.org>
201 lines
4.1 KiB
C
201 lines
4.1 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 | statistic | 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 },
|
|
{ "statistic", cmd_stat },
|
|
{ 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;
|
|
}
|