1 // SPDX-License-Identifier: GPL-2.0 OR Linux-OpenIB
4 * Authors: Leon Romanovsky <leonro@mellanox.com>
9 static int link_help(struct rd
*rd
)
11 pr_out("Usage: %s link show [DEV/PORT_INDEX]\n", rd
->filename
);
15 static const char *caps_to_str(uint32_t idx
)
17 #define RDMA_PORT_FLAGS_LOW(x) \
29 x(SYS_IMAGE_GUID, 11) \
30 x(PKEY_SW_EXT_PORT_TRAP, 12) \
32 x(EXTENDED_SPEEDS, 14) \
40 x(CAP_MASK_NOTICE, 22) \
44 x(OTHER_LOCAL_CHANGES, 26) \
45 x(LINK_SPPED_WIDTH, 27) \
46 x(VENDOR_SPECIFIC_MADS, 28) \
47 x(MULT_PKER_TRAP, 29) \
51 #define RDMA_PORT_FLAGS_HIGH(x) \
55 x(SWITCH_POR_STATE_TABLE, 3) \
60 * Separation below is needed to allow compilation of rdmatool
61 * on 32bits systems. On such systems, C-enum is limited to be
62 * int and can't hold more than 32 bits.
64 enum { RDMA_PORT_FLAGS_LOW(RDMA_BITMAP_ENUM
) };
65 enum { RDMA_PORT_FLAGS_HIGH(RDMA_BITMAP_ENUM
) };
67 static const char * const
68 rdma_port_names_low
[] = { RDMA_PORT_FLAGS_LOW(RDMA_BITMAP_NAMES
) };
69 static const char * const
70 rdma_port_names_high
[] = { RDMA_PORT_FLAGS_HIGH(RDMA_BITMAP_NAMES
) };
72 #undef RDMA_PORT_FLAGS_LOW
73 #undef RDMA_PORT_FLAGS_HIGH
75 if (idx
< ARRAY_SIZE(rdma_port_names_low
) && rdma_port_names_low
[idx
])
76 return rdma_port_names_low
[idx
];
78 high_idx
= idx
- ARRAY_SIZE(rdma_port_names_low
);
79 if (high_idx
< ARRAY_SIZE(rdma_port_names_high
) &&
80 rdma_port_names_high
[high_idx
])
81 return rdma_port_names_high
[high_idx
];
86 static void link_print_caps(struct rd
*rd
, struct nlattr
**tb
)
91 if (!tb
[RDMA_NLDEV_ATTR_CAP_FLAGS
])
94 caps
= mnl_attr_get_u64(tb
[RDMA_NLDEV_ATTR_CAP_FLAGS
]);
96 if (rd
->json_output
) {
97 jsonw_name(rd
->jw
, "caps");
98 jsonw_start_array(rd
->jw
);
100 pr_out("\n caps: <");
102 for (idx
= 0; caps
; idx
++) {
104 if (rd
->json_output
) {
105 jsonw_string(rd
->jw
, caps_to_str(idx
));
107 pr_out("%s", caps_to_str(idx
));
116 jsonw_end_array(rd
->jw
);
121 static void link_print_subnet_prefix(struct rd
*rd
, struct nlattr
**tb
)
123 uint64_t subnet_prefix
;
127 if (!tb
[RDMA_NLDEV_ATTR_SUBNET_PREFIX
])
130 subnet_prefix
= mnl_attr_get_u64(tb
[RDMA_NLDEV_ATTR_SUBNET_PREFIX
]);
131 memcpy(vp
, &subnet_prefix
, sizeof(uint64_t));
132 snprintf(str
, 32, "%04x:%04x:%04x:%04x", vp
[3], vp
[2], vp
[1], vp
[0]);
134 jsonw_string_field(rd
->jw
, "subnet_prefix", str
);
136 pr_out("subnet_prefix %s ", str
);
139 static void link_print_lid(struct rd
*rd
, struct nlattr
**tb
)
143 if (!tb
[RDMA_NLDEV_ATTR_LID
])
146 lid
= mnl_attr_get_u32(tb
[RDMA_NLDEV_ATTR_LID
]);
148 jsonw_uint_field(rd
->jw
, "lid", lid
);
150 pr_out("lid %u ", lid
);
153 static void link_print_sm_lid(struct rd
*rd
, struct nlattr
**tb
)
157 if (!tb
[RDMA_NLDEV_ATTR_SM_LID
])
160 sm_lid
= mnl_attr_get_u32(tb
[RDMA_NLDEV_ATTR_SM_LID
]);
162 jsonw_uint_field(rd
->jw
, "sm_lid", sm_lid
);
164 pr_out("sm_lid %u ", sm_lid
);
167 static void link_print_lmc(struct rd
*rd
, struct nlattr
**tb
)
171 if (!tb
[RDMA_NLDEV_ATTR_LMC
])
174 lmc
= mnl_attr_get_u8(tb
[RDMA_NLDEV_ATTR_LMC
]);
176 jsonw_uint_field(rd
->jw
, "lmc", lmc
);
178 pr_out("lmc %u ", lmc
);
181 static const char *link_state_to_str(uint8_t link_state
)
183 static const char * const link_state_str
[] = { "NOP", "DOWN",
187 if (link_state
< ARRAY_SIZE(link_state_str
))
188 return link_state_str
[link_state
];
192 static void link_print_state(struct rd
*rd
, struct nlattr
**tb
)
196 if (!tb
[RDMA_NLDEV_ATTR_PORT_STATE
])
199 state
= mnl_attr_get_u8(tb
[RDMA_NLDEV_ATTR_PORT_STATE
]);
201 jsonw_string_field(rd
->jw
, "state", link_state_to_str(state
));
203 pr_out("state %s ", link_state_to_str(state
));
206 static const char *phys_state_to_str(uint8_t phys_state
)
208 static const char * const phys_state_str
[] = { "NOP", "SLEEP",
209 "POLLING", "DISABLED",
211 "LINK_ERROR_RECOVER",
212 "PHY_TEST", "UNKNOWN",
214 "UNKNOWN", "OPA_TEST" };
215 if (phys_state
< ARRAY_SIZE(phys_state_str
))
216 return phys_state_str
[phys_state
];
220 static void link_print_phys_state(struct rd
*rd
, struct nlattr
**tb
)
224 if (!tb
[RDMA_NLDEV_ATTR_PORT_PHYS_STATE
])
227 phys_state
= mnl_attr_get_u8(tb
[RDMA_NLDEV_ATTR_PORT_PHYS_STATE
]);
229 jsonw_string_field(rd
->jw
, "physical_state",
230 phys_state_to_str(phys_state
));
232 pr_out("physical_state %s ", phys_state_to_str(phys_state
));
235 static void link_print_netdev(struct rd
*rd
, struct nlattr
**tb
)
237 const char *netdev_name
;
240 if (!tb
[RDMA_NLDEV_ATTR_NDEV_NAME
] || !tb
[RDMA_NLDEV_ATTR_NDEV_INDEX
])
243 netdev_name
= mnl_attr_get_str(tb
[RDMA_NLDEV_ATTR_NDEV_NAME
]);
244 idx
= mnl_attr_get_u32(tb
[RDMA_NLDEV_ATTR_NDEV_INDEX
]);
245 if (rd
->json_output
) {
246 jsonw_string_field(rd
->jw
, "netdev", netdev_name
);
247 jsonw_uint_field(rd
->jw
, "netdev_index", idx
);
249 pr_out("netdev %s ", netdev_name
);
250 if (rd
->show_details
)
251 pr_out("netdev_index %u ", idx
);
255 static int link_parse_cb(const struct nlmsghdr
*nlh
, void *data
)
257 struct nlattr
*tb
[RDMA_NLDEV_ATTR_MAX
] = {};
258 struct rd
*rd
= data
;
262 mnl_attr_parse(nlh
, 0, rd_attr_cb
, tb
);
263 if (!tb
[RDMA_NLDEV_ATTR_DEV_INDEX
] || !tb
[RDMA_NLDEV_ATTR_DEV_NAME
])
266 if (!tb
[RDMA_NLDEV_ATTR_PORT_INDEX
]) {
267 pr_err("This tool doesn't support switches yet\n");
271 idx
= mnl_attr_get_u32(tb
[RDMA_NLDEV_ATTR_DEV_INDEX
]);
272 port
= mnl_attr_get_u32(tb
[RDMA_NLDEV_ATTR_PORT_INDEX
]);
273 snprintf(name
, 32, "%s/%u",
274 mnl_attr_get_str(tb
[RDMA_NLDEV_ATTR_DEV_NAME
]), port
);
276 if (rd
->json_output
) {
277 jsonw_uint_field(rd
->jw
, "ifindex", idx
);
278 jsonw_uint_field(rd
->jw
, "port", port
);
279 jsonw_string_field(rd
->jw
, "ifname", name
);
282 pr_out("%u/%u: %s: ", idx
, port
, name
);
285 link_print_subnet_prefix(rd
, tb
);
286 link_print_lid(rd
, tb
);
287 link_print_sm_lid(rd
, tb
);
288 link_print_lmc(rd
, tb
);
289 link_print_state(rd
, tb
);
290 link_print_phys_state(rd
, tb
);
291 link_print_netdev(rd
, tb
);
292 if (rd
->show_details
)
293 link_print_caps(rd
, tb
);
295 if (!rd
->json_output
)
300 static int link_no_args(struct rd
*rd
)
305 rd_prepare_msg(rd
, RDMA_NLDEV_CMD_PORT_GET
, &seq
,
306 (NLM_F_REQUEST
| NLM_F_ACK
));
307 mnl_attr_put_u32(rd
->nlh
, RDMA_NLDEV_ATTR_DEV_INDEX
, rd
->dev_idx
);
308 mnl_attr_put_u32(rd
->nlh
, RDMA_NLDEV_ATTR_PORT_INDEX
, rd
->port_idx
);
309 ret
= rd_send_msg(rd
);
314 jsonw_start_object(rd
->jw
);
315 ret
= rd_recv_msg(rd
, link_parse_cb
, rd
, seq
);
317 jsonw_end_object(rd
->jw
);
321 static int link_one_show(struct rd
*rd
)
323 const struct rd_cmd cmds
[] = {
324 { NULL
, link_no_args
},
331 return rd_exec_cmd(rd
, cmds
, "parameter");
334 static int link_show(struct rd
*rd
)
336 return rd_exec_link(rd
, link_one_show
, true);
339 int cmd_link(struct rd
*rd
)
341 const struct rd_cmd cmds
[] = {
343 { "show", link_show
},
344 { "list", link_show
},
345 { "help", link_help
},
349 return rd_exec_cmd(rd
, cmds
, "link command");