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