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-2017 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 #include "netlink_route.h"
     32 #include "nlattr.h"
     33 #include "print_fields.h"
     34 
     35 #include "netlink.h"
     36 #include <linux/rtnetlink.h>
     37 #ifdef HAVE_LINUX_NEIGHBOUR_H
     38 # include <linux/neighbour.h>
     39 #endif
     40 
     41 #include "xlat/rtnl_neightbl_attrs.h"
     42 #include "xlat/rtnl_neightbl_parms_attrs.h"
     43 
     44 static bool
     45 decode_ndt_config(struct tcb *const tcp,
     46 		  const kernel_ulong_t addr,
     47 		  const unsigned int len,
     48 		  const void *const opaque_data)
     49 {
     50 #ifdef HAVE_STRUCT_NDT_CONFIG
     51 	struct ndt_config ndtc;
     52 
     53 	if (len < sizeof(ndtc))
     54 		return false;
     55 	else if (!umove_or_printaddr(tcp, addr, &ndtc)) {
     56 		PRINT_FIELD_U("{", ndtc, ndtc_key_len);
     57 		PRINT_FIELD_U(", ", ndtc, ndtc_entry_size);
     58 		PRINT_FIELD_U(", ", ndtc, ndtc_entries);
     59 		PRINT_FIELD_U(", ", ndtc, ndtc_last_flush);
     60 		PRINT_FIELD_U(", ", ndtc, ndtc_last_rand);
     61 		PRINT_FIELD_U(", ", ndtc, ndtc_hash_rnd);
     62 		PRINT_FIELD_0X(", ", ndtc, ndtc_hash_mask);
     63 		PRINT_FIELD_U(", ", ndtc, ndtc_hash_chain_gc);
     64 		PRINT_FIELD_U(", ", ndtc, ndtc_proxy_qlen);
     65 		tprints("}");
     66 	}
     67 
     68 	return true;
     69 #else
     70 	return false;
     71 #endif
     72 }
     73 
     74 static const nla_decoder_t ndt_parms_nla_decoders[] = {
     75 	[NDTPA_IFINDEX]			= decode_nla_ifindex,
     76 	[NDTPA_REFCNT]			= decode_nla_u32,
     77 	[NDTPA_REACHABLE_TIME]		= decode_nla_u64,
     78 	[NDTPA_BASE_REACHABLE_TIME]	= decode_nla_u64,
     79 	[NDTPA_RETRANS_TIME]		= decode_nla_u64,
     80 	[NDTPA_GC_STALETIME]		= decode_nla_u64,
     81 	[NDTPA_DELAY_PROBE_TIME]	= decode_nla_u64,
     82 	[NDTPA_QUEUE_LEN]		= decode_nla_u32,
     83 	[NDTPA_APP_PROBES]		= decode_nla_u32,
     84 	[NDTPA_UCAST_PROBES]		= decode_nla_u32,
     85 	[NDTPA_MCAST_PROBES]		= decode_nla_u32,
     86 	[NDTPA_ANYCAST_DELAY]		= decode_nla_u64,
     87 	[NDTPA_PROXY_DELAY]		= decode_nla_u64,
     88 	[NDTPA_PROXY_QLEN]		= decode_nla_u32,
     89 	[NDTPA_LOCKTIME]		= decode_nla_u64,
     90 	[NDTPA_QUEUE_LENBYTES]		= decode_nla_u32,
     91 	[NDTPA_MCAST_REPROBES]		= decode_nla_u32,
     92 	[NDTPA_PAD]			= NULL
     93 };
     94 
     95 static bool
     96 decode_ndta_parms(struct tcb *const tcp,
     97 		  const kernel_ulong_t addr,
     98 		  const unsigned int len,
     99 		  const void *const opaque_data)
    100 {
    101 	decode_nlattr(tcp, addr, len, rtnl_neightbl_parms_attrs, "NDTPA_???",
    102 		      ndt_parms_nla_decoders,
    103 		      ARRAY_SIZE(ndt_parms_nla_decoders), opaque_data);
    104 
    105 	return true;
    106 }
    107 
    108 static bool
    109 decode_ndt_stats(struct tcb *const tcp,
    110 		 const kernel_ulong_t addr,
    111 		 const unsigned int len,
    112 		 const void *const opaque_data)
    113 {
    114 #ifdef HAVE_STRUCT_NDT_STATS
    115 	struct ndt_stats ndtst;
    116 	const unsigned int min_size =
    117 		offsetofend(struct ndt_stats, ndts_forced_gc_runs);
    118 	const unsigned int def_size = sizeof(ndtst);
    119 	const unsigned int size =
    120 		(len >= def_size) ? def_size :
    121 				    ((len == min_size) ? min_size : 0);
    122 
    123 	if (!size)
    124 		return false;
    125 
    126 	if (!umoven_or_printaddr(tcp, addr, size, &ndtst)) {
    127 		PRINT_FIELD_U("{", ndtst, ndts_allocs);
    128 		PRINT_FIELD_U(", ", ndtst, ndts_destroys);
    129 		PRINT_FIELD_U(", ", ndtst, ndts_hash_grows);
    130 		PRINT_FIELD_U(", ", ndtst, ndts_res_failed);
    131 		PRINT_FIELD_U(", ", ndtst, ndts_lookups);
    132 		PRINT_FIELD_U(", ", ndtst, ndts_hits);
    133 		PRINT_FIELD_U(", ", ndtst, ndts_rcv_probes_mcast);
    134 		PRINT_FIELD_U(", ", ndtst, ndts_rcv_probes_ucast);
    135 		PRINT_FIELD_U(", ", ndtst, ndts_periodic_gc_runs);
    136 		PRINT_FIELD_U(", ", ndtst, ndts_forced_gc_runs);
    137 #ifdef HAVE_STRUCT_NDT_STATS_NDTS_TABLE_FULLS
    138 		if (len >= def_size)
    139 			PRINT_FIELD_U(", ", ndtst, ndts_table_fulls);
    140 #endif
    141 		tprints("}");
    142 	}
    143 
    144 	return true;
    145 #else
    146 	return false;
    147 #endif
    148 }
    149 
    150 static const nla_decoder_t ndtmsg_nla_decoders[] = {
    151 	[NDTA_NAME]		= decode_nla_str,
    152 	[NDTA_THRESH1]		= decode_nla_u32,
    153 	[NDTA_THRESH2]		= decode_nla_u32,
    154 	[NDTA_THRESH3]		= decode_nla_u32,
    155 	[NDTA_CONFIG]		= decode_ndt_config,
    156 	[NDTA_PARMS]		= decode_ndta_parms,
    157 	[NDTA_STATS]		= decode_ndt_stats,
    158 	[NDTA_GC_INTERVAL]	= decode_nla_u64,
    159 	[NDTA_PAD]		= NULL,
    160 };
    161 
    162 DECL_NETLINK_ROUTE_DECODER(decode_ndtmsg)
    163 {
    164 	struct ndtmsg ndtmsg = { .ndtm_family = family };
    165 
    166 	PRINT_FIELD_XVAL("{", ndtmsg, ndtm_family, addrfams, "AF_???");
    167 	tprints("}");
    168 
    169 	const size_t offset = NLMSG_ALIGN(sizeof(ndtmsg));
    170 	if (len > offset) {
    171 		tprints(", ");
    172 		decode_nlattr(tcp, addr + offset, len - offset,
    173 			      rtnl_neightbl_attrs, "NDTA_???",
    174 			      ndtmsg_nla_decoders,
    175 			      ARRAY_SIZE(ndtmsg_nla_decoders), NULL);
    176 	}
    177 }
    178