1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087 |
- /*
- * Copyright (c) 1991, 1992 Paul Kranenburg <pk@cs.few.eur.nl>
- * Copyright (c) 1993 Branko Lankester <branko@hacktic.nl>
- * Copyright (c) 1993, 1994, 1995, 1996 Rick Sladkey <jrs@world.std.com>
- * Copyright (c) 1996-2000 Wichert Akkerman <wichert@cistron.nl>
- * Copyright (c) 1999-2019 The strace developers.
- * All rights reserved.
- *
- * SPDX-License-Identifier: LGPL-2.1-or-later
- */
-
- #include "defs.h"
- #include "print_fields.h"
-
- #include <sys/stat.h>
- #include <sys/socket.h>
- #include <sys/uio.h>
- #include <sys/un.h>
- #include <netinet/in.h>
- #ifdef HAVE_NETINET_TCP_H
- # include <netinet/tcp.h>
- #endif
- #ifdef HAVE_NETINET_UDP_H
- # include <netinet/udp.h>
- #endif
- #ifdef HAVE_NETINET_SCTP_H
- # include <netinet/sctp.h>
- #endif
- #include <arpa/inet.h>
- #include <net/if.h>
- #include <asm/types.h>
- #ifdef HAVE_NETIPX_IPX_H
- # include <netipx/ipx.h>
- #else
- # include <linux/ipx.h>
- #endif
-
- #if defined(HAVE_LINUX_IP_VS_H)
- # include <linux/ip_vs.h>
- #endif
- #include "netlink.h"
- #if defined(HAVE_LINUX_NETFILTER_ARP_ARP_TABLES_H)
- # include <linux/netfilter_arp/arp_tables.h>
- #endif
- #if defined(HAVE_LINUX_NETFILTER_BRIDGE_EBTABLES_H)
- # include <linux/netfilter_bridge/ebtables.h>
- #endif
- #if defined(HAVE_LINUX_NETFILTER_IPV4_IP_TABLES_H)
- # include <linux/netfilter_ipv4/ip_tables.h>
- #endif
- #if defined(HAVE_LINUX_NETFILTER_IPV6_IP6_TABLES_H)
- # include <linux/netfilter_ipv6/ip6_tables.h>
- #endif
- #include <linux/if_packet.h>
- #include <linux/icmp.h>
-
- #include "xlat/socktypes.h"
- #include "xlat/sock_type_flags.h"
- #ifndef SOCK_TYPE_MASK
- # define SOCK_TYPE_MASK 0xf
- #endif
-
- #include "xlat/socketlayers.h"
-
- #include "xlat/inet_protocols.h"
-
- #define XLAT_MACROS_ONLY
- #include "xlat/addrfams.h"
- #include "xlat/ethernet_protocols.h"
- #undef XLAT_MACROS_ONLY
- #include "xlat/ax25_protocols.h"
- #include "xlat/irda_protocols.h"
- #include "xlat/can_protocols.h"
- #include "xlat/bt_protocols.h"
- #include "xlat/isdn_protocols.h"
- #include "xlat/phonet_protocols.h"
- #include "xlat/caif_protocols.h"
- #include "xlat/nfc_protocols.h"
- #include "xlat/kcm_protocols.h"
- #include "xlat/smc_protocols.h"
-
- const size_t inet_protocols_size = ARRAY_SIZE(inet_protocols) - 1;
-
- static void
- decode_sockbuf(struct tcb *const tcp, const int fd, const kernel_ulong_t addr,
- const kernel_ulong_t addrlen)
- {
-
- switch (verbose(tcp) ? getfdproto(tcp, fd) : SOCK_PROTO_UNKNOWN) {
- case SOCK_PROTO_NETLINK:
- decode_netlink(tcp, fd, addr, addrlen);
- break;
- default:
- printstrn(tcp, addr, addrlen);
- }
- }
-
- /*
- * low bits of the socket type define real socket type,
- * other bits are socket type flags.
- */
- static void
- tprint_sock_type(unsigned int flags)
- {
- const char *str = xlookup(socktypes, flags & SOCK_TYPE_MASK);
-
- if (str) {
- print_xlat_ex(flags & SOCK_TYPE_MASK, str, XLAT_STYLE_DEFAULT);
- flags &= ~SOCK_TYPE_MASK;
- if (!flags)
- return;
- tprints("|");
- }
- printflags(sock_type_flags, flags, "SOCK_???");
- }
-
- SYS_FUNC(socket)
- {
- printxval(addrfams, tcp->u_arg[0], "AF_???");
- tprints(", ");
- tprint_sock_type(tcp->u_arg[1]);
- tprints(", ");
- switch (tcp->u_arg[0]) {
- case AF_INET:
- case AF_INET6:
- printxval(inet_protocols, tcp->u_arg[2], "IPPROTO_???");
- break;
-
- case AF_AX25:
- /* Those are not available in public headers. */
- printxval_ex(ax25_protocols, tcp->u_arg[2], "AX25_P_???",
- XLAT_STYLE_VERBOSE);
- break;
-
- case AF_NETLINK:
- printxval(netlink_protocols, tcp->u_arg[2], "NETLINK_???");
- break;
-
- case AF_PACKET:
- tprints("htons(");
- printxval(ethernet_protocols, ntohs(tcp->u_arg[2]),
- "ETH_P_???");
- tprints(")");
- break;
-
- case AF_IRDA:
- printxval(can_protocols, tcp->u_arg[2], "IRDAPROTO_???");
- break;
-
- case AF_CAN:
- printxval(can_protocols, tcp->u_arg[2], "CAN_???");
- break;
-
- case AF_BLUETOOTH:
- printxval(bt_protocols, tcp->u_arg[2], "BTPROTO_???");
- break;
-
- case AF_RXRPC:
- printxval(addrfams, tcp->u_arg[2], "AF_???");
- break;
-
- case AF_ISDN:
- printxval(isdn_protocols, tcp->u_arg[2], "ISDN_P_???");
- break;
-
- case AF_PHONET:
- printxval(phonet_protocols, tcp->u_arg[2], "PN_PROTO_???");
- break;
-
- case AF_CAIF:
- printxval(caif_protocols, tcp->u_arg[2], "CAIFPROTO_???");
- break;
-
- case AF_NFC:
- printxval(nfc_protocols, tcp->u_arg[2], "NFC_SOCKPROTO_???");
- break;
-
- case AF_KCM:
- printxval(kcm_protocols, tcp->u_arg[2], "KCMPROTO_???");
- break;
-
- case AF_SMC:
- printxval(smc_protocols, tcp->u_arg[2], "SMCPROTO_???");
- break;
-
- default:
- tprintf("%" PRI_klu, tcp->u_arg[2]);
- break;
- }
-
- return RVAL_DECODED | RVAL_FD;
- }
-
- static bool
- fetch_socklen(struct tcb *const tcp, int *const plen,
- const kernel_ulong_t sockaddr, const kernel_ulong_t socklen)
- {
- return verbose(tcp) && sockaddr && socklen
- && umove(tcp, socklen, plen) == 0;
- }
-
- static int
- decode_sockname(struct tcb *tcp)
- {
- int ulen, rlen;
-
- if (entering(tcp)) {
- printfd(tcp, tcp->u_arg[0]);
- tprints(", ");
- if (fetch_socklen(tcp, &ulen, tcp->u_arg[1], tcp->u_arg[2])) {
- set_tcb_priv_ulong(tcp, ulen);
- return 0;
- } else {
- printaddr(tcp->u_arg[1]);
- tprints(", ");
- printaddr(tcp->u_arg[2]);
- return RVAL_DECODED;
- }
- }
-
- ulen = get_tcb_priv_ulong(tcp);
-
- if (syserror(tcp) || umove(tcp, tcp->u_arg[2], &rlen) < 0) {
- printaddr(tcp->u_arg[1]);
- tprintf(", [%d]", ulen);
- } else {
- decode_sockaddr(tcp, tcp->u_arg[1], ulen > rlen ? rlen : ulen);
- if (ulen != rlen)
- tprintf(", [%d->%d]", ulen, rlen);
- else
- tprintf(", [%d]", rlen);
- }
-
- return RVAL_DECODED;
- }
-
- SYS_FUNC(accept)
- {
- return decode_sockname(tcp) | RVAL_FD;
- }
-
- SYS_FUNC(accept4)
- {
- int rc = decode_sockname(tcp);
-
- if (rc & RVAL_DECODED) {
- tprints(", ");
- printflags(sock_type_flags, tcp->u_arg[3], "SOCK_???");
- }
-
- return rc | RVAL_FD;
- }
-
- SYS_FUNC(send)
- {
- printfd(tcp, tcp->u_arg[0]);
- tprints(", ");
- decode_sockbuf(tcp, tcp->u_arg[0], tcp->u_arg[1], tcp->u_arg[2]);
- tprintf(", %" PRI_klu ", ", tcp->u_arg[2]);
- /* flags */
- printflags(msg_flags, tcp->u_arg[3], "MSG_???");
-
- return RVAL_DECODED;
- }
-
- SYS_FUNC(sendto)
- {
- printfd(tcp, tcp->u_arg[0]);
- tprints(", ");
- decode_sockbuf(tcp, tcp->u_arg[0], tcp->u_arg[1], tcp->u_arg[2]);
- tprintf(", %" PRI_klu ", ", tcp->u_arg[2]);
- /* flags */
- printflags(msg_flags, tcp->u_arg[3], "MSG_???");
- /* to address */
- const int addrlen = tcp->u_arg[5];
- tprints(", ");
- decode_sockaddr(tcp, tcp->u_arg[4], addrlen);
- /* to length */
- tprintf(", %d", addrlen);
-
- return RVAL_DECODED;
- }
-
- SYS_FUNC(recv)
- {
- if (entering(tcp)) {
- printfd(tcp, tcp->u_arg[0]);
- tprints(", ");
- } else {
- if (syserror(tcp)) {
- printaddr(tcp->u_arg[1]);
- } else {
- decode_sockbuf(tcp, tcp->u_arg[0], tcp->u_arg[1],
- MIN((kernel_ulong_t) tcp->u_rval,
- tcp->u_arg[2]));
- }
-
- tprintf(", %" PRI_klu ", ", tcp->u_arg[2]);
- printflags(msg_flags, tcp->u_arg[3], "MSG_???");
- }
- return 0;
- }
-
- SYS_FUNC(recvfrom)
- {
- int ulen, rlen;
-
- if (entering(tcp)) {
- printfd(tcp, tcp->u_arg[0]);
- tprints(", ");
- if (fetch_socklen(tcp, &ulen, tcp->u_arg[4], tcp->u_arg[5])) {
- set_tcb_priv_ulong(tcp, ulen);
- }
- } else {
- /* buf */
- if (syserror(tcp)) {
- printaddr(tcp->u_arg[1]);
- } else {
- decode_sockbuf(tcp, tcp->u_arg[0], tcp->u_arg[1],
- MIN((kernel_ulong_t) tcp->u_rval,
- tcp->u_arg[2]));
- }
- /* size */
- tprintf(", %" PRI_klu ", ", tcp->u_arg[2]);
- /* flags */
- printflags(msg_flags, tcp->u_arg[3], "MSG_???");
- tprints(", ");
-
- ulen = get_tcb_priv_ulong(tcp);
-
- if (!fetch_socklen(tcp, &rlen, tcp->u_arg[4], tcp->u_arg[5])) {
- /* from address */
- printaddr(tcp->u_arg[4]);
- tprints(", ");
- /* from length */
- printaddr(tcp->u_arg[5]);
- return 0;
- }
- if (syserror(tcp)) {
- /* from address */
- printaddr(tcp->u_arg[4]);
- /* from length */
- tprintf(", [%d]", ulen);
- return 0;
- }
- /* from address */
- decode_sockaddr(tcp, tcp->u_arg[4], ulen > rlen ? rlen : ulen);
- /* from length */
- if (ulen != rlen)
- tprintf(", [%d->%d]", ulen, rlen);
- else
- tprintf(", [%d]", rlen);
- }
- return 0;
- }
-
- SYS_FUNC(getsockname)
- {
- return decode_sockname(tcp);
- }
-
- static void
- printpair_fd(struct tcb *tcp, const int i0, const int i1)
- {
- tprints("[");
- printfd(tcp, i0);
- tprints(", ");
- printfd(tcp, i1);
- tprints("]");
- }
-
- static void
- decode_pair_fd(struct tcb *const tcp, const kernel_ulong_t addr)
- {
- int pair[2];
-
- if (umove_or_printaddr(tcp, addr, &pair))
- return;
-
- printpair_fd(tcp, pair[0], pair[1]);
- }
-
- static int
- do_pipe(struct tcb *tcp, int flags_arg)
- {
- if (exiting(tcp)) {
- decode_pair_fd(tcp, tcp->u_arg[0]);
- if (flags_arg >= 0) {
- tprints(", ");
- printflags(open_mode_flags, tcp->u_arg[flags_arg], "O_???");
- }
- }
- return 0;
- }
-
- SYS_FUNC(pipe)
- {
- #if HAVE_ARCH_GETRVAL2
- if (exiting(tcp) && !syserror(tcp))
- printpair_fd(tcp, tcp->u_rval, getrval2(tcp));
- return 0;
- #else
- return do_pipe(tcp, -1);
- #endif
- }
-
- SYS_FUNC(pipe2)
- {
- return do_pipe(tcp, 1);
- }
-
- SYS_FUNC(socketpair)
- {
- if (entering(tcp)) {
- printxval(addrfams, tcp->u_arg[0], "AF_???");
- tprints(", ");
- tprint_sock_type(tcp->u_arg[1]);
- tprintf(", %" PRI_klu, tcp->u_arg[2]);
- } else {
- tprints(", ");
- decode_pair_fd(tcp, tcp->u_arg[3]);
- }
- return 0;
- }
-
- #include "xlat/sock_options.h"
- #include "xlat/getsock_options.h"
- #include "xlat/setsock_options.h"
- #include "xlat/sock_ip_options.h"
- #include "xlat/getsock_ip_options.h"
- #include "xlat/setsock_ip_options.h"
- #include "xlat/sock_ipv6_options.h"
- #include "xlat/getsock_ipv6_options.h"
- #include "xlat/setsock_ipv6_options.h"
- #include "xlat/sock_ipx_options.h"
- #include "xlat/sock_ax25_options.h"
- #include "xlat/sock_netlink_options.h"
- #include "xlat/sock_packet_options.h"
- #include "xlat/sock_raw_options.h"
- #include "xlat/sock_sctp_options.h"
- #include "xlat/sock_tcp_options.h"
- #include "xlat/sock_udp_options.h"
- #include "xlat/sock_irda_options.h"
- #include "xlat/sock_llc_options.h"
- #include "xlat/sock_dccp_options.h"
- #include "xlat/sock_tipc_options.h"
- #include "xlat/sock_rxrpc_options.h"
- #include "xlat/sock_pppol2tp_options.h"
- #include "xlat/sock_bluetooth_options.h"
- #include "xlat/sock_pnp_options.h"
- #include "xlat/sock_rds_options.h"
- #include "xlat/sock_iucv_options.h"
- #include "xlat/sock_caif_options.h"
- #include "xlat/sock_alg_options.h"
- #include "xlat/sock_nfcllcp_options.h"
- #include "xlat/sock_kcm_options.h"
- #include "xlat/sock_tls_options.h"
- #include "xlat/sock_xdp_options.h"
-
- static void
- print_sockopt_fd_level_name(struct tcb *tcp, int fd, unsigned int level,
- unsigned int name, bool is_getsockopt)
- {
- printfd(tcp, fd);
- tprints(", ");
- printxval(socketlayers, level, "SOL_??");
- tprints(", ");
-
- switch (level) {
- case SOL_SOCKET:
- printxvals(name, "SO_???", sock_options,
- is_getsockopt ? getsock_options :
- setsock_options, NULL);
- break;
- case SOL_IP:
- printxvals(name, "IP_???", sock_ip_options,
- is_getsockopt ? getsock_ip_options :
- setsock_ip_options, NULL);
- break;
- case SOL_IPV6:
- printxvals(name, "IPV6_???", sock_ipv6_options,
- is_getsockopt ? getsock_ipv6_options :
- setsock_ipv6_options, NULL);
- break;
- case SOL_IPX:
- printxval(sock_ipx_options, name, "IPX_???");
- break;
- case SOL_AX25:
- printxval(sock_ax25_options, name, "AX25_???");
- break;
- case SOL_PACKET:
- printxval(sock_packet_options, name, "PACKET_???");
- break;
- case SOL_TCP:
- printxval(sock_tcp_options, name, "TCP_???");
- break;
- case SOL_SCTP:
- printxval(sock_sctp_options, name, "SCTP_???");
- break;
- case SOL_RAW:
- printxval(sock_raw_options, name, "RAW_???");
- break;
- case SOL_NETLINK:
- printxval(sock_netlink_options, name, "NETLINK_???");
- break;
- case SOL_UDP:
- printxval(sock_udp_options, name, "UDP_???");
- break;
- case SOL_IRDA:
- printxval(sock_irda_options, name, "IRLMP_???");
- break;
- case SOL_LLC:
- printxval(sock_llc_options, name, "LLC_OPT_???");
- break;
- case SOL_DCCP:
- printxval(sock_dccp_options, name, "DCCP_SOCKOPT_???");
- break;
- case SOL_TIPC:
- printxval(sock_tipc_options, name, "TIPC_???");
- break;
- case SOL_RXRPC:
- printxval(sock_rxrpc_options, name, "RXRPC_???");
- break;
- case SOL_PPPOL2TP:
- printxval(sock_pppol2tp_options, name, "PPPOL2TP_SO_???");
- break;
- case SOL_BLUETOOTH:
- printxval(sock_bluetooth_options, name, "BT_???");
- break;
- case SOL_PNPIPE:
- printxval(sock_pnp_options, name, "PNPIPE_???");
- break;
- case SOL_RDS:
- printxval(sock_rds_options, name, "RDS_???");
- break;
- case SOL_IUCV:
- printxval(sock_iucv_options, name, "SO_???");
- break;
- case SOL_CAIF:
- printxval(sock_caif_options, name, "CAIFSO_???");
- break;
- case SOL_ALG:
- printxval(sock_alg_options, name, "ALG_???");
- break;
- case SOL_NFC:
- printxval(sock_nfcllcp_options, name, "NFC_LLCP_???");
- break;
- case SOL_KCM:
- printxval(sock_kcm_options, name, "KCM_???");
- break;
- case SOL_TLS:
- printxval(sock_tls_options, name, "TLS_???");
- break;
- case SOL_XDP:
- printxval(sock_xdp_options, name, "XDP_???");
- break;
-
- /* Other SOL_* protocol levels still need work. */
-
- default:
- tprintf("%u", name);
- }
-
- tprints(", ");
- }
-
- static void
- print_get_linger(struct tcb *const tcp, const kernel_ulong_t addr,
- unsigned int len)
- {
- struct linger linger;
-
- /*
- * The kernel cannot return len > sizeof(linger) because struct linger
- * cannot change, but extra safety won't harm either.
- */
- if (len > sizeof(linger))
- len = sizeof(linger);
- if (umoven_or_printaddr(tcp, addr, len, &linger))
- return;
-
- if (len < sizeof(linger.l_onoff)) {
- tprints("{l_onoff=");
- print_quoted_string((void *) &linger.l_onoff,
- len, QUOTE_FORCE_HEX);
- } else {
- PRINT_FIELD_D("{", linger, l_onoff);
-
- if (len > offsetof(struct linger, l_linger)) {
- len -= offsetof(struct linger, l_linger);
- if (len < sizeof(linger.l_linger)) {
- tprints(", l_linger=");
- print_quoted_string((void *) &linger.l_linger,
- len, QUOTE_FORCE_HEX);
- } else {
- PRINT_FIELD_D(", ", linger, l_linger);
- }
- }
- }
- tprints("}");
- }
-
- static void
- print_get_ucred(struct tcb *const tcp, const kernel_ulong_t addr,
- unsigned int len)
- {
- struct ucred uc;
-
- /*
- * The kernel is very unlikely to return len > sizeof(uc)
- * because struct ucred is very unlikely to change,
- * but extra safety won't harm either.
- */
- if (len > sizeof(uc))
- len = sizeof(uc);
-
- if (umoven_or_printaddr(tcp, addr, len, &uc))
- return;
-
- if (len < sizeof(uc.pid)) {
- tprints("{pid=");
- print_quoted_string((void *) &uc.pid,
- len, QUOTE_FORCE_HEX);
- } else {
- PRINT_FIELD_D("{", uc, pid);
-
- if (len > offsetof(struct ucred, uid)) {
- len -= offsetof(struct ucred, uid);
- if (len < sizeof(uc.uid)) {
- tprints(", uid=");
- print_quoted_string((void *) &uc.uid,
- len, QUOTE_FORCE_HEX);
- } else {
- PRINT_FIELD_UID(", ", uc, uid);
-
- if (len > offsetof(struct ucred, gid) -
- offsetof(struct ucred, uid)) {
- len -= offsetof(struct ucred, gid) -
- offsetof(struct ucred, uid);
- if (len < sizeof(uc.gid)) {
- tprints(", gid=");
- print_quoted_string((void *) &uc.gid,
- len,
- QUOTE_FORCE_HEX);
- } else {
- PRINT_FIELD_UID(", ", uc, gid);
- }
- }
- }
- }
- }
- tprints("}");
- }
-
- static void
- print_get_error(struct tcb *const tcp, const kernel_ulong_t addr,
- unsigned int len)
- {
- unsigned int err;
-
- if (len > sizeof(err))
- err = sizeof(err);
-
- if (umoven_or_printaddr(tcp, addr, len, &err))
- return;
-
- tprints("[");
- print_err(err, false);
- tprints("]");
- }
-
- #ifdef PACKET_STATISTICS
- static void
- print_tpacket_stats(struct tcb *const tcp, const kernel_ulong_t addr,
- unsigned int len)
- {
- struct tp_stats {
- unsigned int tp_packets, tp_drops, tp_freeze_q_cnt;
- } stats;
-
- /*
- * The kernel may return len > sizeof(stats) if the kernel structure
- * grew as it happened when tpacket_stats_v3 was introduced.
- */
- if (len > sizeof(stats))
- len = sizeof(stats);
-
- if (umoven_or_printaddr(tcp, addr, len, &stats))
- return;
-
- if (len < sizeof(stats.tp_packets)) {
- tprints("{tp_packets=");
- print_quoted_string((void *) &stats.tp_packets,
- len, QUOTE_FORCE_HEX);
- } else {
- PRINT_FIELD_U("{", stats, tp_packets);
-
- if (len > offsetof(struct tp_stats, tp_drops)) {
- len -= offsetof(struct tp_stats, tp_drops);
- if (len < sizeof(stats.tp_drops)) {
- tprints(", tp_drops=");
- print_quoted_string((void *) &stats.tp_drops,
- len, QUOTE_FORCE_HEX);
- } else {
- PRINT_FIELD_U(", ", stats, tp_drops);
-
- if (len > offsetof(struct tp_stats, tp_freeze_q_cnt) -
- offsetof(struct tp_stats, tp_drops)) {
- len -= offsetof(struct tp_stats, tp_freeze_q_cnt) -
- offsetof(struct tp_stats, tp_drops);
- if (len < sizeof(stats.tp_freeze_q_cnt)) {
- tprints(", tp_freeze_q_cnt=");
- print_quoted_string((void *) &stats.tp_freeze_q_cnt,
- len,
- QUOTE_FORCE_HEX);
- } else {
- PRINT_FIELD_U(", ", stats, tp_freeze_q_cnt);
- }
- }
- }
- }
- }
- tprints("}");
- }
- #endif /* PACKET_STATISTICS */
-
- #include "xlat/icmpfilterflags.h"
-
- static void
- print_icmp_filter(struct tcb *const tcp, const kernel_ulong_t addr, int len)
- {
- struct icmp_filter filter = {};
-
- if (len > (int) sizeof(filter))
- len = sizeof(filter);
- else if (len <= 0) {
- printaddr(addr);
- return;
- }
-
- if (umoven_or_printaddr(tcp, addr, len, &filter))
- return;
-
- tprints("~(");
- printflags(icmpfilterflags, ~filter.data, "ICMP_???");
- tprints(")");
- }
-
- static void
- print_getsockopt(struct tcb *const tcp, const unsigned int level,
- const unsigned int name, const kernel_ulong_t addr,
- const int ulen, const int rlen)
- {
- if (ulen <= 0 || rlen <= 0) {
- /*
- * As the kernel neither accepts nor returns a negative
- * length in case of successful getsockopt syscall
- * invocation, negative values must have been forged
- * by userspace.
- */
- printaddr(addr);
- return;
- }
-
- if (addr && verbose(tcp))
- switch (level) {
- case SOL_SOCKET:
- switch (name) {
- case SO_LINGER:
- print_get_linger(tcp, addr, rlen);
- return;
- case SO_PEERCRED:
- print_get_ucred(tcp, addr, rlen);
- return;
- case SO_ATTACH_FILTER:
- /*
- * The length returned by the kernel in case of
- * successful getsockopt syscall invocation is struct
- * sock_fprog.len that has type unsigned short,
- * anything else must have been forged by userspace.
- */
- if ((unsigned short) rlen == (unsigned int) rlen)
- print_sock_fprog(tcp, addr, rlen);
- else
- printaddr(addr);
- return;
- case SO_ERROR:
- print_get_error(tcp, addr, rlen);
- return;
- }
- break;
-
- case SOL_PACKET:
- switch (name) {
- #ifdef PACKET_STATISTICS
- case PACKET_STATISTICS:
- print_tpacket_stats(tcp, addr, rlen);
- return;
- #endif
- }
- break;
-
- case SOL_RAW:
- switch (name) {
- case ICMP_FILTER:
- print_icmp_filter(tcp, addr, rlen);
- return;
- }
- break;
-
- case SOL_NETLINK:
- switch (name) {
- case NETLINK_LIST_MEMBERSHIPS: {
- uint32_t buf;
- print_array(tcp, addr, MIN(ulen, rlen) / sizeof(buf),
- &buf, sizeof(buf),
- tfetch_mem, print_uint32_array_member, 0);
- break;
- }
- default:
- printnum_int(tcp, addr, "%d");
- break;
- }
- return;
- }
-
- /* default arg printing */
-
- if (verbose(tcp)) {
- if (rlen == sizeof(int)) {
- printnum_int(tcp, addr, "%d");
- } else {
- printstrn(tcp, addr, rlen);
- }
- } else {
- printaddr(addr);
- }
- }
-
- SYS_FUNC(getsockopt)
- {
- int ulen, rlen;
-
- if (entering(tcp)) {
- print_sockopt_fd_level_name(tcp, tcp->u_arg[0],
- tcp->u_arg[1], tcp->u_arg[2], true);
-
- if (verbose(tcp) && tcp->u_arg[4]
- && umove(tcp, tcp->u_arg[4], &ulen) == 0) {
- set_tcb_priv_ulong(tcp, ulen);
- return 0;
- } else {
- printaddr(tcp->u_arg[3]);
- tprints(", ");
- printaddr(tcp->u_arg[4]);
- return RVAL_DECODED;
- }
- } else {
- ulen = get_tcb_priv_ulong(tcp);
-
- if (syserror(tcp) || umove(tcp, tcp->u_arg[4], &rlen) < 0) {
- printaddr(tcp->u_arg[3]);
- tprintf(", [%d]", ulen);
- } else {
- print_getsockopt(tcp, tcp->u_arg[1], tcp->u_arg[2],
- tcp->u_arg[3], ulen, rlen);
- if (ulen != rlen)
- tprintf(", [%d->%d]", ulen, rlen);
- else
- tprintf(", [%d]", rlen);
- }
- }
- return 0;
- }
-
- static void
- print_set_linger(struct tcb *const tcp, const kernel_ulong_t addr,
- const int len)
- {
- struct linger linger;
-
- if (len < (int) sizeof(linger)) {
- printaddr(addr);
- } else if (!umove_or_printaddr(tcp, addr, &linger)) {
- PRINT_FIELD_D("{", linger, l_onoff);
- PRINT_FIELD_D(", ", linger, l_linger);
- tprints("}");
- }
- }
-
- #ifdef IP_ADD_MEMBERSHIP
- static void
- print_mreq(struct tcb *const tcp, const kernel_ulong_t addr,
- const int len)
- {
- struct ip_mreq mreq;
-
- if (len < (int) sizeof(mreq)) {
- printaddr(addr);
- } else if (!umove_or_printaddr(tcp, addr, &mreq)) {
- PRINT_FIELD_INET_ADDR("{", mreq, imr_multiaddr, AF_INET);
- PRINT_FIELD_INET_ADDR(", ", mreq, imr_interface, AF_INET);
- tprints("}");
- }
- }
- #endif /* IP_ADD_MEMBERSHIP */
-
- #ifdef IPV6_ADD_MEMBERSHIP
- static void
- print_mreq6(struct tcb *const tcp, const kernel_ulong_t addr,
- const int len)
- {
- struct ipv6_mreq mreq;
-
- if (len < (int) sizeof(mreq)) {
- printaddr(addr);
- } else if (!umove_or_printaddr(tcp, addr, &mreq)) {
- PRINT_FIELD_INET_ADDR("{", mreq, ipv6mr_multiaddr, AF_INET6);
- PRINT_FIELD_IFINDEX(", ", mreq, ipv6mr_interface);
- tprints("}");
- }
- }
- #endif /* IPV6_ADD_MEMBERSHIP */
-
- #ifdef PACKET_RX_RING
- static void
- print_tpacket_req(struct tcb *const tcp, const kernel_ulong_t addr, const int len)
- {
- struct tpacket_req req;
-
- if (len != sizeof(req) ||
- umove(tcp, addr, &req) < 0) {
- printaddr(addr);
- } else {
- PRINT_FIELD_U("{", req, tp_block_size);
- PRINT_FIELD_U(", ", req, tp_block_nr);
- PRINT_FIELD_U(", ", req, tp_frame_size);
- PRINT_FIELD_U(", ", req, tp_frame_nr);
- tprints("}");
- }
- }
- #endif /* PACKET_RX_RING */
-
- #ifdef PACKET_ADD_MEMBERSHIP
- # include "xlat/packet_mreq_type.h"
-
- static void
- print_packet_mreq(struct tcb *const tcp, const kernel_ulong_t addr, const int len)
- {
- struct packet_mreq mreq;
-
- if (len != sizeof(mreq) ||
- umove(tcp, addr, &mreq) < 0) {
- printaddr(addr);
- } else {
- PRINT_FIELD_IFINDEX("{", mreq, mr_ifindex);
- PRINT_FIELD_XVAL(", ", mreq, mr_type, packet_mreq_type,
- "PACKET_MR_???");
- PRINT_FIELD_U(", ", mreq, mr_alen);
- PRINT_FIELD_MAC_SZ(", ", mreq, mr_address,
- (mreq.mr_alen > sizeof(mreq.mr_address)
- ? sizeof(mreq.mr_address) : mreq.mr_alen));
- tprints("}");
- }
- }
- #endif /* PACKET_ADD_MEMBERSHIP */
-
- static void
- print_setsockopt(struct tcb *const tcp, const unsigned int level,
- const unsigned int name, const kernel_ulong_t addr,
- const int len)
- {
- if (addr && verbose(tcp))
- switch (level) {
- case SOL_SOCKET:
- switch (name) {
- case SO_LINGER:
- print_set_linger(tcp, addr, len);
- return;
- case SO_ATTACH_FILTER:
- case SO_ATTACH_REUSEPORT_CBPF:
- if ((unsigned int) len == get_sock_fprog_size())
- decode_sock_fprog(tcp, addr);
- else
- printaddr(addr);
- return;
- }
- break;
-
- case SOL_IP:
- switch (name) {
- #ifdef IP_ADD_MEMBERSHIP
- case IP_ADD_MEMBERSHIP:
- case IP_DROP_MEMBERSHIP:
- print_mreq(tcp, addr, len);
- return;
- #endif /* IP_ADD_MEMBERSHIP */
- #ifdef MCAST_JOIN_GROUP
- case MCAST_JOIN_GROUP:
- case MCAST_LEAVE_GROUP:
- print_group_req(tcp, addr, len);
- return;
- #endif /* MCAST_JOIN_GROUP */
- }
- break;
-
- case SOL_IPV6:
- switch (name) {
- #ifdef IPV6_ADD_MEMBERSHIP
- case IPV6_ADD_MEMBERSHIP:
- case IPV6_DROP_MEMBERSHIP:
- # ifdef IPV6_JOIN_ANYCAST
- case IPV6_JOIN_ANYCAST:
- # endif
- # ifdef IPV6_LEAVE_ANYCAST
- case IPV6_LEAVE_ANYCAST:
- # endif
- print_mreq6(tcp, addr, len);
- return;
- #endif /* IPV6_ADD_MEMBERSHIP */
- #ifdef MCAST_JOIN_GROUP
- case MCAST_JOIN_GROUP:
- case MCAST_LEAVE_GROUP:
- print_group_req(tcp, addr, len);
- return;
- #endif /* MCAST_JOIN_GROUP */
- }
- break;
-
- case SOL_PACKET:
- switch (name) {
- #ifdef PACKET_RX_RING
- case PACKET_RX_RING:
- # ifdef PACKET_TX_RING
- case PACKET_TX_RING:
- # endif
- print_tpacket_req(tcp, addr, len);
- return;
- #endif /* PACKET_RX_RING */
- #ifdef PACKET_ADD_MEMBERSHIP
- case PACKET_ADD_MEMBERSHIP:
- case PACKET_DROP_MEMBERSHIP:
- print_packet_mreq(tcp, addr, len);
- return;
- #endif /* PACKET_ADD_MEMBERSHIP */
- }
- break;
-
- case SOL_RAW:
- switch (name) {
- case ICMP_FILTER:
- print_icmp_filter(tcp, addr, len);
- return;
- }
- break;
-
- case SOL_NETLINK:
- if (len < (int) sizeof(int))
- printaddr(addr);
- else
- printnum_int(tcp, addr, "%d");
- return;
- }
-
- /* default arg printing */
-
- if (verbose(tcp)) {
- if (len == sizeof(int)) {
- printnum_int(tcp, addr, "%d");
- } else {
- printstrn(tcp, addr, len);
- }
- } else {
- printaddr(addr);
- }
- }
-
- SYS_FUNC(setsockopt)
- {
- print_sockopt_fd_level_name(tcp, tcp->u_arg[0],
- tcp->u_arg[1], tcp->u_arg[2], false);
- print_setsockopt(tcp, tcp->u_arg[1], tcp->u_arg[2],
- tcp->u_arg[3], tcp->u_arg[4]);
- tprintf(", %d", (int) tcp->u_arg[4]);
-
- return RVAL_DECODED;
- }
|