1 // SPDX-License-Identifier: GPL-2.0 OR Linux-OpenIB
4 * Authors: Leon Romanovsky <leonro@mellanox.com>
10 static int res_help(struct rd
*rd
)
12 pr_out("Usage: %s resource\n", rd
->filename
);
13 pr_out(" resource show [DEV]\n");
14 pr_out(" resource show [qp|cm_id|pd|mr|cq]\n");
15 pr_out(" resource show qp link [DEV/PORT]\n");
16 pr_out(" resource show qp link [DEV/PORT] [FILTER-NAME FILTER-VALUE]\n");
17 pr_out(" resource show cm_id link [DEV/PORT]\n");
18 pr_out(" resource show cm_id link [DEV/PORT] [FILTER-NAME FILTER-VALUE]\n");
19 pr_out(" resource show cq link [DEV/PORT]\n");
20 pr_out(" resource show cq link [DEV/PORT] [FILTER-NAME FILTER-VALUE]\n");
21 pr_out(" resource show pd dev [DEV]\n");
22 pr_out(" resource show pd dev [DEV] [FILTER-NAME FILTER-VALUE]\n");
23 pr_out(" resource show mr dev [DEV]\n");
24 pr_out(" resource show mr dev [DEV] [FILTER-NAME FILTER-VALUE]\n");
28 static int res_print_summary(struct rd
*rd
, struct nlattr
**tb
)
30 struct nlattr
*nla_table
= tb
[RDMA_NLDEV_ATTR_RES_SUMMARY
];
31 struct nlattr
*nla_entry
;
36 mnl_attr_for_each_nested(nla_entry
, nla_table
) {
37 struct nlattr
*nla_line
[RDMA_NLDEV_ATTR_MAX
] = {};
39 err
= mnl_attr_parse_nested(nla_entry
, rd_attr_cb
, nla_line
);
43 if (!nla_line
[RDMA_NLDEV_ATTR_RES_SUMMARY_ENTRY_NAME
] ||
44 !nla_line
[RDMA_NLDEV_ATTR_RES_SUMMARY_ENTRY_CURR
]) {
48 name
= mnl_attr_get_str(nla_line
[RDMA_NLDEV_ATTR_RES_SUMMARY_ENTRY_NAME
]);
49 curr
= mnl_attr_get_u64(nla_line
[RDMA_NLDEV_ATTR_RES_SUMMARY_ENTRY_CURR
]);
52 nla_line
[RDMA_NLDEV_ATTR_RES_SUMMARY_ENTRY_CURR
]);
57 static int res_no_args_idx_parse_cb(const struct nlmsghdr
*nlh
, void *data
)
62 static int res_no_args_parse_cb(const struct nlmsghdr
*nlh
, void *data
)
64 struct nlattr
*tb
[RDMA_NLDEV_ATTR_MAX
] = {};
69 mnl_attr_parse(nlh
, 0, rd_attr_cb
, tb
);
70 if (!tb
[RDMA_NLDEV_ATTR_DEV_INDEX
] ||
71 !tb
[RDMA_NLDEV_ATTR_DEV_NAME
] ||
72 !tb
[RDMA_NLDEV_ATTR_RES_SUMMARY
])
75 idx
= mnl_attr_get_u32(tb
[RDMA_NLDEV_ATTR_DEV_INDEX
]);
76 name
= mnl_attr_get_str(tb
[RDMA_NLDEV_ATTR_DEV_NAME
]);
77 if (rd
->json_output
) {
78 jsonw_uint_field(rd
->jw
, "ifindex", idx
);
79 jsonw_string_field(rd
->jw
, "ifname", name
);
81 pr_out("%u: %s: ", idx
, name
);
84 res_print_summary(rd
, tb
);
91 int _res_send_idx_msg(struct rd
*rd
, uint32_t command
, mnl_cb_t callback
,
92 uint32_t idx
, uint32_t id
)
94 uint32_t flags
= NLM_F_REQUEST
| NLM_F_ACK
;
98 rd_prepare_msg(rd
, command
, &seq
, flags
);
99 mnl_attr_put_u32(rd
->nlh
, RDMA_NLDEV_ATTR_DEV_INDEX
, rd
->dev_idx
);
101 mnl_attr_put_u32(rd
->nlh
,
102 RDMA_NLDEV_ATTR_PORT_INDEX
, rd
->port_idx
);
104 mnl_attr_put_u32(rd
->nlh
, id
, idx
);
106 if (command
== RDMA_NLDEV_CMD_STAT_GET
)
107 mnl_attr_put_u32(rd
->nlh
, RDMA_NLDEV_ATTR_STAT_RES
,
108 RDMA_NLDEV_ATTR_RES_MR
);
110 ret
= rd_send_msg(rd
);
115 jsonw_start_object(rd
->jw
);
116 ret
= rd_recv_msg(rd
, callback
, rd
, seq
);
118 jsonw_end_object(rd
->jw
);
122 int _res_send_msg(struct rd
*rd
, uint32_t command
, mnl_cb_t callback
)
124 uint32_t flags
= NLM_F_REQUEST
| NLM_F_ACK
;
128 if (command
!= RDMA_NLDEV_CMD_RES_GET
)
131 rd_prepare_msg(rd
, command
, &seq
, flags
);
132 mnl_attr_put_u32(rd
->nlh
, RDMA_NLDEV_ATTR_DEV_INDEX
, rd
->dev_idx
);
134 mnl_attr_put_u32(rd
->nlh
,
135 RDMA_NLDEV_ATTR_PORT_INDEX
, rd
->port_idx
);
137 if (command
== RDMA_NLDEV_CMD_STAT_GET
)
138 mnl_attr_put_u32(rd
->nlh
, RDMA_NLDEV_ATTR_STAT_RES
,
139 RDMA_NLDEV_ATTR_RES_MR
);
141 ret
= rd_send_msg(rd
);
146 jsonw_start_object(rd
->jw
);
147 ret
= rd_recv_msg(rd
, callback
, rd
, seq
);
149 jsonw_end_object(rd
->jw
);
153 const char *qp_types_to_str(uint8_t idx
)
155 static const char * const qp_types_str
[] = { "SMI", "GSI", "RC",
156 "UC", "UD", "RAW_IPV6",
158 "UNKNOWN", "RAW_PACKET",
159 "XRC_INI", "XRC_TGT",
163 if (idx
< ARRAY_SIZE(qp_types_str
) && qp_types_str
[idx
])
164 return qp_types_str
[idx
];
168 void print_comm(struct rd
*rd
, const char *str
, struct nlattr
**nla_line
)
175 if (rd
->json_output
) {
176 /* Don't beatify output in JSON format */
177 jsonw_string_field(rd
->jw
, "comm", str
);
181 if (nla_line
[RDMA_NLDEV_ATTR_RES_PID
])
182 snprintf(tmp
, sizeof(tmp
), "%s", str
);
184 snprintf(tmp
, sizeof(tmp
), "[%s]", str
);
186 pr_out("comm %s ", tmp
);
189 void print_dev(struct rd
*rd
, uint32_t idx
, const char *name
)
191 if (rd
->json_output
) {
192 jsonw_uint_field(rd
->jw
, "ifindex", idx
);
193 jsonw_string_field(rd
->jw
, "ifname", name
);
195 pr_out("dev %s ", name
);
199 void print_link(struct rd
*rd
, uint32_t idx
, const char *name
, uint32_t port
,
200 struct nlattr
**nla_line
)
202 if (rd
->json_output
) {
203 jsonw_uint_field(rd
->jw
, "ifindex", idx
);
205 if (nla_line
[RDMA_NLDEV_ATTR_PORT_INDEX
])
206 jsonw_uint_field(rd
->jw
, "port", port
);
208 jsonw_string_field(rd
->jw
, "ifname", name
);
210 if (nla_line
[RDMA_NLDEV_ATTR_PORT_INDEX
])
211 pr_out("link %s/%u ", name
, port
);
213 pr_out("link %s/- ", name
);
217 char *get_task_name(uint32_t pid
)
225 if (asprintf(&comm
, "/proc/%d/comm", pid
) < 0)
228 f
= fopen(comm
, "r");
233 if (fscanf(f
, "%ms\n", &comm
) != 1)
241 void print_key(struct rd
*rd
, const char *name
, uint64_t val
,
242 struct nlattr
*nlattr
)
248 jsonw_xint_field(rd
->jw
, name
, val
);
250 pr_out("%s 0x%" PRIx64
" ", name
, val
);
253 void res_print_uint(struct rd
*rd
, const char *name
, uint64_t val
,
254 struct nlattr
*nlattr
)
260 jsonw_u64_field(rd
->jw
, name
, val
);
262 pr_out("%s %" PRIu64
" ", name
, val
);
265 RES_FUNC(res_no_args
, RDMA_NLDEV_CMD_RES_GET
, NULL
, true, 0);
267 static int res_show(struct rd
*rd
)
269 const struct rd_cmd cmds
[] = {
270 { NULL
, res_no_args
},
272 { "cm_id", res_cm_id
},
280 * Special case to support "rdma res show DEV_NAME"
282 if (rd_argc(rd
) == 1 && dev_map_lookup(rd
, false))
283 return rd_exec_dev(rd
, _res_no_args
);
285 return rd_exec_cmd(rd
, cmds
, "parameter");
288 int cmd_res(struct rd
*rd
)
290 const struct rd_cmd cmds
[] = {
292 { "show", res_show
},
293 { "list", res_show
},
294 { "help", res_help
},
298 return rd_exec_cmd(rd
, cmds
, "resource command");