| /* |
| * Copyright (c) 2016 Fabien Siron <[email protected]> |
| * Copyright (c) 2017 JingPiao Chen <[email protected]> |
| * Copyright (c) 2016-2017 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 "defs.h" |
| #include "netlink_route.h" |
| #include "nlattr.h" |
| #include "print_fields.h" |
| |
| #include "netlink.h" |
| #include <linux/rtnetlink.h> |
| #ifdef HAVE_LINUX_NEIGHBOUR_H |
| # include <linux/neighbour.h> |
| #endif |
| |
| #include "xlat/rtnl_neightbl_attrs.h" |
| #include "xlat/rtnl_neightbl_parms_attrs.h" |
| |
| static bool |
| decode_ndt_config(struct tcb *const tcp, |
| const kernel_ulong_t addr, |
| const unsigned int len, |
| const void *const opaque_data) |
| { |
| #ifdef HAVE_STRUCT_NDT_CONFIG |
| struct ndt_config ndtc; |
| |
| if (len < sizeof(ndtc)) |
| return false; |
| else if (!umove_or_printaddr(tcp, addr, &ndtc)) { |
| PRINT_FIELD_U("{", ndtc, ndtc_key_len); |
| PRINT_FIELD_U(", ", ndtc, ndtc_entry_size); |
| PRINT_FIELD_U(", ", ndtc, ndtc_entries); |
| PRINT_FIELD_U(", ", ndtc, ndtc_last_flush); |
| PRINT_FIELD_U(", ", ndtc, ndtc_last_rand); |
| PRINT_FIELD_U(", ", ndtc, ndtc_hash_rnd); |
| PRINT_FIELD_0X(", ", ndtc, ndtc_hash_mask); |
| PRINT_FIELD_U(", ", ndtc, ndtc_hash_chain_gc); |
| PRINT_FIELD_U(", ", ndtc, ndtc_proxy_qlen); |
| tprints("}"); |
| } |
| |
| return true; |
| #else |
| return false; |
| #endif |
| } |
| |
| static const nla_decoder_t ndt_parms_nla_decoders[] = { |
| [NDTPA_IFINDEX] = decode_nla_ifindex, |
| [NDTPA_REFCNT] = decode_nla_u32, |
| [NDTPA_REACHABLE_TIME] = decode_nla_u64, |
| [NDTPA_BASE_REACHABLE_TIME] = decode_nla_u64, |
| [NDTPA_RETRANS_TIME] = decode_nla_u64, |
| [NDTPA_GC_STALETIME] = decode_nla_u64, |
| [NDTPA_DELAY_PROBE_TIME] = decode_nla_u64, |
| [NDTPA_QUEUE_LEN] = decode_nla_u32, |
| [NDTPA_APP_PROBES] = decode_nla_u32, |
| [NDTPA_UCAST_PROBES] = decode_nla_u32, |
| [NDTPA_MCAST_PROBES] = decode_nla_u32, |
| [NDTPA_ANYCAST_DELAY] = decode_nla_u64, |
| [NDTPA_PROXY_DELAY] = decode_nla_u64, |
| [NDTPA_PROXY_QLEN] = decode_nla_u32, |
| [NDTPA_LOCKTIME] = decode_nla_u64, |
| [NDTPA_QUEUE_LENBYTES] = decode_nla_u32, |
| [NDTPA_MCAST_REPROBES] = decode_nla_u32, |
| [NDTPA_PAD] = NULL |
| }; |
| |
| static bool |
| decode_ndta_parms(struct tcb *const tcp, |
| const kernel_ulong_t addr, |
| const unsigned int len, |
| const void *const opaque_data) |
| { |
| decode_nlattr(tcp, addr, len, rtnl_neightbl_parms_attrs, "NDTPA_???", |
| ndt_parms_nla_decoders, |
| ARRAY_SIZE(ndt_parms_nla_decoders), opaque_data); |
| |
| return true; |
| } |
| |
| static bool |
| decode_ndt_stats(struct tcb *const tcp, |
| const kernel_ulong_t addr, |
| const unsigned int len, |
| const void *const opaque_data) |
| { |
| #ifdef HAVE_STRUCT_NDT_STATS |
| struct ndt_stats ndtst; |
| const unsigned int min_size = |
| offsetofend(struct ndt_stats, ndts_forced_gc_runs); |
| const unsigned int def_size = sizeof(ndtst); |
| const unsigned int size = |
| (len >= def_size) ? def_size : |
| ((len == min_size) ? min_size : 0); |
| |
| if (!size) |
| return false; |
| |
| if (!umoven_or_printaddr(tcp, addr, size, &ndtst)) { |
| PRINT_FIELD_U("{", ndtst, ndts_allocs); |
| PRINT_FIELD_U(", ", ndtst, ndts_destroys); |
| PRINT_FIELD_U(", ", ndtst, ndts_hash_grows); |
| PRINT_FIELD_U(", ", ndtst, ndts_res_failed); |
| PRINT_FIELD_U(", ", ndtst, ndts_lookups); |
| PRINT_FIELD_U(", ", ndtst, ndts_hits); |
| PRINT_FIELD_U(", ", ndtst, ndts_rcv_probes_mcast); |
| PRINT_FIELD_U(", ", ndtst, ndts_rcv_probes_ucast); |
| PRINT_FIELD_U(", ", ndtst, ndts_periodic_gc_runs); |
| PRINT_FIELD_U(", ", ndtst, ndts_forced_gc_runs); |
| #ifdef HAVE_STRUCT_NDT_STATS_NDTS_TABLE_FULLS |
| if (len >= def_size) |
| PRINT_FIELD_U(", ", ndtst, ndts_table_fulls); |
| #endif |
| tprints("}"); |
| } |
| |
| return true; |
| #else |
| return false; |
| #endif |
| } |
| |
| static const nla_decoder_t ndtmsg_nla_decoders[] = { |
| [NDTA_NAME] = decode_nla_str, |
| [NDTA_THRESH1] = decode_nla_u32, |
| [NDTA_THRESH2] = decode_nla_u32, |
| [NDTA_THRESH3] = decode_nla_u32, |
| [NDTA_CONFIG] = decode_ndt_config, |
| [NDTA_PARMS] = decode_ndta_parms, |
| [NDTA_STATS] = decode_ndt_stats, |
| [NDTA_GC_INTERVAL] = decode_nla_u64, |
| [NDTA_PAD] = NULL, |
| }; |
| |
| DECL_NETLINK_ROUTE_DECODER(decode_ndtmsg) |
| { |
| struct ndtmsg ndtmsg = { .ndtm_family = family }; |
| |
| PRINT_FIELD_XVAL("{", ndtmsg, ndtm_family, addrfams, "AF_???"); |
| tprints("}"); |
| |
| const size_t offset = NLMSG_ALIGN(sizeof(ndtmsg)); |
| if (len > offset) { |
| tprints(", "); |
| decode_nlattr(tcp, addr + offset, len - offset, |
| rtnl_neightbl_attrs, "NDTA_???", |
| ndtmsg_nla_decoders, |
| ARRAY_SIZE(ndtmsg_nla_decoders), NULL); |
| } |
| } |