/* * Copyright (c) 2017 JingPiao Chen <chenjingpiao@gmail.com> * Copyright (c) 2017-2018 The strace developers. * All rights reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions * are met: * 1. Redistributions of source code must retain the above copyright * notice, this list of conditions and the following disclaimer. * 2. Redistributions in binary form must reproduce the above copyright * notice, this list of conditions and the following disclaimer in the * documentation and/or other materials provided with the distribution. * 3. The name of the author may not be used to endorse or promote products * derived from this software without specific prior written permission. * * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT, * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. */ #include "tests.h" #ifdef HAVE_STRUCT_BR_PORT_MSG # include <stdio.h> # include "test_nlattr.h" # include <arpa/inet.h> # include <linux/if_bridge.h> # include <linux/rtnetlink.h> # ifndef MDB_TEMPORARY # define MDB_TEMPORARY 0 # endif # ifndef MDBA_MDB_ENTRY_INFO # define MDBA_MDB_ENTRY_INFO 1 # endif # ifndef MDBA_MDB_EATTR_TIMER # define MDBA_MDB_EATTR_TIMER 1 # endif const unsigned int hdrlen = sizeof(struct br_port_msg); static void init_br_port_msg(struct nlmsghdr *const nlh, const unsigned int msg_len) { unsigned int len = msg_len; SET_STRUCT(struct nlmsghdr, nlh, .nlmsg_len = 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)); len -= NLMSG_SPACE(hdrlen); SET_STRUCT(struct nlattr, nla, .nla_len = len, .nla_type = MDBA_MDB ); nla = nla + 1; len -= NLA_HDRLEN; SET_STRUCT(struct nlattr, nla, .nla_len = len, .nla_type = MDBA_MDB_ENTRY ); } static void print_br_port_msg(const unsigned int msg_len) { printf("{len=%u, type=RTM_GETMDB, flags=NLM_F_DUMP" ", seq=0, pid=0}, {family=AF_UNIX" ", ifindex=" IFINDEX_LO_STR "}" ", {{nla_len=%u, nla_type=MDBA_MDB}" ", {{nla_len=%u, nla_type=MDBA_MDB_ENTRY}", msg_len, msg_len - NLMSG_SPACE(hdrlen), msg_len - NLMSG_SPACE(hdrlen) - NLA_HDRLEN); } int main(void) { skip_if_unavailable("/proc/self/fd/"); const int fd = create_nl_socket(NETLINK_ROUTE); void *nlh0 = midtail_alloc(NLMSG_SPACE(hdrlen), NLA_HDRLEN + 4 # ifdef HAVE_STRUCT_BR_MDB_ENTRY - 4 + NLA_HDRLEN * 2 + sizeof(struct nlattr) + sizeof(struct br_mdb_entry) # endif ); static char pattern[4096]; fill_memory_ex(pattern, sizeof(pattern), 'a', 'z' - 'a' + 1); const unsigned int nla_type = 0xffff & NLA_TYPE_MASK; char nla_type_str[256]; sprintf(nla_type_str, "%#x /* MDBA_MDB_ENTRY_??? */", nla_type); TEST_NLATTR_(fd, nlh0 - NLA_HDRLEN * 2, hdrlen + NLA_HDRLEN * 2, init_br_port_msg, print_br_port_msg, nla_type, nla_type_str, 4, pattern, 4, print_quoted_hex(pattern, 4); printf("}}")); # ifdef HAVE_STRUCT_BR_MDB_ENTRY struct br_mdb_entry entry = { .ifindex = ifindex_lo(), .state = MDB_TEMPORARY, # ifdef HAVE_STRUCT_BR_MDB_ENTRY_FLAGS .flags = MDB_FLAGS_OFFLOAD, # endif # ifdef HAVE_STRUCT_BR_MDB_ENTRY_VID .vid = 0xcdef, # endif .addr = { .proto = htons(AF_UNSPEC) } }; memcpy(&entry.addr.u, pattern, sizeof(entry.addr.u)); TEST_NESTED_NLATTR_OBJECT_EX(fd, nlh0, hdrlen, init_br_port_msg, print_br_port_msg, MDBA_MDB_ENTRY_INFO, pattern, entry, 2, printf("{ifindex=" IFINDEX_LO_STR); printf(", state=MDB_TEMPORARY"); # ifdef HAVE_STRUCT_BR_MDB_ENTRY_FLAGS printf(", flags=MDB_FLAGS_OFFLOAD"); # endif # ifdef HAVE_STRUCT_BR_MDB_ENTRY_VID PRINT_FIELD_U(", ", entry, vid); # endif printf(", addr={u="); print_quoted_hex(&entry.addr.u, sizeof(entry.addr.u)); printf(", proto=htons(AF_UNSPEC)}}")); static const struct nlattr nla = { .nla_len = sizeof(nla), .nla_type = MDBA_MDB_EATTR_TIMER }; char buf[NLMSG_ALIGN(sizeof(entry)) + sizeof(nla)]; memcpy(buf, &entry, sizeof(entry)); memcpy(buf + NLMSG_ALIGN(sizeof(entry)), &nla, sizeof(nla)); TEST_NLATTR(fd, nlh0 - NLA_HDRLEN * 2, hdrlen + NLA_HDRLEN * 2, init_br_port_msg, print_br_port_msg, MDBA_MDB_ENTRY_INFO, sizeof(buf), buf, sizeof(buf), printf("{ifindex=" IFINDEX_LO_STR); printf(", state=MDB_TEMPORARY"); # ifdef HAVE_STRUCT_BR_MDB_ENTRY_FLAGS printf(", flags=MDB_FLAGS_OFFLOAD"); # endif # ifdef HAVE_STRUCT_BR_MDB_ENTRY_VID PRINT_FIELD_U(", ", entry, vid); # endif printf(", addr={u="); print_quoted_hex(&entry.addr.u, sizeof(entry.addr.u)); printf(", proto=htons(AF_UNSPEC)}}" ", {nla_len=%u, nla_type=MDBA_MDB_EATTR_TIMER}}}", nla.nla_len)); # endif /* HAVE_STRUCT_BR_MDB_ENTRY */ puts("+++ exited with 0 +++"); return 0; } #else SKIP_MAIN_UNDEFINED("HAVE_STRUCT_BR_PORT_MSG") #endif