xref: /openbmc/linux/tools/net/ynl/generated/netdev-user.c (revision 7234415b8f86c496fec2d62a48f136cde530ad95)
1d75fdfbcSJakub Kicinski // SPDX-License-Identifier: ((GPL-2.0 WITH Linux-syscall-note) OR BSD-3-Clause)
2d75fdfbcSJakub Kicinski /* Do not edit directly, auto-generated from: */
3d75fdfbcSJakub Kicinski /*	Documentation/netlink/specs/netdev.yaml */
4d75fdfbcSJakub Kicinski /* YNL-GEN user source */
5d75fdfbcSJakub Kicinski 
6d75fdfbcSJakub Kicinski #include <stdlib.h>
79b52fd4bSJakub Kicinski #include <string.h>
8d75fdfbcSJakub Kicinski #include "netdev-user.h"
9d75fdfbcSJakub Kicinski #include "ynl.h"
10d75fdfbcSJakub Kicinski #include <linux/netdev.h>
11d75fdfbcSJakub Kicinski 
12d75fdfbcSJakub Kicinski #include <libmnl/libmnl.h>
13d75fdfbcSJakub Kicinski #include <linux/genetlink.h>
14d75fdfbcSJakub Kicinski 
15d75fdfbcSJakub Kicinski /* Enums */
16d75fdfbcSJakub Kicinski static const char * const netdev_op_strmap[] = {
17d75fdfbcSJakub Kicinski 	[NETDEV_CMD_DEV_GET] = "dev-get",
18d75fdfbcSJakub Kicinski 	[NETDEV_CMD_DEV_ADD_NTF] = "dev-add-ntf",
19d75fdfbcSJakub Kicinski 	[NETDEV_CMD_DEV_DEL_NTF] = "dev-del-ntf",
20d75fdfbcSJakub Kicinski 	[NETDEV_CMD_DEV_CHANGE_NTF] = "dev-change-ntf",
21d75fdfbcSJakub Kicinski };
22d75fdfbcSJakub Kicinski 
23d75fdfbcSJakub Kicinski const char *netdev_op_str(int op)
24d75fdfbcSJakub Kicinski {
25d75fdfbcSJakub Kicinski 	if (op < 0 || op >= (int)MNL_ARRAY_SIZE(netdev_op_strmap))
26d75fdfbcSJakub Kicinski 		return NULL;
27d75fdfbcSJakub Kicinski 	return netdev_op_strmap[op];
28d75fdfbcSJakub Kicinski }
29d75fdfbcSJakub Kicinski 
30d75fdfbcSJakub Kicinski static const char * const netdev_xdp_act_strmap[] = {
31d75fdfbcSJakub Kicinski 	[0] = "basic",
32d75fdfbcSJakub Kicinski 	[1] = "redirect",
33d75fdfbcSJakub Kicinski 	[2] = "ndo-xmit",
34d75fdfbcSJakub Kicinski 	[3] = "xsk-zerocopy",
35d75fdfbcSJakub Kicinski 	[4] = "hw-offload",
36d75fdfbcSJakub Kicinski 	[5] = "rx-sg",
37d75fdfbcSJakub Kicinski 	[6] = "ndo-xmit-sg",
38d75fdfbcSJakub Kicinski };
39d75fdfbcSJakub Kicinski 
40d75fdfbcSJakub Kicinski const char *netdev_xdp_act_str(enum netdev_xdp_act value)
41d75fdfbcSJakub Kicinski {
42d75fdfbcSJakub Kicinski 	value = ffs(value) - 1;
43d75fdfbcSJakub Kicinski 	if (value < 0 || value >= (int)MNL_ARRAY_SIZE(netdev_xdp_act_strmap))
44d75fdfbcSJakub Kicinski 		return NULL;
45d75fdfbcSJakub Kicinski 	return netdev_xdp_act_strmap[value];
46d75fdfbcSJakub Kicinski }
47d75fdfbcSJakub Kicinski 
48d75fdfbcSJakub Kicinski /* Policies */
49d75fdfbcSJakub Kicinski struct ynl_policy_attr netdev_dev_policy[NETDEV_A_DEV_MAX + 1] = {
50d75fdfbcSJakub Kicinski 	[NETDEV_A_DEV_IFINDEX] = { .name = "ifindex", .type = YNL_PT_U32, },
51d75fdfbcSJakub Kicinski 	[NETDEV_A_DEV_PAD] = { .name = "pad", .type = YNL_PT_IGNORE, },
52d75fdfbcSJakub Kicinski 	[NETDEV_A_DEV_XDP_FEATURES] = { .name = "xdp-features", .type = YNL_PT_U64, },
53d75fdfbcSJakub Kicinski };
54d75fdfbcSJakub Kicinski 
55d75fdfbcSJakub Kicinski struct ynl_policy_nest netdev_dev_nest = {
56d75fdfbcSJakub Kicinski 	.max_attr = NETDEV_A_DEV_MAX,
57d75fdfbcSJakub Kicinski 	.table = netdev_dev_policy,
58d75fdfbcSJakub Kicinski };
59d75fdfbcSJakub Kicinski 
60d75fdfbcSJakub Kicinski /* Common nested types */
61d75fdfbcSJakub Kicinski /* ============== NETDEV_CMD_DEV_GET ============== */
62d75fdfbcSJakub Kicinski /* NETDEV_CMD_DEV_GET - do */
63d75fdfbcSJakub Kicinski void netdev_dev_get_req_free(struct netdev_dev_get_req *req)
64d75fdfbcSJakub Kicinski {
65d75fdfbcSJakub Kicinski 	free(req);
66d75fdfbcSJakub Kicinski }
67d75fdfbcSJakub Kicinski 
68d75fdfbcSJakub Kicinski void netdev_dev_get_rsp_free(struct netdev_dev_get_rsp *rsp)
69d75fdfbcSJakub Kicinski {
70d75fdfbcSJakub Kicinski 	free(rsp);
71d75fdfbcSJakub Kicinski }
72d75fdfbcSJakub Kicinski 
73d75fdfbcSJakub Kicinski int netdev_dev_get_rsp_parse(const struct nlmsghdr *nlh, void *data)
74d75fdfbcSJakub Kicinski {
75d75fdfbcSJakub Kicinski 	struct ynl_parse_arg *yarg = data;
76d75fdfbcSJakub Kicinski 	struct netdev_dev_get_rsp *dst;
77d75fdfbcSJakub Kicinski 	const struct nlattr *attr;
78d75fdfbcSJakub Kicinski 
79d75fdfbcSJakub Kicinski 	dst = yarg->data;
80d75fdfbcSJakub Kicinski 
81d75fdfbcSJakub Kicinski 	mnl_attr_for_each(attr, nlh, sizeof(struct genlmsghdr)) {
82*7234415bSJakub Kicinski 		unsigned int type = mnl_attr_get_type(attr);
83*7234415bSJakub Kicinski 
84*7234415bSJakub Kicinski 		if (type == NETDEV_A_DEV_IFINDEX) {
85d75fdfbcSJakub Kicinski 			if (ynl_attr_validate(yarg, attr))
86d75fdfbcSJakub Kicinski 				return MNL_CB_ERROR;
87d75fdfbcSJakub Kicinski 			dst->_present.ifindex = 1;
88d75fdfbcSJakub Kicinski 			dst->ifindex = mnl_attr_get_u32(attr);
89*7234415bSJakub Kicinski 		} else if (type == NETDEV_A_DEV_XDP_FEATURES) {
90d75fdfbcSJakub Kicinski 			if (ynl_attr_validate(yarg, attr))
91d75fdfbcSJakub Kicinski 				return MNL_CB_ERROR;
92d75fdfbcSJakub Kicinski 			dst->_present.xdp_features = 1;
93d75fdfbcSJakub Kicinski 			dst->xdp_features = mnl_attr_get_u64(attr);
94d75fdfbcSJakub Kicinski 		}
95d75fdfbcSJakub Kicinski 	}
96d75fdfbcSJakub Kicinski 
97d75fdfbcSJakub Kicinski 	return MNL_CB_OK;
98d75fdfbcSJakub Kicinski }
99d75fdfbcSJakub Kicinski 
100d75fdfbcSJakub Kicinski struct netdev_dev_get_rsp *
101d75fdfbcSJakub Kicinski netdev_dev_get(struct ynl_sock *ys, struct netdev_dev_get_req *req)
102d75fdfbcSJakub Kicinski {
103d75fdfbcSJakub Kicinski 	struct ynl_req_state yrs = { .yarg = { .ys = ys, }, };
104d75fdfbcSJakub Kicinski 	struct netdev_dev_get_rsp *rsp;
105d75fdfbcSJakub Kicinski 	struct nlmsghdr *nlh;
106d75fdfbcSJakub Kicinski 	int err;
107d75fdfbcSJakub Kicinski 
108d75fdfbcSJakub Kicinski 	nlh = ynl_gemsg_start_req(ys, ys->family_id, NETDEV_CMD_DEV_GET, 1);
109d75fdfbcSJakub Kicinski 	ys->req_policy = &netdev_dev_nest;
110d75fdfbcSJakub Kicinski 	yrs.yarg.rsp_policy = &netdev_dev_nest;
111d75fdfbcSJakub Kicinski 
112d75fdfbcSJakub Kicinski 	if (req->_present.ifindex)
113d75fdfbcSJakub Kicinski 		mnl_attr_put_u32(nlh, NETDEV_A_DEV_IFINDEX, req->ifindex);
114d75fdfbcSJakub Kicinski 
115d75fdfbcSJakub Kicinski 	rsp = calloc(1, sizeof(*rsp));
116d75fdfbcSJakub Kicinski 	yrs.yarg.data = rsp;
117d75fdfbcSJakub Kicinski 	yrs.cb = netdev_dev_get_rsp_parse;
118d75fdfbcSJakub Kicinski 	yrs.rsp_cmd = NETDEV_CMD_DEV_GET;
119d75fdfbcSJakub Kicinski 
120d75fdfbcSJakub Kicinski 	err = ynl_exec(ys, nlh, &yrs);
121d75fdfbcSJakub Kicinski 	if (err < 0)
122d75fdfbcSJakub Kicinski 		goto err_free;
123d75fdfbcSJakub Kicinski 
124d75fdfbcSJakub Kicinski 	return rsp;
125d75fdfbcSJakub Kicinski 
126d75fdfbcSJakub Kicinski err_free:
127d75fdfbcSJakub Kicinski 	netdev_dev_get_rsp_free(rsp);
128d75fdfbcSJakub Kicinski 	return NULL;
129d75fdfbcSJakub Kicinski }
130d75fdfbcSJakub Kicinski 
131d75fdfbcSJakub Kicinski /* NETDEV_CMD_DEV_GET - dump */
132d75fdfbcSJakub Kicinski void netdev_dev_get_list_free(struct netdev_dev_get_list *rsp)
133d75fdfbcSJakub Kicinski {
134d75fdfbcSJakub Kicinski 	struct netdev_dev_get_list *next = rsp;
135d75fdfbcSJakub Kicinski 
136d75fdfbcSJakub Kicinski 	while ((void *)next != YNL_LIST_END) {
137d75fdfbcSJakub Kicinski 		rsp = next;
138d75fdfbcSJakub Kicinski 		next = rsp->next;
139d75fdfbcSJakub Kicinski 
140d75fdfbcSJakub Kicinski 		free(rsp);
141d75fdfbcSJakub Kicinski 	}
142d75fdfbcSJakub Kicinski }
143d75fdfbcSJakub Kicinski 
144d75fdfbcSJakub Kicinski struct netdev_dev_get_list *netdev_dev_get_dump(struct ynl_sock *ys)
145d75fdfbcSJakub Kicinski {
146d75fdfbcSJakub Kicinski 	struct ynl_dump_state yds = {};
147d75fdfbcSJakub Kicinski 	struct nlmsghdr *nlh;
148d75fdfbcSJakub Kicinski 	int err;
149d75fdfbcSJakub Kicinski 
150d75fdfbcSJakub Kicinski 	yds.ys = ys;
151d75fdfbcSJakub Kicinski 	yds.alloc_sz = sizeof(struct netdev_dev_get_list);
152d75fdfbcSJakub Kicinski 	yds.cb = netdev_dev_get_rsp_parse;
153d75fdfbcSJakub Kicinski 	yds.rsp_cmd = NETDEV_CMD_DEV_GET;
154d75fdfbcSJakub Kicinski 	yds.rsp_policy = &netdev_dev_nest;
155d75fdfbcSJakub Kicinski 
156d75fdfbcSJakub Kicinski 	nlh = ynl_gemsg_start_dump(ys, ys->family_id, NETDEV_CMD_DEV_GET, 1);
157d75fdfbcSJakub Kicinski 
158d75fdfbcSJakub Kicinski 	err = ynl_exec_dump(ys, nlh, &yds);
159d75fdfbcSJakub Kicinski 	if (err < 0)
160d75fdfbcSJakub Kicinski 		goto free_list;
161d75fdfbcSJakub Kicinski 
162d75fdfbcSJakub Kicinski 	return yds.first;
163d75fdfbcSJakub Kicinski 
164d75fdfbcSJakub Kicinski free_list:
165d75fdfbcSJakub Kicinski 	netdev_dev_get_list_free(yds.first);
166d75fdfbcSJakub Kicinski 	return NULL;
167d75fdfbcSJakub Kicinski }
168d75fdfbcSJakub Kicinski 
169d75fdfbcSJakub Kicinski /* NETDEV_CMD_DEV_GET - notify */
170d75fdfbcSJakub Kicinski void netdev_dev_get_ntf_free(struct netdev_dev_get_ntf *rsp)
171d75fdfbcSJakub Kicinski {
172d75fdfbcSJakub Kicinski 	free(rsp);
173d75fdfbcSJakub Kicinski }
174d75fdfbcSJakub Kicinski 
175d75fdfbcSJakub Kicinski /* --------------- Common notification parsing --------------- */
176d75fdfbcSJakub Kicinski struct ynl_ntf_base_type *netdev_ntf_parse(struct ynl_sock *ys)
177d75fdfbcSJakub Kicinski {
178d75fdfbcSJakub Kicinski 	struct ynl_parse_arg yarg = { .ys = ys, };
179d75fdfbcSJakub Kicinski 	struct ynl_ntf_base_type *rsp;
180d75fdfbcSJakub Kicinski 	struct genlmsghdr *genlh;
181d75fdfbcSJakub Kicinski 	struct nlmsghdr *nlh;
182d75fdfbcSJakub Kicinski 	mnl_cb_t parse;
183d75fdfbcSJakub Kicinski 	int len, err;
184d75fdfbcSJakub Kicinski 
185d75fdfbcSJakub Kicinski 	len = mnl_socket_recvfrom(ys->sock, ys->rx_buf, MNL_SOCKET_BUFFER_SIZE);
186d75fdfbcSJakub Kicinski 	if (len < (ssize_t)(sizeof(*nlh) + sizeof(*genlh)))
187d75fdfbcSJakub Kicinski 		return NULL;
188d75fdfbcSJakub Kicinski 
189d75fdfbcSJakub Kicinski 	nlh = (struct nlmsghdr *)ys->rx_buf;
190d75fdfbcSJakub Kicinski 	genlh = mnl_nlmsg_get_payload(nlh);
191d75fdfbcSJakub Kicinski 
192d75fdfbcSJakub Kicinski 	switch (genlh->cmd) {
193d75fdfbcSJakub Kicinski 	case NETDEV_CMD_DEV_ADD_NTF:
194d75fdfbcSJakub Kicinski 	case NETDEV_CMD_DEV_DEL_NTF:
195d75fdfbcSJakub Kicinski 	case NETDEV_CMD_DEV_CHANGE_NTF:
196d75fdfbcSJakub Kicinski 		rsp = calloc(1, sizeof(struct netdev_dev_get_ntf));
197d75fdfbcSJakub Kicinski 		parse = netdev_dev_get_rsp_parse;
198d75fdfbcSJakub Kicinski 		yarg.rsp_policy = &netdev_dev_nest;
199d75fdfbcSJakub Kicinski 		rsp->free = (void *)netdev_dev_get_ntf_free;
200d75fdfbcSJakub Kicinski 		break;
201d75fdfbcSJakub Kicinski 	default:
202d75fdfbcSJakub Kicinski 		ynl_error_unknown_notification(ys, genlh->cmd);
203d75fdfbcSJakub Kicinski 		return NULL;
204d75fdfbcSJakub Kicinski 	}
205d75fdfbcSJakub Kicinski 
206d75fdfbcSJakub Kicinski 	yarg.data = rsp->data;
207d75fdfbcSJakub Kicinski 
208d75fdfbcSJakub Kicinski 	err = mnl_cb_run2(ys->rx_buf, len, 0, 0, parse, &yarg,
209d75fdfbcSJakub Kicinski 			 ynl_cb_array, NLMSG_MIN_TYPE);
210d75fdfbcSJakub Kicinski 	if (err < 0)
211d75fdfbcSJakub Kicinski 		goto err_free;
212d75fdfbcSJakub Kicinski 
213d75fdfbcSJakub Kicinski 	rsp->family = nlh->nlmsg_type;
214d75fdfbcSJakub Kicinski 	rsp->cmd = genlh->cmd;
215d75fdfbcSJakub Kicinski 	return rsp;
216d75fdfbcSJakub Kicinski 
217d75fdfbcSJakub Kicinski err_free:
218d75fdfbcSJakub Kicinski 	free(rsp);
219d75fdfbcSJakub Kicinski 	return NULL;
220d75fdfbcSJakub Kicinski }
221d75fdfbcSJakub Kicinski 
222d75fdfbcSJakub Kicinski static const struct ynl_ntf_info netdev_ntf_info[] =  {
223d75fdfbcSJakub Kicinski 	[NETDEV_CMD_DEV_ADD_NTF] =  {
224d75fdfbcSJakub Kicinski 		.alloc_sz	= sizeof(struct netdev_dev_get_ntf),
225d75fdfbcSJakub Kicinski 		.cb		= netdev_dev_get_rsp_parse,
226d75fdfbcSJakub Kicinski 		.policy		= &netdev_dev_nest,
227d75fdfbcSJakub Kicinski 		.free		= (void *)netdev_dev_get_ntf_free,
228d75fdfbcSJakub Kicinski 	},
229d75fdfbcSJakub Kicinski 	[NETDEV_CMD_DEV_DEL_NTF] =  {
230d75fdfbcSJakub Kicinski 		.alloc_sz	= sizeof(struct netdev_dev_get_ntf),
231d75fdfbcSJakub Kicinski 		.cb		= netdev_dev_get_rsp_parse,
232d75fdfbcSJakub Kicinski 		.policy		= &netdev_dev_nest,
233d75fdfbcSJakub Kicinski 		.free		= (void *)netdev_dev_get_ntf_free,
234d75fdfbcSJakub Kicinski 	},
235d75fdfbcSJakub Kicinski 	[NETDEV_CMD_DEV_CHANGE_NTF] =  {
236d75fdfbcSJakub Kicinski 		.alloc_sz	= sizeof(struct netdev_dev_get_ntf),
237d75fdfbcSJakub Kicinski 		.cb		= netdev_dev_get_rsp_parse,
238d75fdfbcSJakub Kicinski 		.policy		= &netdev_dev_nest,
239d75fdfbcSJakub Kicinski 		.free		= (void *)netdev_dev_get_ntf_free,
240d75fdfbcSJakub Kicinski 	},
241d75fdfbcSJakub Kicinski };
242d75fdfbcSJakub Kicinski 
243d75fdfbcSJakub Kicinski const struct ynl_family ynl_netdev_family =  {
244d75fdfbcSJakub Kicinski 	.name		= "netdev",
245d75fdfbcSJakub Kicinski 	.ntf_info	= netdev_ntf_info,
246d75fdfbcSJakub Kicinski 	.ntf_info_size	= MNL_ARRAY_SIZE(netdev_ntf_info),
247d75fdfbcSJakub Kicinski };
248