/* * Copyright (c) 2016-2017 Dmitry V. Levin * 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. */ #ifndef STRACE_PRINT_FIELDS_H #define STRACE_PRINT_FIELDS_H /* * The printf-like function to use in header files * shared between strace and its tests. */ #ifndef STRACE_PRINTF # define STRACE_PRINTF tprintf #endif #define PRINT_FIELD_D(prefix_, where_, field_) \ STRACE_PRINTF("%s%s=%lld", (prefix_), #field_, \ sign_extend_unsigned_to_ll((where_).field_)) #define PRINT_FIELD_U(prefix_, where_, field_) \ STRACE_PRINTF("%s%s=%llu", (prefix_), #field_, \ zero_extend_signed_to_ull((where_).field_)) #define PRINT_FIELD_U_CAST(prefix_, where_, field_, type_) \ STRACE_PRINTF("%s%s=%llu", (prefix_), #field_, \ zero_extend_signed_to_ull((type_) (where_).field_)) #define PRINT_FIELD_X(prefix_, where_, field_) \ STRACE_PRINTF("%s%s=%#llx", (prefix_), #field_, \ zero_extend_signed_to_ull((where_).field_)) #define PRINT_FIELD_ADDR(prefix_, where_, field_) \ do { \ STRACE_PRINTF("%s%s=", (prefix_), #field_); \ printaddr((where_).field_); \ } while (0) #define PRINT_FIELD_ADDR64(prefix_, where_, field_) \ do { \ STRACE_PRINTF("%s%s=", (prefix_), #field_); \ printaddr64((where_).field_); \ } while (0) #define PRINT_FIELD_0X(prefix_, where_, field_) \ STRACE_PRINTF("%s%s=%#0*llx", (prefix_), #field_, \ (int) sizeof((where_).field_) * 2, \ zero_extend_signed_to_ull((where_).field_)) #define PRINT_FIELD_COOKIE(prefix_, where_, field_) \ STRACE_PRINTF("%s%s=[%llu, %llu]", (prefix_), #field_, \ zero_extend_signed_to_ull((where_).field_[0]), \ zero_extend_signed_to_ull((where_).field_[1])) #define PRINT_FIELD_FLAGS(prefix_, where_, field_, xlat_, dflt_) \ do { \ STRACE_PRINTF("%s%s=", (prefix_), #field_); \ printflags64((xlat_), \ zero_extend_signed_to_ull((where_).field_),\ (dflt_)); \ } while (0) #define PRINT_FIELD_XVAL(prefix_, where_, field_, xlat_, dflt_) \ do { \ STRACE_PRINTF("%s%s=", (prefix_), #field_); \ printxval64((xlat_), \ zero_extend_signed_to_ull((where_).field_), \ (dflt_)); \ } while (0) #define PRINT_FIELD_XVAL_U(prefix_, where_, field_, xlat_, dflt_) \ do { \ STRACE_PRINTF("%s%s=", (prefix_), #field_); \ printxvals_ex(zero_extend_signed_to_ull((where_).field_), \ (dflt_), XLAT_STYLE_FMT_U, \ (xlat_), NULL); \ } while (0) #define PRINT_FIELD_XVAL_SORTED_SIZED(prefix_, where_, field_, xlat_, \ xlat_size_, dflt_) \ do { \ STRACE_PRINTF("%s%s=", (prefix_), #field_); \ printxval_searchn((xlat_), (xlat_size_), \ zero_extend_signed_to_ull((where_).field_), \ (dflt_)); \ } while (0) #define PRINT_FIELD_XVAL_INDEX(prefix_, where_, field_, xlat_, dflt_) \ do { \ STRACE_PRINTF("%s%s=", (prefix_), #field_); \ printxval_index((xlat_), \ zero_extend_signed_to_ull((where_).field_), \ (dflt_)); \ } while (0) /* * Generic "ID" printing. ID is considered unsigned except for the special value * of -1. */ #define PRINT_FIELD_ID(prefix_, where_, field_) \ do { \ if (sign_extend_unsigned_to_ll((where_).field_) == -1LL) \ STRACE_PRINTF("%s%s=-1", (prefix_), #field_); \ else \ STRACE_PRINTF("%s%s=%llu", (prefix_), #field_, \ zero_extend_signed_to_ull((where_).field_)); \ } while (0) #define PRINT_FIELD_UID PRINT_FIELD_ID #define PRINT_FIELD_U64(prefix_, where_, field_) \ do { \ STRACE_PRINTF("%s%s=", (prefix_), #field_); \ if (zero_extend_signed_to_ull((where_).field_) == UINT64_MAX) \ print_xlat_u(UINT64_MAX); \ else \ STRACE_PRINTF("%llu", \ zero_extend_signed_to_ull((where_).field_)); \ } while (0) #define PRINT_FIELD_STRING(prefix_, where_, field_, len_, style_) \ do { \ STRACE_PRINTF("%s%s=", (prefix_), #field_); \ print_quoted_string((const char *)(where_).field_, \ (len_), (style_)); \ } while (0) #define PRINT_FIELD_CSTRING(prefix_, where_, field_) \ do { \ STRACE_PRINTF("%s%s=", (prefix_), #field_); \ print_quoted_cstring((const char *) (where_).field_, \ sizeof((where_).field_) + \ MUST_BE_ARRAY((where_).field_)); \ } while (0) #define PRINT_FIELD_CSTRING_SZ(prefix_, where_, field_, size_) \ do { \ STRACE_PRINTF("%s%s=", (prefix_), #field_); \ print_quoted_cstring((const char *) (where_).field_, \ (size_)); \ } while (0) #define PRINT_FIELD_HEX_ARRAY(prefix_, where_, field_) \ do { \ STRACE_PRINTF("%s%s=", (prefix_), #field_); \ print_quoted_string((const char *)(where_).field_, \ sizeof((where_).field_) + \ MUST_BE_ARRAY((where_).field_), \ QUOTE_FORCE_HEX); \ } while (0) #define PRINT_FIELD_INET_ADDR(prefix_, where_, field_, af_) \ do { \ STRACE_PRINTF(prefix_); \ print_inet_addr((af_), &(where_).field_, \ sizeof((where_).field_), #field_); \ } while (0) #define PRINT_FIELD_INET4_ADDR(prefix_, where_, field_) \ STRACE_PRINTF("%s%s=inet_addr(\"%s\")", (prefix_), #field_, \ inet_ntoa((where_).field_)) #define PRINT_FIELD_NET_PORT(prefix_, where_, field_) \ STRACE_PRINTF("%s%s=htons(%u)", (prefix_), #field_, \ ntohs((where_).field_)) #define PRINT_FIELD_IFINDEX(prefix_, where_, field_) \ do { \ STRACE_PRINTF("%s%s=", (prefix_), #field_); \ print_ifindex((where_).field_); \ } while (0) #define PRINT_FIELD_SOCKADDR(prefix_, where_, field_) \ do { \ STRACE_PRINTF("%s%s=", (prefix_), #field_); \ print_sockaddr(&(where_).field_, \ sizeof((where_).field_)); \ } while (0) #define PRINT_FIELD_DEV(prefix_, where_, field_) \ do { \ STRACE_PRINTF("%s%s=", (prefix_), #field_); \ print_dev_t((where_).field_); \ } while (0) #define PRINT_FIELD_PTR(prefix_, where_, field_) \ do { \ STRACE_PRINTF("%s%s=", (prefix_), #field_); \ printaddr((mpers_ptr_t) (where_).field_); \ } while (0) #define PRINT_FIELD_FD(prefix_, where_, field_, tcp_) \ do { \ STRACE_PRINTF("%s%s=", (prefix_), #field_); \ printfd((tcp_), (where_).field_); \ } while (0) #define PRINT_FIELD_STRN(prefix_, where_, field_, len_, tcp_) \ do { \ STRACE_PRINTF("%s%s=", (prefix_), #field_); \ printstrn((tcp_), (where_).field_, (len_)); \ } while (0) #define PRINT_FIELD_STR(prefix_, where_, field_, tcp_) \ do { \ STRACE_PRINTF("%s%s=", (prefix_), #field_); \ printstr((tcp_), (where_).field_); \ } while (0) #define PRINT_FIELD_PATH(prefix_, where_, field_, tcp_) \ do { \ STRACE_PRINTF("%s%s=", (prefix_), #field_); \ printpath((tcp_), (where_).field_); \ } while (0) #define PRINT_FIELD_MAC(prefix_, where_, field_) \ PRINT_FIELD_MAC_SZ((prefix_), (where_), field_, \ ARRAY_SIZE((where_).field_)) #define PRINT_FIELD_MAC_SZ(prefix_, where_, field_, size_) \ do { \ static_assert(sizeof(((where_).field_)[0]) == 1, \ "MAC address is not a byte array"); \ STRACE_PRINTF("%s%s=", (prefix_), #field_); \ print_mac_addr("", (const uint8_t *) ((where_).field_), \ (size_)); \ } while (0) #endif /* !STRACE_PRINT_FIELDS_H */