| /* |
| * Copyright (c) 2016 Fabien Siron <[email protected]> |
| * Copyright (c) 2017 JingPiao Chen <[email protected]> |
| * 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 "defs.h" |
| #include "netlink.h" |
| #include "netlink_sock_diag.h" |
| #include "nlattr.h" |
| #include "print_fields.h" |
| |
| #include <arpa/inet.h> |
| |
| #include <linux/sock_diag.h> |
| #include <linux/inet_diag.h> |
| |
| #include "xlat/inet_diag_attrs.h" |
| #include "xlat/inet_diag_bytecodes.h" |
| #include "xlat/inet_diag_extended_flags.h" |
| #include "xlat/inet_diag_req_attrs.h" |
| |
| #include "xlat/tcp_states.h" |
| #include "xlat/tcp_state_flags.h" |
| |
| void |
| print_inet_diag_sockid(const struct inet_diag_sockid *id, const uint8_t family) |
| { |
| PRINT_FIELD_NET_PORT("{", *id, idiag_sport); |
| PRINT_FIELD_NET_PORT(", ", *id, idiag_dport); |
| PRINT_FIELD_INET_ADDR(", ", *id, idiag_src, family); |
| PRINT_FIELD_INET_ADDR(", ", *id, idiag_dst, family); |
| PRINT_FIELD_IFINDEX(", ", *id, idiag_if); |
| PRINT_FIELD_COOKIE(", ", *id, idiag_cookie); |
| tprints("}"); |
| } |
| |
| static void |
| decode_inet_diag_hostcond(struct tcb *const tcp, |
| const kernel_ulong_t addr, |
| const unsigned int len) |
| { |
| struct inet_diag_hostcond cond; |
| |
| if (len < sizeof(cond)) { |
| printstr_ex(tcp, addr, len, QUOTE_FORCE_HEX); |
| return; |
| } |
| if (umove_or_printaddr(tcp, addr, &cond)) |
| return; |
| |
| PRINT_FIELD_XVAL("{", cond, family, addrfams, "AF_???"); |
| PRINT_FIELD_U(", ", cond, prefix_len); |
| PRINT_FIELD_U(", ", cond, port); |
| |
| if (len > sizeof(cond)) { |
| tprints(", "); |
| decode_inet_addr(tcp, addr + sizeof(cond), |
| len - sizeof(cond), cond.family, "addr"); |
| } |
| tprints("}"); |
| } |
| |
| static void |
| print_inet_diag_bc_op(const struct inet_diag_bc_op *const op) |
| { |
| PRINT_FIELD_XVAL("{", *op, code, inet_diag_bytecodes, |
| "INET_DIAG_BC_???"); |
| PRINT_FIELD_U(", ", *op, yes); |
| PRINT_FIELD_U(", ", *op, no); |
| tprints("}"); |
| } |
| |
| static void |
| decode_inet_diag_markcond(struct tcb *const tcp, |
| const kernel_ulong_t addr, |
| const unsigned int len) |
| { |
| struct inet_diag_markcond markcond; |
| |
| if (len < sizeof(markcond)) |
| printstr_ex(tcp, addr, len, QUOTE_FORCE_HEX); |
| else if (!umove_or_printaddr(tcp, addr, &markcond)) { |
| PRINT_FIELD_U("{", markcond, mark); |
| PRINT_FIELD_U(", ", markcond, mask); |
| tprints("}"); |
| } |
| } |
| |
| static void |
| decode_bytecode_data(struct tcb *const tcp, |
| const kernel_ulong_t addr, |
| const unsigned int len, |
| const unsigned char code) |
| { |
| switch (code) { |
| case INET_DIAG_BC_S_COND: |
| case INET_DIAG_BC_D_COND: |
| decode_inet_diag_hostcond(tcp, addr, len); |
| break; |
| case INET_DIAG_BC_DEV_COND: { |
| uint32_t ifindex; |
| |
| if (len < sizeof(ifindex)) |
| printstr_ex(tcp, addr, len, QUOTE_FORCE_HEX); |
| else if (!umove_or_printaddr(tcp, addr, &ifindex)) |
| print_ifindex(ifindex); |
| break; |
| } |
| case INET_DIAG_BC_S_GE: |
| case INET_DIAG_BC_S_LE: |
| case INET_DIAG_BC_D_GE: |
| case INET_DIAG_BC_D_LE: { |
| struct inet_diag_bc_op op; |
| |
| if (len < sizeof(op)) |
| printstr_ex(tcp, addr, len, QUOTE_FORCE_HEX); |
| else if (!umove_or_printaddr(tcp, addr, &op)) |
| print_inet_diag_bc_op(&op); |
| break; |
| } |
| case INET_DIAG_BC_MARK_COND: |
| decode_inet_diag_markcond(tcp, addr, len); |
| break; |
| case INET_DIAG_BC_AUTO: |
| case INET_DIAG_BC_JMP: |
| case INET_DIAG_BC_NOP: |
| default: |
| printstr_ex(tcp, addr, len, QUOTE_FORCE_HEX); |
| break; |
| } |
| } |
| |
| static bool |
| decode_inet_diag_bc_op(struct tcb *const tcp, |
| const kernel_ulong_t addr, |
| const unsigned int len, |
| const void *const opaque_data) |
| { |
| struct inet_diag_bc_op op; |
| |
| if (len < sizeof(op)) |
| return false; |
| if (umove_or_printaddr(tcp, addr, &op)) |
| return true; |
| |
| if (len > sizeof(op)) |
| tprints("{"); |
| |
| print_inet_diag_bc_op(&op); |
| |
| if (len > sizeof(op)) { |
| tprints(", "); |
| decode_bytecode_data(tcp, addr + sizeof(op), |
| len - sizeof(op), op.code); |
| tprints("}"); |
| } |
| |
| return true; |
| } |
| |
| static const nla_decoder_t inet_diag_req_nla_decoders[] = { |
| [INET_DIAG_REQ_BYTECODE] = decode_inet_diag_bc_op |
| }; |
| |
| static void |
| decode_inet_diag_req_compat(struct tcb *const tcp, |
| const struct nlmsghdr *const nlmsghdr, |
| const uint8_t family, |
| const kernel_ulong_t addr, |
| const unsigned int len) |
| { |
| struct inet_diag_req req = { .idiag_family = family }; |
| size_t offset = sizeof(req.idiag_family); |
| bool decode_nla = false; |
| |
| PRINT_FIELD_XVAL("{", req, idiag_family, addrfams, "AF_???"); |
| tprints(", "); |
| if (len >= sizeof(req)) { |
| if (!umoven_or_printaddr(tcp, addr + offset, |
| sizeof(req) - offset, |
| (char *) &req + offset)) { |
| PRINT_FIELD_U("", req, idiag_src_len); |
| PRINT_FIELD_U(", ", req, idiag_dst_len); |
| PRINT_FIELD_FLAGS(", ", req, idiag_ext, |
| inet_diag_extended_flags, |
| "1<<INET_DIAG_\?\?\?-1"); |
| PRINT_FIELD_INET_DIAG_SOCKID(", ", req, id, |
| req.idiag_family); |
| PRINT_FIELD_FLAGS(", ", req, idiag_states, |
| tcp_state_flags, "1<<TCP_???"); |
| PRINT_FIELD_U(", ", req, idiag_dbs); |
| decode_nla = true; |
| } |
| } else |
| tprints("..."); |
| tprints("}"); |
| |
| offset = NLMSG_ALIGN(sizeof(req)); |
| if (decode_nla && len > offset) { |
| tprints(", "); |
| decode_nlattr(tcp, addr + offset, len - offset, |
| inet_diag_req_attrs, "INET_DIAG_REQ_???", |
| inet_diag_req_nla_decoders, |
| ARRAY_SIZE(inet_diag_req_nla_decoders), NULL); |
| } |
| } |
| |
| static void |
| decode_inet_diag_req_v2(struct tcb *const tcp, |
| const struct nlmsghdr *const nlmsghdr, |
| const uint8_t family, |
| const kernel_ulong_t addr, |
| const unsigned int len) |
| { |
| struct inet_diag_req_v2 req = { .sdiag_family = family }; |
| size_t offset = sizeof(req.sdiag_family); |
| bool decode_nla = false; |
| |
| PRINT_FIELD_XVAL("{", req, sdiag_family, addrfams, "AF_???"); |
| tprints(", "); |
| if (len >= sizeof(req)) { |
| if (!umoven_or_printaddr(tcp, addr + offset, |
| sizeof(req) - offset, |
| (char *) &req + offset)) { |
| PRINT_FIELD_XVAL("", req, sdiag_protocol, |
| inet_protocols, "IPPROTO_???"); |
| PRINT_FIELD_FLAGS(", ", req, idiag_ext, |
| inet_diag_extended_flags, |
| "1<<INET_DIAG_\?\?\?-1"); |
| PRINT_FIELD_FLAGS(", ", req, idiag_states, |
| tcp_state_flags, "1<<TCP_???"); |
| PRINT_FIELD_INET_DIAG_SOCKID(", ", req, id, |
| req.sdiag_family); |
| decode_nla = true; |
| } |
| } else |
| tprints("..."); |
| tprints("}"); |
| |
| offset = NLMSG_ALIGN(sizeof(req)); |
| if (decode_nla && len > offset) { |
| tprints(", "); |
| decode_nlattr(tcp, addr + offset, len - offset, |
| inet_diag_req_attrs, "INET_DIAG_REQ_???", |
| inet_diag_req_nla_decoders, |
| ARRAY_SIZE(inet_diag_req_nla_decoders), NULL); |
| } |
| } |
| |
| DECL_NETLINK_DIAG_DECODER(decode_inet_diag_req) |
| { |
| if (nlmsghdr->nlmsg_type == TCPDIAG_GETSOCK |
| || nlmsghdr->nlmsg_type == DCCPDIAG_GETSOCK) |
| decode_inet_diag_req_compat(tcp, nlmsghdr, family, addr, len); |
| else |
| decode_inet_diag_req_v2(tcp, nlmsghdr, family, addr, len); |
| } |
| |
| static bool |
| decode_inet_diag_meminfo(struct tcb *const tcp, |
| const kernel_ulong_t addr, |
| const unsigned int len, |
| const void *const opaque_data) |
| { |
| struct inet_diag_meminfo minfo; |
| |
| if (len < sizeof(minfo)) |
| return false; |
| if (umove_or_printaddr(tcp, addr, &minfo)) |
| return true; |
| |
| PRINT_FIELD_U("{", minfo, idiag_rmem); |
| PRINT_FIELD_U(", ", minfo, idiag_wmem); |
| PRINT_FIELD_U(", ", minfo, idiag_fmem); |
| PRINT_FIELD_U(", ", minfo, idiag_tmem); |
| tprints("}"); |
| |
| return true; |
| } |
| |
| static bool |
| decode_tcpvegas_info(struct tcb *const tcp, |
| const kernel_ulong_t addr, |
| const unsigned int len, |
| const void *const opaque_data) |
| { |
| struct tcpvegas_info vegas; |
| |
| if (len < sizeof(vegas)) |
| return false; |
| if (umove_or_printaddr(tcp, addr, &vegas)) |
| return true; |
| |
| PRINT_FIELD_U("{", vegas, tcpv_enabled); |
| PRINT_FIELD_U(", ", vegas, tcpv_rttcnt); |
| PRINT_FIELD_U(", ", vegas, tcpv_rtt); |
| PRINT_FIELD_U(", ", vegas, tcpv_minrtt); |
| tprints("}"); |
| |
| return true; |
| } |
| |
| static bool |
| decode_tcp_dctcp_info(struct tcb *const tcp, |
| const kernel_ulong_t addr, |
| const unsigned int len, |
| const void *const opaque_data) |
| { |
| struct tcp_dctcp_info dctcp; |
| |
| if (len < sizeof(dctcp)) |
| return false; |
| if (umove_or_printaddr(tcp, addr, &dctcp)) |
| return true; |
| |
| PRINT_FIELD_U("{", dctcp, dctcp_enabled); |
| PRINT_FIELD_U(", ", dctcp, dctcp_ce_state); |
| PRINT_FIELD_U(", ", dctcp, dctcp_alpha); |
| PRINT_FIELD_U(", ", dctcp, dctcp_ab_ecn); |
| PRINT_FIELD_U(", ", dctcp, dctcp_ab_tot); |
| tprints("}"); |
| |
| return true; |
| } |
| |
| static bool |
| decode_tcp_bbr_info(struct tcb *const tcp, |
| const kernel_ulong_t addr, |
| const unsigned int len, |
| const void *const opaque_data) |
| { |
| struct tcp_bbr_info bbr; |
| |
| if (len < sizeof(bbr)) |
| return false; |
| if (umove_or_printaddr(tcp, addr, &bbr)) |
| return true; |
| |
| PRINT_FIELD_X("{", bbr, bbr_bw_lo); |
| PRINT_FIELD_X(", ", bbr, bbr_bw_hi); |
| PRINT_FIELD_U(", ", bbr, bbr_min_rtt); |
| PRINT_FIELD_U(", ", bbr, bbr_pacing_gain); |
| PRINT_FIELD_U(", ", bbr, bbr_cwnd_gain); |
| tprints("}"); |
| |
| return true; |
| } |
| |
| static const nla_decoder_t inet_diag_msg_nla_decoders[] = { |
| [INET_DIAG_MEMINFO] = decode_inet_diag_meminfo, |
| [INET_DIAG_INFO] = NULL, /* unimplemented */ |
| [INET_DIAG_VEGASINFO] = decode_tcpvegas_info, |
| [INET_DIAG_CONG] = decode_nla_str, |
| [INET_DIAG_TOS] = decode_nla_u8, |
| [INET_DIAG_TCLASS] = decode_nla_u8, |
| [INET_DIAG_SKMEMINFO] = decode_nla_meminfo, |
| [INET_DIAG_SHUTDOWN] = decode_nla_u8, |
| [INET_DIAG_DCTCPINFO] = decode_tcp_dctcp_info, |
| [INET_DIAG_PROTOCOL] = decode_nla_u8, |
| [INET_DIAG_SKV6ONLY] = decode_nla_u8, |
| [INET_DIAG_LOCALS] = NULL, /* unimplemented */ |
| [INET_DIAG_PEERS] = NULL, /* unimplemented */ |
| [INET_DIAG_PAD] = NULL, |
| [INET_DIAG_MARK] = decode_nla_u32, |
| [INET_DIAG_BBRINFO] = decode_tcp_bbr_info, |
| [INET_DIAG_CLASS_ID] = decode_nla_u32 |
| }; |
| |
| DECL_NETLINK_DIAG_DECODER(decode_inet_diag_msg) |
| { |
| struct inet_diag_msg msg = { .idiag_family = family }; |
| size_t offset = sizeof(msg.idiag_family); |
| bool decode_nla = false; |
| |
| PRINT_FIELD_XVAL("{", msg, idiag_family, addrfams, "AF_???"); |
| tprints(", "); |
| if (len >= sizeof(msg)) { |
| if (!umoven_or_printaddr(tcp, addr + offset, |
| sizeof(msg) - offset, |
| (char *) &msg + offset)) { |
| PRINT_FIELD_XVAL("", msg, idiag_state, |
| tcp_states, "TCP_???"); |
| PRINT_FIELD_U(", ", msg, idiag_timer); |
| PRINT_FIELD_U(", ", msg, idiag_retrans); |
| PRINT_FIELD_INET_DIAG_SOCKID(", ", msg, id, |
| msg.idiag_family); |
| PRINT_FIELD_U(", ", msg, idiag_expires); |
| PRINT_FIELD_U(", ", msg, idiag_rqueue); |
| PRINT_FIELD_U(", ", msg, idiag_wqueue); |
| PRINT_FIELD_U(", ", msg, idiag_uid); |
| PRINT_FIELD_U(", ", msg, idiag_inode); |
| decode_nla = true; |
| } |
| } else |
| tprints("..."); |
| tprints("}"); |
| |
| offset = NLMSG_ALIGN(sizeof(msg)); |
| if (decode_nla && len > offset) { |
| tprints(", "); |
| decode_nlattr(tcp, addr + offset, len - offset, |
| inet_diag_attrs, "INET_DIAG_???", |
| inet_diag_msg_nla_decoders, |
| ARRAY_SIZE(inet_diag_msg_nla_decoders), NULL); |
| } |
| } |