blob: 0ee8a0dbaaa20f330630e1dc7ef7c74f7bf314c0 [file] [log] [blame]
/*
* rss.c - netlink implementation of RSS context commands
*
* Implementation of "ethtool -x <dev>"
*/
#include <errno.h>
#include <string.h>
#include <stdio.h>
#include "../internal.h"
#include "../common.h"
#include "netlink.h"
#include "strset.h"
#include "parser.h"
struct cb_args {
struct nl_context *nlctx;
u32 num_rings;
};
void dump_json_rss_info(struct cmd_context *ctx, u32 *indir_table,
u32 indir_size, u8 *hkey, u32 hkey_size,
const struct stringset *hash_funcs, u8 hfunc,
u32 input_xfrm)
{
unsigned int i;
open_json_object(NULL);
print_string(PRINT_JSON, "ifname", NULL, ctx->devname);
if (indir_size) {
open_json_array("rss-indirection-table", NULL);
for (i = 0; i < indir_size; i++)
print_uint(PRINT_JSON, NULL, NULL, indir_table[i]);
close_json_array("\n");
}
if (hkey_size) {
open_json_array("rss-hash-key", NULL);
for (i = 0; i < hkey_size; i++)
print_uint(PRINT_JSON, NULL, NULL, (u8)hkey[i]);
close_json_array("\n");
}
if (hfunc) {
for (i = 0; i < get_count(hash_funcs); i++) {
if (hfunc & (1 << i)) {
print_string(PRINT_JSON, "rss-hash-function",
NULL, get_string(hash_funcs, i));
open_json_object("rss-input-transformation");
print_bool(PRINT_JSON, "symmetric-xor", NULL,
(input_xfrm & RXH_XFRM_SYM_XOR) ?
true : false);
close_json_object();
break;
}
}
}
close_json_object();
}
/* There is no netlink equivalent for ETHTOOL_GRXRINGS. */
static int get_num_rings(struct cb_args *args)
{
struct nl_context *nlctx = args->nlctx;
struct cmd_context *ctx = nlctx->ctx;
struct ethtool_rxnfc ring_count = {
.cmd = ETHTOOL_GRXRINGS,
};
int ret;
ret = ioctl_init(ctx, false);
if (ret)
return ret;
ret = send_ioctl(ctx, &ring_count);
if (ret) {
perror("Cannot get RX ring count");
return ret;
}
args->num_rings = (u32)ring_count.data;
return 0;
}
int rss_reply_cb(const struct nlmsghdr *nlhdr, void *data)
{
const struct nlattr *tb[ETHTOOL_A_RSS_MAX + 1] = {};
unsigned int indir_bytes = 0, hkey_bytes = 0;
DECLARE_ATTR_TB_INFO(tb);
struct cb_args *args = data;
struct nl_context *nlctx = args->nlctx;
const struct stringset *hash_funcs;
u32 rss_hfunc = 0, indir_size;
u32 *indir_table = NULL;
u32 input_xfrm = 0;
u8 *hkey = NULL;
bool silent;
int err_ret;
int ret;
silent = nlctx->is_dump || nlctx->is_monitor;
err_ret = silent ? MNL_CB_OK : MNL_CB_ERROR;
ret = mnl_attr_parse(nlhdr, GENL_HDRLEN, attr_cb, &tb_info);
if (ret < 0)
return err_ret;
nlctx->devname = get_dev_name(tb[ETHTOOL_A_RSS_HEADER]);
if (!dev_ok(nlctx))
return err_ret;
show_cr();
if (tb[ETHTOOL_A_RSS_HFUNC])
rss_hfunc = mnl_attr_get_u32(tb[ETHTOOL_A_RSS_HFUNC]);
if (tb[ETHTOOL_A_RSS_INDIR]) {
indir_bytes = mnl_attr_get_payload_len(tb[ETHTOOL_A_RSS_INDIR]);
indir_table = mnl_attr_get_payload(tb[ETHTOOL_A_RSS_INDIR]);
}
if (tb[ETHTOOL_A_RSS_HKEY]) {
hkey_bytes = mnl_attr_get_payload_len(tb[ETHTOOL_A_RSS_HKEY]);
hkey = mnl_attr_get_payload(tb[ETHTOOL_A_RSS_HKEY]);
}
if (tb[ETHTOOL_A_RSS_INPUT_XFRM])
input_xfrm = mnl_attr_get_u32(tb[ETHTOOL_A_RSS_INPUT_XFRM]);
/* Fetch RSS hash functions and their status and print */
if (!nlctx->is_monitor) {
ret = netlink_init_ethnl2_socket(nlctx);
if (ret < 0)
return MNL_CB_ERROR;
}
hash_funcs = global_stringset(ETH_SS_RSS_HASH_FUNCS,
nlctx->ethnl2_socket);
ret = mnl_attr_parse(nlhdr, GENL_HDRLEN, attr_cb, &tb_info);
if (ret < 0)
return silent ? MNL_CB_OK : MNL_CB_ERROR;
ret = get_num_rings(args);
if (ret < 0)
return MNL_CB_ERROR;
indir_size = indir_bytes / sizeof(u32);
if (is_json_context()) {
dump_json_rss_info(nlctx->ctx, (u32 *)indir_table, indir_size,
hkey, hkey_bytes, hash_funcs, rss_hfunc,
input_xfrm);
} else {
print_indir_table(nlctx->ctx, args->num_rings,
indir_size, (u32 *)indir_table);
print_rss_hkey(hkey, hkey_bytes);
printf("RSS hash function:\n");
if (!rss_hfunc) {
printf(" Operation not supported\n");
return 0;
}
for (unsigned int i = 0; i < get_count(hash_funcs); i++) {
printf(" %s: %s\n", get_string(hash_funcs, i),
(rss_hfunc & (1 << i)) ? "on" : "off");
}
printf("RSS input transformation:\n");
printf(" symmetric-xor: %s\n",
(input_xfrm & RXH_XFRM_SYM_XOR) ? "on" : "off");
}
return MNL_CB_OK;
}
/* RSS_GET */
static const struct param_parser grss_params[] = {
{
.arg = "context",
.type = ETHTOOL_A_RSS_CONTEXT,
.handler = nl_parse_direct_u32,
.min_argc = 1,
},
{}
};
int nl_grss(struct cmd_context *ctx)
{
struct nl_context *nlctx = ctx->nlctx;
struct nl_socket *nlsk = nlctx->ethnl_socket;
struct nl_msg_buff *msgbuff;
struct cb_args args = {};
int ret;
nlctx->cmd = "-x";
nlctx->argp = ctx->argp;
nlctx->argc = ctx->argc;
nlctx->devname = ctx->devname;
nlsk = nlctx->ethnl_socket;
msgbuff = &nlsk->msgbuff;
if (netlink_cmd_check(ctx, ETHTOOL_MSG_RSS_GET, true))
return -EOPNOTSUPP;
ret = msg_init(nlctx, msgbuff, ETHTOOL_MSG_RSS_GET,
NLM_F_REQUEST | NLM_F_ACK);
if (ret < 0)
return 1;
if (ethnla_fill_header(msgbuff, ETHTOOL_A_RSS_HEADER,
ctx->devname, 0))
return -EMSGSIZE;
ret = nl_parser(nlctx, grss_params, NULL, PARSER_GROUP_NONE, NULL);
if (ret < 0)
goto err;
ret = nlsock_sendmsg(nlsk, NULL);
if (ret < 0)
goto err;
args.nlctx = nlctx;
new_json_obj(ctx->json);
ret = nlsock_process_reply(nlsk, rss_reply_cb, &args);
delete_json_obj();
if (ret == 0)
return 0;
err:
return nlctx->exit_code ?: 1;
}