linux/tools/net/ynl/tests/netdev.c
Jakub Kicinski 285804d63f tools: ynl: convert netdev sample to selftest
Convert netdev.c to produce KTAP output with 3 tests:
- dev_dump: dump all netdev devices, skip if empty
- dev_get: query first device from dump by ifindex
- ntf_check: subscribe to "mgmt", create a veth via rt-link,
  verify netdev notification is received, then delete the veth

Remove stdin/scanf-based UI. Add rt-link dependency for the veth
notification test.

  TAP version 13
  1..3
  # Starting 3 tests from 1 test cases.
  #  RUN           netdev.dump ...
  #       lo[1]	xdp-features (0): xdp-rx-metadata-features (0): xsk-fea...
  #     sit0[2]	xdp-features (0): xdp-rx-metadata-features (0): xsk-fea...
  #            OK  netdev.dump
  ok 1 netdev.dump
  #  RUN           netdev.get ...
  #       lo[1]	xdp-features (0): xdp-rx-metadata-features (0): xsk-fea...
  #            OK  netdev.get
  ok 2 netdev.get
  #  RUN           netdev.ntf_check ...
  #    veth0[7]	xdp-features (0): xdp-rx-metadata-features (7): timesta...
  #            OK  netdev.ntf_check
  ok 3 netdev.ntf_check
  # PASSED: 3 / 3 tests passed.
  # Totals: pass:3 fail:0 xfail:0 xpass:0 skip:0 error:0

Reviewed-by: Donald Hunter <donald.hunter@gmail.com>
Tested-by: Donald Hunter <donald.hunter@gmail.com>
Link: https://patch.msgid.link/20260307033630.1396085-3-kuba@kernel.org
Signed-off-by: Jakub Kicinski <kuba@kernel.org>
2026-03-09 17:02:26 -07:00

232 lines
4.8 KiB
C

// 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