| // SPDX-License-Identifier: GPL-2.0 |
| #include <stdio.h> |
| #include <string.h> |
| |
| #include <ynl.h> |
| |
| #include <net/if.h> |
| |
| #include <kselftest_harness.h> |
| |
| #include "netdev-user.h" |
| #include "rt-link-user.h" |
| |
| static void netdev_print_device(struct __test_metadata *_metadata, |
| struct netdev_dev_get_rsp *d, unsigned int op) |
| { |
| char ifname[IF_NAMESIZE]; |
| const char *name; |
| |
| EXPECT_TRUE((bool)d->_present.ifindex); |
| if (!d->_present.ifindex) |
| return; |
| |
| name = if_indextoname(d->ifindex, ifname); |
| EXPECT_TRUE((bool)name); |
| if (name) |
| ksft_print_msg("%8s[%d]\t", name, d->ifindex); |
| else |
| ksft_print_msg("[%d]\t", d->ifindex); |
| |
| EXPECT_TRUE((bool)d->_present.xdp_features); |
| if (!d->_present.xdp_features) |
| return; |
| |
| printf("xdp-features (%llx):", d->xdp_features); |
| for (int i = 0; d->xdp_features >= 1U << i; i++) { |
| if (d->xdp_features & (1U << i)) |
| printf(" %s", netdev_xdp_act_str(1 << i)); |
| } |
| |
| printf(" xdp-rx-metadata-features (%llx):", |
| d->xdp_rx_metadata_features); |
| for (int i = 0; d->xdp_rx_metadata_features >= 1U << i; i++) { |
| if (d->xdp_rx_metadata_features & (1U << i)) |
| printf(" %s", |
| netdev_xdp_rx_metadata_str(1 << i)); |
| } |
| |
| printf(" xsk-features (%llx):", d->xsk_features); |
| for (int i = 0; d->xsk_features >= 1U << i; i++) { |
| if (d->xsk_features & (1U << i)) |
| printf(" %s", netdev_xsk_flags_str(1 << i)); |
| } |
| |
| printf(" xdp-zc-max-segs=%u", d->xdp_zc_max_segs); |
| |
| name = netdev_op_str(op); |
| if (name) |
| printf(" (ntf: %s)", name); |
| printf("\n"); |
| } |
| |
| static int veth_create(struct ynl_sock *ys_link) |
| { |
| struct rt_link_getlink_ntf *ntf_gl; |
| struct rt_link_newlink_req *req; |
| struct ynl_ntf_base_type *ntf; |
| int ret; |
| |
| req = rt_link_newlink_req_alloc(); |
| if (!req) |
| return -1; |
| |
| rt_link_newlink_req_set_nlflags(req, NLM_F_CREATE | NLM_F_ECHO); |
| rt_link_newlink_req_set_linkinfo_kind(req, "veth"); |
| |
| ret = rt_link_newlink(ys_link, req); |
| rt_link_newlink_req_free(req); |
| if (ret) |
| return -1; |
| |
| if (!ynl_has_ntf(ys_link)) |
| return 0; |
| |
| ntf = ynl_ntf_dequeue(ys_link); |
| if (!ntf || ntf->cmd != RTM_NEWLINK) { |
| ynl_ntf_free(ntf); |
| return 0; |
| } |
| ntf_gl = (void *)ntf; |
| ret = ntf_gl->obj._hdr.ifi_index; |
| ynl_ntf_free(ntf); |
| |
| return ret; |
| } |
| |
| static void veth_delete(struct __test_metadata *_metadata, |
| struct ynl_sock *ys_link, int ifindex) |
| { |
| struct rt_link_dellink_req *req; |
| |
| req = rt_link_dellink_req_alloc(); |
| ASSERT_NE(NULL, req); |
| |
| req->_hdr.ifi_index = ifindex; |
| EXPECT_EQ(0, rt_link_dellink(ys_link, req)); |
| rt_link_dellink_req_free(req); |
| } |
| |
| FIXTURE(netdev) |
| { |
| struct ynl_sock *ys; |
| struct ynl_sock *ys_link; |
| }; |
| |
| FIXTURE_SETUP(netdev) |
| { |
| struct ynl_error yerr; |
| |
| self->ys = ynl_sock_create(&ynl_netdev_family, &yerr); |
| ASSERT_NE(NULL, self->ys) { |
| TH_LOG("Failed to create YNL netdev socket: %s", yerr.msg); |
| } |
| } |
| |
| FIXTURE_TEARDOWN(netdev) |
| { |
| if (self->ys_link) |
| ynl_sock_destroy(self->ys_link); |
| ynl_sock_destroy(self->ys); |
| } |
| |
| TEST_F(netdev, dump) |
| { |
| struct netdev_dev_get_list *devs; |
| |
| devs = netdev_dev_get_dump(self->ys); |
| ASSERT_NE(NULL, devs) { |
| TH_LOG("dump failed: %s", self->ys->err.msg); |
| } |
| |
| if (ynl_dump_empty(devs)) { |
| netdev_dev_get_list_free(devs); |
| SKIP(return, "no entries in dump"); |
| } |
| |
| ynl_dump_foreach(devs, d) |
| netdev_print_device(_metadata, d, 0); |
| |
| netdev_dev_get_list_free(devs); |
| } |
| |
| TEST_F(netdev, get) |
| { |
| struct netdev_dev_get_list *devs; |
| struct netdev_dev_get_req *req; |
| struct netdev_dev_get_rsp *dev; |
| int ifindex = 0; |
| |
| devs = netdev_dev_get_dump(self->ys); |
| ASSERT_NE(NULL, devs) { |
| TH_LOG("dump failed: %s", self->ys->err.msg); |
| } |
| |
| ynl_dump_foreach(devs, d) { |
| if (d->_present.ifindex) { |
| ifindex = d->ifindex; |
| break; |
| } |
| } |
| netdev_dev_get_list_free(devs); |
| |
| if (!ifindex) |
| SKIP(return, "no device to query"); |
| |
| req = netdev_dev_get_req_alloc(); |
| ASSERT_NE(NULL, req); |
| netdev_dev_get_req_set_ifindex(req, ifindex); |
| |
| dev = netdev_dev_get(self->ys, req); |
| netdev_dev_get_req_free(req); |
| ASSERT_NE(NULL, dev) { |
| TH_LOG("dev_get failed: %s", self->ys->err.msg); |
| } |
| |
| netdev_print_device(_metadata, dev, 0); |
| netdev_dev_get_rsp_free(dev); |
| } |
| |
| TEST_F(netdev, ntf_check) |
| { |
| struct ynl_ntf_base_type *ntf; |
| int veth_ifindex; |
| bool received; |
| int ret; |
| |
| ret = ynl_subscribe(self->ys, "mgmt"); |
| ASSERT_EQ(0, ret) { |
| TH_LOG("subscribe failed: %s", self->ys->err.msg); |
| } |
| |
| self->ys_link = ynl_sock_create(&ynl_rt_link_family, NULL); |
| ASSERT_NE(NULL, self->ys_link) |
| TH_LOG("failed to create rt-link socket"); |
| |
| veth_ifindex = veth_create(self->ys_link); |
| ASSERT_GT(veth_ifindex, 0) |
| TH_LOG("failed to create veth"); |
| |
| ynl_ntf_check(self->ys); |
| |
| ntf = ynl_ntf_dequeue(self->ys); |
| received = ntf; |
| if (ntf) { |
| netdev_print_device(_metadata, |
| (struct netdev_dev_get_rsp *)&ntf->data, |
| ntf->cmd); |
| ynl_ntf_free(ntf); |
| } |
| |
| /* Drain any remaining notifications */ |
| while ((ntf = ynl_ntf_dequeue(self->ys))) |
| ynl_ntf_free(ntf); |
| |
| veth_delete(_metadata, self->ys_link, veth_ifindex); |
| |
| ASSERT_TRUE(received) |
| TH_LOG("no notification received"); |
| } |
| |
| TEST_HARNESS_MAIN |