xref: /aosp_15_r20/external/ethtool/netlink/rss.c (revision 1b481fc3bb1b45d4cf28d1ec12969dc1055f555d)
1*1b481fc3SMaciej Żenczykowski /*
2*1b481fc3SMaciej Żenczykowski  * rss.c - netlink implementation of RSS context commands
3*1b481fc3SMaciej Żenczykowski  *
4*1b481fc3SMaciej Żenczykowski  * Implementation of "ethtool -x <dev>"
5*1b481fc3SMaciej Żenczykowski  */
6*1b481fc3SMaciej Żenczykowski 
7*1b481fc3SMaciej Żenczykowski #include <errno.h>
8*1b481fc3SMaciej Żenczykowski #include <string.h>
9*1b481fc3SMaciej Żenczykowski #include <stdio.h>
10*1b481fc3SMaciej Żenczykowski 
11*1b481fc3SMaciej Żenczykowski #include "../internal.h"
12*1b481fc3SMaciej Żenczykowski #include "../common.h"
13*1b481fc3SMaciej Żenczykowski #include "netlink.h"
14*1b481fc3SMaciej Żenczykowski #include "strset.h"
15*1b481fc3SMaciej Żenczykowski #include "parser.h"
16*1b481fc3SMaciej Żenczykowski 
17*1b481fc3SMaciej Żenczykowski struct cb_args {
18*1b481fc3SMaciej Żenczykowski 	struct nl_context	*nlctx;
19*1b481fc3SMaciej Żenczykowski 	u32			num_rings;
20*1b481fc3SMaciej Żenczykowski };
21*1b481fc3SMaciej Żenczykowski 
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)22*1b481fc3SMaciej Żenczykowski void dump_json_rss_info(struct cmd_context *ctx, u32 *indir_table,
23*1b481fc3SMaciej Żenczykowski 			u32 indir_size, u8 *hkey, u32 hkey_size,
24*1b481fc3SMaciej Żenczykowski 			const struct stringset *hash_funcs, u8 hfunc)
25*1b481fc3SMaciej Żenczykowski {
26*1b481fc3SMaciej Żenczykowski 	unsigned int i;
27*1b481fc3SMaciej Żenczykowski 
28*1b481fc3SMaciej Żenczykowski 	open_json_object(NULL);
29*1b481fc3SMaciej Żenczykowski 	print_string(PRINT_JSON, "ifname", NULL, ctx->devname);
30*1b481fc3SMaciej Żenczykowski 	if (indir_size) {
31*1b481fc3SMaciej Żenczykowski 		open_json_array("rss-indirection-table", NULL);
32*1b481fc3SMaciej Żenczykowski 		for (i = 0; i < indir_size; i++)
33*1b481fc3SMaciej Żenczykowski 			print_uint(PRINT_JSON, NULL, NULL, indir_table[i]);
34*1b481fc3SMaciej Żenczykowski 		close_json_array("\n");
35*1b481fc3SMaciej Żenczykowski 	}
36*1b481fc3SMaciej Żenczykowski 
37*1b481fc3SMaciej Żenczykowski 	if (hkey_size) {
38*1b481fc3SMaciej Żenczykowski 		open_json_array("rss-hash-key", NULL);
39*1b481fc3SMaciej Żenczykowski 		for (i = 0; i < hkey_size; i++)
40*1b481fc3SMaciej Żenczykowski 			print_uint(PRINT_JSON, NULL, NULL, (u8)hkey[i]);
41*1b481fc3SMaciej Żenczykowski 		close_json_array("\n");
42*1b481fc3SMaciej Żenczykowski 	}
43*1b481fc3SMaciej Żenczykowski 
44*1b481fc3SMaciej Żenczykowski 	if (hfunc) {
45*1b481fc3SMaciej Żenczykowski 		for (i = 0; i < get_count(hash_funcs); i++) {
46*1b481fc3SMaciej Żenczykowski 			if (hfunc & (1 << i)) {
47*1b481fc3SMaciej Żenczykowski 				print_string(PRINT_JSON, "rss-hash-function",
48*1b481fc3SMaciej Żenczykowski 					     NULL, get_string(hash_funcs, i));
49*1b481fc3SMaciej Żenczykowski 				break;
50*1b481fc3SMaciej Żenczykowski 			}
51*1b481fc3SMaciej Żenczykowski 		}
52*1b481fc3SMaciej Żenczykowski 	}
53*1b481fc3SMaciej Żenczykowski 
54*1b481fc3SMaciej Żenczykowski 	close_json_object();
55*1b481fc3SMaciej Żenczykowski }
56*1b481fc3SMaciej Żenczykowski 
get_channels_cb(const struct nlmsghdr * nlhdr,void * data)57*1b481fc3SMaciej Żenczykowski int get_channels_cb(const struct nlmsghdr *nlhdr, void *data)
58*1b481fc3SMaciej Żenczykowski {
59*1b481fc3SMaciej Żenczykowski 	const struct nlattr *tb[ETHTOOL_A_CHANNELS_MAX + 1] = {};
60*1b481fc3SMaciej Żenczykowski 	DECLARE_ATTR_TB_INFO(tb);
61*1b481fc3SMaciej Żenczykowski 	struct cb_args *args = data;
62*1b481fc3SMaciej Żenczykowski 	struct nl_context *nlctx = args->nlctx;
63*1b481fc3SMaciej Żenczykowski 	bool silent;
64*1b481fc3SMaciej Żenczykowski 	int err_ret;
65*1b481fc3SMaciej Żenczykowski 	int ret;
66*1b481fc3SMaciej Żenczykowski 
67*1b481fc3SMaciej Żenczykowski 	silent = nlctx->is_dump || nlctx->is_monitor;
68*1b481fc3SMaciej Żenczykowski 	err_ret = silent ? MNL_CB_OK : MNL_CB_ERROR;
69*1b481fc3SMaciej Żenczykowski 	ret = mnl_attr_parse(nlhdr, GENL_HDRLEN, attr_cb, &tb_info);
70*1b481fc3SMaciej Żenczykowski 	if (ret < 0)
71*1b481fc3SMaciej Żenczykowski 		return err_ret;
72*1b481fc3SMaciej Żenczykowski 	nlctx->devname = get_dev_name(tb[ETHTOOL_A_CHANNELS_HEADER]);
73*1b481fc3SMaciej Żenczykowski 	if (!dev_ok(nlctx))
74*1b481fc3SMaciej Żenczykowski 		return err_ret;
75*1b481fc3SMaciej Żenczykowski 	if (tb[ETHTOOL_A_CHANNELS_COMBINED_COUNT])
76*1b481fc3SMaciej Żenczykowski 		args->num_rings = mnl_attr_get_u32(tb[ETHTOOL_A_CHANNELS_COMBINED_COUNT]);
77*1b481fc3SMaciej Żenczykowski 	if (tb[ETHTOOL_A_CHANNELS_RX_COUNT])
78*1b481fc3SMaciej Żenczykowski 		args->num_rings += mnl_attr_get_u32(tb[ETHTOOL_A_CHANNELS_RX_COUNT]);
79*1b481fc3SMaciej Żenczykowski 	return MNL_CB_OK;
80*1b481fc3SMaciej Żenczykowski }
81*1b481fc3SMaciej Żenczykowski 
rss_reply_cb(const struct nlmsghdr * nlhdr,void * data)82*1b481fc3SMaciej Żenczykowski int rss_reply_cb(const struct nlmsghdr *nlhdr, void *data)
83*1b481fc3SMaciej Żenczykowski {
84*1b481fc3SMaciej Żenczykowski 	const struct nlattr *tb[ETHTOOL_A_RSS_MAX + 1] = {};
85*1b481fc3SMaciej Żenczykowski 	unsigned int indir_bytes = 0, hkey_bytes = 0;
86*1b481fc3SMaciej Żenczykowski 	DECLARE_ATTR_TB_INFO(tb);
87*1b481fc3SMaciej Żenczykowski 	struct cb_args *args = data;
88*1b481fc3SMaciej Żenczykowski 	struct nl_context *nlctx = args->nlctx;
89*1b481fc3SMaciej Żenczykowski 	const struct stringset *hash_funcs;
90*1b481fc3SMaciej Żenczykowski 	u32 rss_hfunc = 0, indir_size;
91*1b481fc3SMaciej Żenczykowski 	u32 *indir_table = NULL;
92*1b481fc3SMaciej Żenczykowski 	u8 *hkey = NULL;
93*1b481fc3SMaciej Żenczykowski 	bool silent;
94*1b481fc3SMaciej Żenczykowski 	int err_ret;
95*1b481fc3SMaciej Żenczykowski 	int ret;
96*1b481fc3SMaciej Żenczykowski 
97*1b481fc3SMaciej Żenczykowski 	silent = nlctx->is_dump || nlctx->is_monitor;
98*1b481fc3SMaciej Żenczykowski 	err_ret = silent ? MNL_CB_OK : MNL_CB_ERROR;
99*1b481fc3SMaciej Żenczykowski 	ret = mnl_attr_parse(nlhdr, GENL_HDRLEN, attr_cb, &tb_info);
100*1b481fc3SMaciej Żenczykowski 	if (ret < 0)
101*1b481fc3SMaciej Żenczykowski 		return err_ret;
102*1b481fc3SMaciej Żenczykowski 	nlctx->devname = get_dev_name(tb[ETHTOOL_A_RSS_HEADER]);
103*1b481fc3SMaciej Żenczykowski 	if (!dev_ok(nlctx))
104*1b481fc3SMaciej Żenczykowski 		return err_ret;
105*1b481fc3SMaciej Żenczykowski 
106*1b481fc3SMaciej Żenczykowski 	show_cr();
107*1b481fc3SMaciej Żenczykowski 
108*1b481fc3SMaciej Żenczykowski 	if (tb[ETHTOOL_A_RSS_HFUNC])
109*1b481fc3SMaciej Żenczykowski 		rss_hfunc = mnl_attr_get_u32(tb[ETHTOOL_A_RSS_HFUNC]);
110*1b481fc3SMaciej Żenczykowski 
111*1b481fc3SMaciej Żenczykowski 	if (tb[ETHTOOL_A_RSS_INDIR]) {
112*1b481fc3SMaciej Żenczykowski 		indir_bytes = mnl_attr_get_payload_len(tb[ETHTOOL_A_RSS_INDIR]);
113*1b481fc3SMaciej Żenczykowski 		indir_table = mnl_attr_get_payload(tb[ETHTOOL_A_RSS_INDIR]);
114*1b481fc3SMaciej Żenczykowski 	}
115*1b481fc3SMaciej Żenczykowski 
116*1b481fc3SMaciej Żenczykowski 	if (tb[ETHTOOL_A_RSS_HKEY]) {
117*1b481fc3SMaciej Żenczykowski 		hkey_bytes = mnl_attr_get_payload_len(tb[ETHTOOL_A_RSS_HKEY]);
118*1b481fc3SMaciej Żenczykowski 		hkey = mnl_attr_get_payload(tb[ETHTOOL_A_RSS_HKEY]);
119*1b481fc3SMaciej Żenczykowski 	}
120*1b481fc3SMaciej Żenczykowski 
121*1b481fc3SMaciej Żenczykowski 	/* Fetch RSS hash functions and their status and print */
122*1b481fc3SMaciej Żenczykowski 	if (!nlctx->is_monitor) {
123*1b481fc3SMaciej Żenczykowski 		ret = netlink_init_ethnl2_socket(nlctx);
124*1b481fc3SMaciej Żenczykowski 		if (ret < 0)
125*1b481fc3SMaciej Żenczykowski 			return MNL_CB_ERROR;
126*1b481fc3SMaciej Żenczykowski 	}
127*1b481fc3SMaciej Żenczykowski 	hash_funcs = global_stringset(ETH_SS_RSS_HASH_FUNCS,
128*1b481fc3SMaciej Żenczykowski 				      nlctx->ethnl2_socket);
129*1b481fc3SMaciej Żenczykowski 
130*1b481fc3SMaciej Żenczykowski 	ret = mnl_attr_parse(nlhdr, GENL_HDRLEN, attr_cb, &tb_info);
131*1b481fc3SMaciej Żenczykowski 	if (ret < 0)
132*1b481fc3SMaciej Żenczykowski 		return silent ? MNL_CB_OK : MNL_CB_ERROR;
133*1b481fc3SMaciej Żenczykowski 
134*1b481fc3SMaciej Żenczykowski 	nlctx->devname = get_dev_name(tb[ETHTOOL_A_RSS_HEADER]);
135*1b481fc3SMaciej Żenczykowski 	if (!dev_ok(nlctx))
136*1b481fc3SMaciej Żenczykowski 		return MNL_CB_OK;
137*1b481fc3SMaciej Żenczykowski 
138*1b481fc3SMaciej Żenczykowski 	/* Fetch ring count info into args->num_rings */
139*1b481fc3SMaciej Żenczykowski 	ret = nlsock_prep_get_request(nlctx->ethnl2_socket,
140*1b481fc3SMaciej Żenczykowski 				      ETHTOOL_MSG_CHANNELS_GET,
141*1b481fc3SMaciej Żenczykowski 				      ETHTOOL_A_CHANNELS_HEADER, 0);
142*1b481fc3SMaciej Żenczykowski 	if (ret < 0)
143*1b481fc3SMaciej Żenczykowski 		return MNL_CB_ERROR;
144*1b481fc3SMaciej Żenczykowski 
145*1b481fc3SMaciej Żenczykowski 	ret = nlsock_sendmsg(nlctx->ethnl2_socket, NULL);
146*1b481fc3SMaciej Żenczykowski 	if (ret < 0)
147*1b481fc3SMaciej Żenczykowski 		return MNL_CB_ERROR;
148*1b481fc3SMaciej Żenczykowski 
149*1b481fc3SMaciej Żenczykowski 	ret = nlsock_process_reply(nlctx->ethnl2_socket, get_channels_cb, args);
150*1b481fc3SMaciej Żenczykowski 	if (ret < 0)
151*1b481fc3SMaciej Żenczykowski 		return MNL_CB_ERROR;
152*1b481fc3SMaciej Żenczykowski 
153*1b481fc3SMaciej Żenczykowski 	indir_size = indir_bytes / sizeof(u32);
154*1b481fc3SMaciej Żenczykowski 	if (is_json_context()) {
155*1b481fc3SMaciej Żenczykowski 		dump_json_rss_info(nlctx->ctx, (u32 *)indir_table, indir_size,
156*1b481fc3SMaciej Żenczykowski 				   hkey, hkey_bytes, hash_funcs, rss_hfunc);
157*1b481fc3SMaciej Żenczykowski 	} else {
158*1b481fc3SMaciej Żenczykowski 		print_indir_table(nlctx->ctx, args->num_rings,
159*1b481fc3SMaciej Żenczykowski 				  indir_size, (u32 *)indir_table);
160*1b481fc3SMaciej Żenczykowski 		print_rss_hkey(hkey, hkey_bytes);
161*1b481fc3SMaciej Żenczykowski 		printf("RSS hash function:\n");
162*1b481fc3SMaciej Żenczykowski 		if (!rss_hfunc) {
163*1b481fc3SMaciej Żenczykowski 			printf("    Operation not supported\n");
164*1b481fc3SMaciej Żenczykowski 			return 0;
165*1b481fc3SMaciej Żenczykowski 		}
166*1b481fc3SMaciej Żenczykowski 		for (unsigned int i = 0; i < get_count(hash_funcs); i++) {
167*1b481fc3SMaciej Żenczykowski 			printf("    %s: %s\n", get_string(hash_funcs, i),
168*1b481fc3SMaciej Żenczykowski 			       (rss_hfunc & (1 << i)) ? "on" : "off");
169*1b481fc3SMaciej Żenczykowski 		}
170*1b481fc3SMaciej Żenczykowski 	}
171*1b481fc3SMaciej Żenczykowski 
172*1b481fc3SMaciej Żenczykowski 	return MNL_CB_OK;
173*1b481fc3SMaciej Żenczykowski }
174*1b481fc3SMaciej Żenczykowski 
175*1b481fc3SMaciej Żenczykowski /* RSS_GET */
176*1b481fc3SMaciej Żenczykowski static const struct param_parser grss_params[] = {
177*1b481fc3SMaciej Żenczykowski 	{
178*1b481fc3SMaciej Żenczykowski 		.arg		= "context",
179*1b481fc3SMaciej Żenczykowski 		.type		= ETHTOOL_A_RSS_CONTEXT,
180*1b481fc3SMaciej Żenczykowski 		.handler	= nl_parse_direct_u32,
181*1b481fc3SMaciej Żenczykowski 		.min_argc	= 1,
182*1b481fc3SMaciej Żenczykowski 	},
183*1b481fc3SMaciej Żenczykowski 	{}
184*1b481fc3SMaciej Żenczykowski };
185*1b481fc3SMaciej Żenczykowski 
nl_grss(struct cmd_context * ctx)186*1b481fc3SMaciej Żenczykowski int nl_grss(struct cmd_context *ctx)
187*1b481fc3SMaciej Żenczykowski {
188*1b481fc3SMaciej Żenczykowski 	struct nl_context *nlctx = ctx->nlctx;
189*1b481fc3SMaciej Żenczykowski 	struct nl_socket *nlsk = nlctx->ethnl_socket;
190*1b481fc3SMaciej Żenczykowski 	struct nl_msg_buff *msgbuff;
191*1b481fc3SMaciej Żenczykowski 	struct cb_args args = {};
192*1b481fc3SMaciej Żenczykowski 	int ret;
193*1b481fc3SMaciej Żenczykowski 
194*1b481fc3SMaciej Żenczykowski 	nlctx->cmd = "-x";
195*1b481fc3SMaciej Żenczykowski 	nlctx->argp = ctx->argp;
196*1b481fc3SMaciej Żenczykowski 	nlctx->argc = ctx->argc;
197*1b481fc3SMaciej Żenczykowski 	nlctx->devname = ctx->devname;
198*1b481fc3SMaciej Żenczykowski 	nlsk = nlctx->ethnl_socket;
199*1b481fc3SMaciej Żenczykowski 	msgbuff = &nlsk->msgbuff;
200*1b481fc3SMaciej Żenczykowski 
201*1b481fc3SMaciej Żenczykowski 	if (netlink_cmd_check(ctx, ETHTOOL_MSG_RSS_GET, true))
202*1b481fc3SMaciej Żenczykowski 		return -EOPNOTSUPP;
203*1b481fc3SMaciej Żenczykowski 
204*1b481fc3SMaciej Żenczykowski 	ret = msg_init(nlctx, msgbuff, ETHTOOL_MSG_RSS_GET,
205*1b481fc3SMaciej Żenczykowski 		       NLM_F_REQUEST | NLM_F_ACK);
206*1b481fc3SMaciej Żenczykowski 	if (ret < 0)
207*1b481fc3SMaciej Żenczykowski 		return 1;
208*1b481fc3SMaciej Żenczykowski 
209*1b481fc3SMaciej Żenczykowski 	if (ethnla_fill_header(msgbuff, ETHTOOL_A_RSS_HEADER,
210*1b481fc3SMaciej Żenczykowski 			       ctx->devname, 0))
211*1b481fc3SMaciej Żenczykowski 		return -EMSGSIZE;
212*1b481fc3SMaciej Żenczykowski 
213*1b481fc3SMaciej Żenczykowski 	ret = nl_parser(nlctx, grss_params, NULL, PARSER_GROUP_NONE, NULL);
214*1b481fc3SMaciej Żenczykowski 	if (ret < 0)
215*1b481fc3SMaciej Żenczykowski 		goto err;
216*1b481fc3SMaciej Żenczykowski 
217*1b481fc3SMaciej Żenczykowski 	ret = nlsock_sendmsg(nlsk, NULL);
218*1b481fc3SMaciej Żenczykowski 	if (ret < 0)
219*1b481fc3SMaciej Żenczykowski 		goto err;
220*1b481fc3SMaciej Żenczykowski 
221*1b481fc3SMaciej Żenczykowski 	args.nlctx = nlctx;
222*1b481fc3SMaciej Żenczykowski 	new_json_obj(ctx->json);
223*1b481fc3SMaciej Żenczykowski 	ret = nlsock_process_reply(nlsk, rss_reply_cb, &args);
224*1b481fc3SMaciej Żenczykowski 	delete_json_obj();
225*1b481fc3SMaciej Żenczykowski 
226*1b481fc3SMaciej Żenczykowski 	if (ret == 0)
227*1b481fc3SMaciej Żenczykowski 		return 0;
228*1b481fc3SMaciej Żenczykowski err:
229*1b481fc3SMaciej Żenczykowski 	return nlctx->exit_code ?: 1;
230*1b481fc3SMaciej Żenczykowski }
231