| 12
 3
 4
 5
 6
 7
 8
 9
 10
 11
 12
 13
 14
 15
 16
 17
 18
 19
 20
 21
 22
 23
 24
 25
 26
 27
 28
 29
 30
 31
 32
 33
 34
 35
 36
 37
 38
 39
 40
 41
 42
 43
 44
 45
 46
 47
 48
 49
 50
 51
 52
 53
 54
 55
 56
 57
 58
 59
 60
 61
 62
 63
 64
 65
 66
 67
 68
 69
 70
 71
 72
 73
 74
 75
 76
 77
 78
 79
 80
 81
 82
 83
 84
 85
 86
 87
 88
 89
 90
 91
 92
 93
 94
 95
 96
 97
 98
 99
 100
 101
 102
 103
 104
 105
 106
 107
 108
 109
 110
 111
 112
 113
 114
 115
 116
 117
 118
 119
 120
 121
 122
 123
 124
 125
 126
 127
 128
 129
 130
 131
 132
 133
 134
 135
 136
 137
 
 | /*
 * Copyright (c) 2017 JingPiao Chen <chenjingpiao@gmail.com>
 * Copyright (c) 2017-2021 The strace developers.
 * All rights reserved.
 *
 * SPDX-License-Identifier: GPL-2.0-or-later
 */
#include "tests.h"
#include <math.h>
#include <stdio.h>
#include <unistd.h>
#include "test_nlattr.h"
#include <linux/if_bridge.h>
#include <linux/rtnetlink.h>
const unsigned int hdrlen = sizeof(struct br_port_msg);
static void
init_br_port_msg(struct nlmsghdr *const nlh, const unsigned int msg_len)
{
	SET_STRUCT(struct nlmsghdr, nlh,
		.nlmsg_len = msg_len,
		.nlmsg_type = RTM_GETMDB,
		.nlmsg_flags = NLM_F_DUMP
	);
	struct br_port_msg *const msg = NLMSG_DATA(nlh);
	SET_STRUCT(struct br_port_msg, msg,
		.family = AF_UNIX,
		.ifindex = ifindex_lo()
	);
	struct nlattr *nla = NLMSG_ATTR(nlh, sizeof(*msg));
	SET_STRUCT(struct nlattr, nla,
		.nla_len = msg_len - NLMSG_SPACE(hdrlen),
		.nla_type = MDBA_ROUTER
	);
}
static void
print_br_port_msg(const unsigned int msg_len)
{
	printf("{nlmsg_len=%u, nlmsg_type=RTM_GETMDB, nlmsg_flags=NLM_F_DUMP"
	       ", nlmsg_seq=0, nlmsg_pid=0}, {family=AF_UNIX"
	       ", ifindex=" IFINDEX_LO_STR "}"
	       ", [{nla_len=%u, nla_type=MDBA_ROUTER}",
	       msg_len, msg_len - NLMSG_SPACE(hdrlen));
}
int
main(void)
{
	skip_if_unavailable("/proc/self/fd/");
	const uint32_t ifindex = ifindex_lo();
	const uint8_t type = MDB_RTR_TYPE_DISABLED;
	static const struct nlattr nla = {
		.nla_len = NLA_HDRLEN + sizeof(type),
		.nla_type = MDBA_ROUTER_PATTR_TYPE
	};
	char buf[NLMSG_ALIGN(ifindex) + NLA_HDRLEN + sizeof(type)];
	const int fd = create_nl_socket(NETLINK_ROUTE);
	void *nlh0 = midtail_alloc(NLMSG_SPACE(hdrlen),
				   NLA_HDRLEN + sizeof(buf));
	static char pattern[4096];
	fill_memory_ex(pattern, sizeof(pattern), 'a', 'z' - 'a' + 1);
	TEST_NESTED_NLATTR_OBJECT(fd, nlh0, hdrlen,
				  init_br_port_msg, print_br_port_msg,
				  MDBA_ROUTER_PORT, pattern, ifindex,
				  printf(IFINDEX_LO_STR));
	memcpy(buf, &ifindex, sizeof(ifindex));
	memcpy(buf + NLMSG_ALIGN(ifindex), &nla, sizeof(nla));
	memcpy(buf + NLMSG_ALIGN(ifindex) + NLA_HDRLEN, &type, sizeof(type));
	TEST_NLATTR(fd, nlh0 - NLA_HDRLEN, hdrlen + NLA_HDRLEN,
		    init_br_port_msg, print_br_port_msg,
		    MDBA_ROUTER_PORT, sizeof(buf), buf, sizeof(buf),
		    printf(IFINDEX_LO_STR
			   ", [{nla_len=%u, nla_type=MDBA_ROUTER_PATTR_TYPE}"
			   ", MDB_RTR_TYPE_DISABLED]]",
			   nla.nla_len));
	/* timers */
	static const struct {
		uint32_t val;
		const char *str;
	} pattrs[] = {
		{ ARG_STR(MDBA_ROUTER_PATTR_TIMER) },
		{ ARG_STR(MDBA_ROUTER_PATTR_INET_TIMER) },
		{ ARG_STR(MDBA_ROUTER_PATTR_INET6_TIMER) },
	};
	uint32_t timer = 0xdead;
	long clk_tck;
	int precision = 0;
	clk_tck = sysconf(_SC_CLK_TCK);
	if (clk_tck > 0) {
		precision = clk_tck > 1 ? MIN((int) ceil(log10(clk_tck - 1)), 9)
					: 0;
		timer *= clk_tck;
	}
	struct nlattr nla_timer = {
		.nla_len = NLA_HDRLEN + sizeof(timer),
	};
	char buf_timer[NLMSG_ALIGN(ifindex) + NLA_HDRLEN + sizeof(timer)];
	memcpy(buf_timer, &ifindex, sizeof(ifindex));
	memcpy(buf_timer + NLMSG_ALIGN(ifindex) + NLA_HDRLEN,
	       &timer, sizeof(timer));
	for (size_t i = 0; i < ARRAY_SIZE(pattrs); i++) {
		nla_timer.nla_type = pattrs[i].val;
		memcpy(buf_timer + NLMSG_ALIGN(ifindex),
		       &nla_timer, sizeof(nla_timer));
		TEST_NLATTR(fd, nlh0 - NLA_HDRLEN, hdrlen + NLA_HDRLEN,
			    init_br_port_msg, print_br_port_msg,
			    MDBA_ROUTER_PORT,
			    sizeof(buf_timer), buf_timer, sizeof(buf_timer),
			    printf(IFINDEX_LO_STR
				   ", [{nla_len=%u, nla_type=%s}, %u",
				   nla_timer.nla_len, pattrs[i].str, timer);
			    if (clk_tck > 0)
				    printf(" /* 57005.%0*u s */", precision, 0);
			    printf("]]"));
	}
	puts("+++ exited with 0 +++");
	return 0;
}
 |