Home | History | Annotate | Download | only in strace
      1 /*
      2  * Copyright (c) 2016 Fabien Siron <fabien.siron (at) epita.fr>
      3  * Copyright (c) 2017 JingPiao Chen <chenjingpiao (at) gmail.com>
      4  * Copyright (c) 2016-2018 The strace developers.
      5  * All rights reserved.
      6  *
      7  * Redistribution and use in source and binary forms, with or without
      8  * modification, are permitted provided that the following conditions
      9  * are met:
     10  * 1. Redistributions of source code must retain the above copyright
     11  *    notice, this list of conditions and the following disclaimer.
     12  * 2. Redistributions in binary form must reproduce the above copyright
     13  *    notice, this list of conditions and the following disclaimer in the
     14  *    documentation and/or other materials provided with the distribution.
     15  * 3. The name of the author may not be used to endorse or promote products
     16  *    derived from this software without specific prior written permission.
     17  *
     18  * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
     19  * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
     20  * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
     21  * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
     22  * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
     23  * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
     24  * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
     25  * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
     26  * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
     27  * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
     28  */
     29 
     30 #include "defs.h"
     31 
     32 #ifdef HAVE_STRUCT_BR_PORT_MSG
     33 
     34 # include "netlink_route.h"
     35 # include "nlattr.h"
     36 # include "print_fields.h"
     37 
     38 # include <netinet/in.h>
     39 # include <linux/if_bridge.h>
     40 # include "netlink.h"
     41 
     42 # ifdef HAVE_STRUCT_BR_MDB_ENTRY_FLAGS
     43 #  include "xlat/mdb_flags.h"
     44 # endif
     45 # include "xlat/mdb_states.h"
     46 # include "xlat/multicast_router_types.h"
     47 # include "xlat/rtnl_mdb_attrs.h"
     48 # include "xlat/rtnl_mdba_mdb_attrs.h"
     49 # include "xlat/rtnl_mdba_mdb_eattr_attrs.h"
     50 # include "xlat/rtnl_mdba_mdb_entry_attrs.h"
     51 # include "xlat/rtnl_mdba_router_attrs.h"
     52 # include "xlat/rtnl_mdba_router_pattr_attrs.h"
     53 
     54 static const nla_decoder_t mdba_mdb_eattr_nla_decoders[] = {
     55 	[MDBA_MDB_EATTR_TIMER]	= decode_nla_u32
     56 };
     57 
     58 static bool
     59 decode_mdba_mdb_entry_info(struct tcb *const tcp,
     60 			   const kernel_ulong_t addr,
     61 			   const unsigned int len,
     62 			   const void *const opaque_data)
     63 {
     64 # ifdef HAVE_STRUCT_BR_MDB_ENTRY
     65 	struct br_mdb_entry entry;
     66 
     67 	if (len < sizeof(entry))
     68 		return false;
     69 	else if (!umove_or_printaddr(tcp, addr, &entry)) {
     70 		PRINT_FIELD_IFINDEX("{", entry, ifindex);
     71 		PRINT_FIELD_XVAL(", ", entry, state, mdb_states, "MDB_???");
     72 #  ifdef HAVE_STRUCT_BR_MDB_ENTRY_FLAGS
     73 		PRINT_FIELD_FLAGS(", ", entry, flags,
     74 				  mdb_flags, "MDB_FLAGS_???");
     75 #  endif
     76 #  ifdef HAVE_STRUCT_BR_MDB_ENTRY_VID
     77 		PRINT_FIELD_U(", ", entry, vid);
     78 #  endif
     79 
     80 		const int proto = ntohs(entry.addr.proto);
     81 
     82 		tprints(", addr={");
     83 		print_inet_addr(proto, &entry.addr.u,
     84 				sizeof(entry.addr.u), "u");
     85 		tprints(", proto=htons(");
     86 		printxval(addrfams, proto, "AF_???");
     87 		tprints(")}}");
     88 	}
     89 
     90 	const size_t offset = NLMSG_ALIGN(sizeof(entry));
     91 	if (len > offset) {
     92 		tprints(", ");
     93 		decode_nlattr(tcp, addr + offset, len - offset,
     94 			      rtnl_mdba_mdb_eattr_attrs, "MDBA_MDB_EATTR_???",
     95 			      mdba_mdb_eattr_nla_decoders,
     96 			      ARRAY_SIZE(mdba_mdb_eattr_nla_decoders), NULL);
     97 	}
     98 
     99 	return true;
    100 # else
    101 	return false;
    102 # endif /* HAVE_STRUCT_BR_MDB_ENTRY */
    103 }
    104 
    105 static const nla_decoder_t mdba_mdb_entry_nla_decoders[] = {
    106 	[MDBA_MDB_ENTRY_INFO]	= decode_mdba_mdb_entry_info
    107 };
    108 
    109 static bool
    110 decode_mdba_mdb_entry(struct tcb *const tcp,
    111 		      const kernel_ulong_t addr,
    112 		      const unsigned int len,
    113 		      const void *const opaque_data)
    114 {
    115 	decode_nlattr(tcp, addr, len, rtnl_mdba_mdb_entry_attrs,
    116 		      "MDBA_MDB_ENTRY_???", mdba_mdb_entry_nla_decoders,
    117 		      ARRAY_SIZE(mdba_mdb_entry_nla_decoders), NULL);
    118 
    119 	return true;
    120 }
    121 
    122 static const nla_decoder_t mdba_mdb_nla_decoders[] = {
    123 	[MDBA_MDB_ENTRY]	= decode_mdba_mdb_entry
    124 };
    125 
    126 static bool
    127 decode_mdba_mdb(struct tcb *const tcp,
    128 		const kernel_ulong_t addr,
    129 		const unsigned int len,
    130 		const void *const opaque_data)
    131 {
    132 	decode_nlattr(tcp, addr, len, rtnl_mdba_mdb_attrs, "MDBA_MDB_???",
    133 		      mdba_mdb_nla_decoders,
    134 		      ARRAY_SIZE(mdba_mdb_nla_decoders), NULL);
    135 
    136 	return true;
    137 }
    138 
    139 static bool
    140 decode_multicast_router_type(struct tcb *const tcp,
    141 			     const kernel_ulong_t addr,
    142 			     const unsigned int len,
    143 			     const void *const opaque_data)
    144 {
    145 	uint8_t type;
    146 
    147 	if (!umove_or_printaddr(tcp, addr, &type))
    148 		printxval(multicast_router_types, type, "MDB_RTR_TYPE_???");
    149 
    150 	return true;
    151 }
    152 
    153 static const nla_decoder_t mdba_router_pattr_nla_decoders[] = {
    154 	[MDBA_ROUTER_PATTR_TIMER]	= decode_nla_u32,
    155 	[MDBA_ROUTER_PATTR_TYPE]	= decode_multicast_router_type
    156 };
    157 
    158 static bool
    159 decode_mdba_router_port(struct tcb *const tcp,
    160 			const kernel_ulong_t addr,
    161 			const unsigned int len,
    162 			const void *const opaque_data)
    163 {
    164 	uint32_t ifindex;
    165 
    166 	if (len < sizeof(ifindex))
    167 		return false;
    168 	else if (!umove_or_printaddr(tcp, addr, &ifindex))
    169 		print_ifindex(ifindex);
    170 
    171 	const size_t offset = NLMSG_ALIGN(sizeof(ifindex));
    172 	if (len > offset) {
    173 		tprints(", ");
    174 		decode_nlattr(tcp, addr + offset, len - offset,
    175 			      rtnl_mdba_router_pattr_attrs,
    176 			      "MDBA_ROUTER_PATTR_???",
    177 			      mdba_router_pattr_nla_decoders,
    178 			      ARRAY_SIZE(mdba_router_pattr_nla_decoders), NULL);
    179 	}
    180 
    181 	return true;
    182 }
    183 
    184 static const nla_decoder_t mdba_router_nla_decoders[] = {
    185 	[MDBA_ROUTER_PORT]	= decode_mdba_router_port
    186 };
    187 
    188 static bool
    189 decode_mdba_router(struct tcb *const tcp,
    190 		   const kernel_ulong_t addr,
    191 		   const unsigned int len,
    192 		   const void *const opaque_data)
    193 {
    194 	decode_nlattr(tcp, addr, len, rtnl_mdba_router_attrs, "MDBA_ROUTER_???",
    195 		      mdba_router_nla_decoders,
    196 		      ARRAY_SIZE(mdba_router_nla_decoders), NULL);
    197 
    198 	return true;
    199 }
    200 
    201 static const nla_decoder_t br_port_msg_nla_decoders[] = {
    202 	[MDBA_MDB]	= decode_mdba_mdb,
    203 	[MDBA_ROUTER]	= decode_mdba_router
    204 };
    205 
    206 DECL_NETLINK_ROUTE_DECODER(decode_br_port_msg)
    207 {
    208 	struct br_port_msg bpm = { .family = family };
    209 	size_t offset = sizeof(bpm.family);
    210 	bool decode_nla = false;
    211 
    212 	PRINT_FIELD_XVAL("{", bpm, family, addrfams, "AF_???");
    213 
    214 	tprints(", ");
    215 	if (len >= sizeof(bpm)) {
    216 		if (!umoven_or_printaddr(tcp, addr + offset,
    217 					 sizeof(bpm) - offset,
    218 					 (char *) &bpm + offset)) {
    219 			PRINT_FIELD_IFINDEX("", bpm, ifindex);
    220 			decode_nla = true;
    221 		}
    222 	} else
    223 		tprints("...");
    224 	tprints("}");
    225 
    226 	offset = NLMSG_ALIGN(sizeof(bpm));
    227 	if (decode_nla && len > offset) {
    228 		tprints(", ");
    229 		decode_nlattr(tcp, addr + offset, len - offset,
    230 			      rtnl_mdb_attrs, "MDBA_???",
    231 			      br_port_msg_nla_decoders,
    232 			      ARRAY_SIZE(br_port_msg_nla_decoders), NULL);
    233 	}
    234 }
    235 
    236 #endif
    237