346e8002b0
netlink tests happen to access memory located before the tail_malloc'ed pointers, a practice that doesn't go well with the latest compilers because tail_malloc is marked with ATTRIBUTE_MALLOC. For example, glibc in -D_FORTIFY_SOURCE=2 mode and gcc 8 with -Warray-bounds enabled complain about negative offsets out of bounds. Fix this issue by introducing midtail_alloc. * tests/tests.h (midtail_alloc): New macro. * tests/netlink_crypto.c: Use it instead of tail_malloc for nlh0 allocation. * tests/netlink_netfilter.c: Likewise. * tests/netlink_protocol.c: Likewise. * tests/netlink_route.c: Likewise. * tests/netlink_selinux.c: Likewise. * tests/netlink_sock_diag.c: Likewise. * tests/nlattr_br_port_msg.c: Likewise. * tests/nlattr_crypto_user_alg.c: Likewise. * tests/nlattr_dcbmsg.c: Likewise. * tests/nlattr_fib_rule_hdr.c: Likewise. * tests/nlattr_ifaddrlblmsg.c: Likewise. * tests/nlattr_ifaddrmsg.c: Likewise. * tests/nlattr_ifinfomsg.c: Likewise. * tests/nlattr_ifla_brport.c: Likewise. * tests/nlattr_ifla_port.c: Likewise. * tests/nlattr_ifla_xdp.c: Likewise. * tests/nlattr_inet_diag_msg.c: Likewise. * tests/nlattr_inet_diag_req_compat.c: Likewise. * tests/nlattr_inet_diag_req_v2.c: Likewise. * tests/nlattr_mdba_mdb_entry.c: Likewise. * tests/nlattr_mdba_router_port.c: Likewise. * tests/nlattr_ndmsg.c: Likewise. * tests/nlattr_ndtmsg.c: Likewise. * tests/nlattr_netconfmsg.c: Likewise. * tests/nlattr_netlink_diag_msg.c: Likewise. * tests/nlattr_nlmsgerr.c: Likewise. * tests/nlattr_packet_diag_msg.c: Likewise. * tests/nlattr_rtgenmsg.c: Likewise. * tests/nlattr_rtmsg.c: Likewise. * tests/nlattr_smc_diag_msg.c: Likewise. * tests/nlattr_tc_stats.c: Likewise. * tests/nlattr_tca_stab.c: Likewise. * tests/nlattr_tcamsg.c: Likewise. * tests/nlattr_tcmsg.c: Likewise. * tests/nlattr_unix_diag_msg.c: Likewise. Co-Authored-by: Dmitry V. Levin <ldv@altlinux.org>
308 lines
10 KiB
C
308 lines
10 KiB
C
/*
|
|
* Copyright (c) 2016 Dmitry V. Levin <ldv@altlinux.org>
|
|
* Copyright (c) 2016-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_TESTS_H
|
|
#define STRACE_TESTS_H
|
|
|
|
#ifdef HAVE_CONFIG_H
|
|
# include "config.h"
|
|
#endif
|
|
|
|
#ifdef TESTS_SIZEOF_KERNEL_LONG_T
|
|
# undef SIZEOF_KERNEL_LONG_T
|
|
# define SIZEOF_KERNEL_LONG_T TESTS_SIZEOF_KERNEL_LONG_T
|
|
#endif
|
|
|
|
#ifdef TESTS_SIZEOF_LONG
|
|
# undef SIZEOF_LONG
|
|
# define SIZEOF_LONG TESTS_SIZEOF_LONG
|
|
#endif
|
|
|
|
#include <stdbool.h>
|
|
#include <sys/types.h>
|
|
#include "kernel_types.h"
|
|
#include "gcc_compat.h"
|
|
#include "macros.h"
|
|
|
|
/*
|
|
* The printf-like function to use in header files
|
|
* shared between strace and its tests.
|
|
*/
|
|
#ifndef STRACE_PRINTF
|
|
# define STRACE_PRINTF printf
|
|
#endif
|
|
|
|
/* Tests of "strace -v" are expected to define VERBOSE to 1. */
|
|
#ifndef VERBOSE
|
|
# define VERBOSE 0
|
|
#endif
|
|
|
|
/* xlat verbosity defaults */
|
|
#ifndef XLAT_RAW
|
|
# define XLAT_RAW 0
|
|
#endif
|
|
#ifndef XLAT_VERBOSE
|
|
# define XLAT_VERBOSE 0
|
|
#endif
|
|
|
|
#ifndef DEFAULT_STRLEN
|
|
/* Default maximum # of bytes printed in printstr et al. */
|
|
# define DEFAULT_STRLEN 32
|
|
#endif
|
|
|
|
/* Cached sysconf(_SC_PAGESIZE). */
|
|
size_t get_page_size(void);
|
|
|
|
/* The size of kernel's sigset_t. */
|
|
unsigned int get_sigset_size(void);
|
|
|
|
/* Print message and strerror(errno) to stderr, then exit(1). */
|
|
void perror_msg_and_fail(const char *, ...)
|
|
ATTRIBUTE_FORMAT((printf, 1, 2)) ATTRIBUTE_NORETURN;
|
|
/* Print message to stderr, then exit(1). */
|
|
void error_msg_and_fail(const char *, ...)
|
|
ATTRIBUTE_FORMAT((printf, 1, 2)) ATTRIBUTE_NORETURN;
|
|
/* Print message to stderr, then exit(77). */
|
|
void error_msg_and_skip(const char *, ...)
|
|
ATTRIBUTE_FORMAT((printf, 1, 2)) ATTRIBUTE_NORETURN;
|
|
/* Print message and strerror(errno) to stderr, then exit(77). */
|
|
void perror_msg_and_skip(const char *, ...)
|
|
ATTRIBUTE_FORMAT((printf, 1, 2)) ATTRIBUTE_NORETURN;
|
|
|
|
#ifndef perror_msg_and_fail
|
|
# define perror_msg_and_fail(fmt_, ...) \
|
|
perror_msg_and_fail("%s:%d: " fmt_, __FILE__, __LINE__, ##__VA_ARGS__)
|
|
#endif
|
|
#ifndef perror_msg_and_fail
|
|
# define error_msg_and_fail(fmt_, ...) \
|
|
error_msg_and_fail("%s:%d: " fmt_, __FILE__, __LINE__, ##__VA_ARGS__)
|
|
#endif
|
|
|
|
/* Stat the specified file and skip the test if the stat call failed. */
|
|
void skip_if_unavailable(const char *);
|
|
|
|
/*
|
|
* Allocate memory that ends on the page boundary.
|
|
* Pages allocated by this call are preceded by an unmapped page
|
|
* and followed also by an unmapped page.
|
|
*/
|
|
void *tail_alloc(const size_t)
|
|
ATTRIBUTE_MALLOC ATTRIBUTE_ALLOC_SIZE((1));
|
|
/* Allocate memory using tail_alloc, then memcpy. */
|
|
void *tail_memdup(const void *, const size_t)
|
|
ATTRIBUTE_MALLOC ATTRIBUTE_ALLOC_SIZE((2));
|
|
|
|
#define midtail_alloc(after_, before_) \
|
|
((void *) ((char *) tail_alloc(((before_) + (after_))) + (before_)))
|
|
|
|
/*
|
|
* Allocate an object of the specified type at the end
|
|
* of a mapped memory region.
|
|
* Assign its address to the specified constant pointer.
|
|
*/
|
|
#define TAIL_ALLOC_OBJECT_CONST_PTR(type_name, type_ptr) \
|
|
type_name *const type_ptr = tail_alloc(sizeof(*type_ptr))
|
|
|
|
/*
|
|
* Allocate an object of the specified type at the end
|
|
* of a mapped memory region.
|
|
* Assign its address to the specified variable pointer.
|
|
*/
|
|
#define TAIL_ALLOC_OBJECT_VAR_PTR(type_name, type_ptr) \
|
|
type_name *type_ptr = tail_alloc(sizeof(*type_ptr))
|
|
|
|
/*
|
|
* Fill memory (pointed by ptr, having size bytes) with different bytes (with
|
|
* values starting with start and resetting every period) in order to catch
|
|
* sign, byte order and/or alignment errors.
|
|
*/
|
|
void fill_memory_ex(void *ptr, size_t size, unsigned char start,
|
|
unsigned char period);
|
|
/* Shortcut for fill_memory_ex(ptr, size, 0x80, 0x80) */
|
|
void fill_memory(void *ptr, size_t size);
|
|
|
|
/* Close stdin, move stdout to a non-standard descriptor, and print. */
|
|
void tprintf(const char *, ...)
|
|
ATTRIBUTE_FORMAT((printf, 1, 2));
|
|
|
|
/* Make a hexdump copy of C string */
|
|
const char *hexdump_strdup(const char *);
|
|
|
|
/* Make a hexdump copy of memory */
|
|
const char *hexdump_memdup(const char *, size_t);
|
|
|
|
/* Make a hexquoted copy of a string */
|
|
const char *hexquote_strndup(const char *, size_t);
|
|
|
|
/* Return inode number of socket descriptor. */
|
|
unsigned long inode_of_sockfd(int);
|
|
|
|
/* Print string in a quoted form with optional escape characters. */
|
|
void print_quoted_string_ex(const char *, bool quote, const char *escape_str);
|
|
|
|
/* Print string in a quoted form. */
|
|
void print_quoted_string(const char *);
|
|
|
|
/*
|
|
* Print a NUL-terminated string `str' of length up to `size' - 1
|
|
* in a quoted form.
|
|
*/
|
|
void print_quoted_cstring(const char *str, size_t size);
|
|
|
|
/* Print memory in a quoted form with optional escape characters. */
|
|
void print_quoted_memory_ex(const void *, size_t, bool quote,
|
|
const char *escape_chars);
|
|
|
|
/* Print memory in a quoted form. */
|
|
void print_quoted_memory(const void *, size_t);
|
|
|
|
/* Print memory in a hexquoted form. */
|
|
void print_quoted_hex(const void *, size_t);
|
|
|
|
/* Print time_t and nanoseconds in symbolic format. */
|
|
void print_time_t_nsec(time_t, unsigned long long, int);
|
|
|
|
/* Print time_t and microseconds in symbolic format. */
|
|
void print_time_t_usec(time_t, unsigned long long, int);
|
|
|
|
/* Read an int from the file. */
|
|
int read_int_from_file(const char *, int *);
|
|
|
|
/* Check whether given uid matches kernel overflowuid. */
|
|
void check_overflowuid(const int);
|
|
|
|
/* Check whether given gid matches kernel overflowgid. */
|
|
void check_overflowgid(const int);
|
|
|
|
/* Translate errno to its name. */
|
|
const char *errno2name(void);
|
|
|
|
/* Translate signal number to its name. */
|
|
const char *signal2name(int);
|
|
|
|
/* Print return code and, in case return code is -1, errno information. */
|
|
const char *sprintrc(long rc);
|
|
/* sprintrc variant suitable for usage as part of grep pattern. */
|
|
const char *sprintrc_grep(long rc);
|
|
|
|
struct xlat;
|
|
|
|
/* Print flags in symbolic form according to xlat table. */
|
|
int printflags(const struct xlat *, const unsigned long long, const char *);
|
|
|
|
/* Print constant in symbolic form according to xlat table. */
|
|
int printxval(const struct xlat *, const unsigned long long, const char *);
|
|
|
|
/* Invoke a socket syscall, either directly or via __NR_socketcall. */
|
|
int socketcall(const int nr, const int call,
|
|
long a1, long a2, long a3, long a4, long a5);
|
|
|
|
/* Wrappers for recvmmsg and sendmmsg syscalls. */
|
|
struct mmsghdr;
|
|
struct timespec;
|
|
int recv_mmsg(int, struct mmsghdr *, unsigned int, unsigned int, struct timespec *);
|
|
int send_mmsg(int, struct mmsghdr *, unsigned int, unsigned int);
|
|
|
|
/* Create a netlink socket. */
|
|
int create_nl_socket_ext(int proto, const char *name);
|
|
#define create_nl_socket(proto) create_nl_socket_ext((proto), #proto)
|
|
|
|
/* Create a pipe with maximized descriptor numbers. */
|
|
void pipe_maxfd(int pipefd[2]);
|
|
|
|
/* if_nametoindex("lo") */
|
|
unsigned int ifindex_lo(void);
|
|
|
|
#ifdef HAVE_IF_INDEXTONAME
|
|
# define IFINDEX_LO_STR "if_nametoindex(\"lo\")"
|
|
#else
|
|
# define IFINDEX_LO_STR "1"
|
|
#endif
|
|
|
|
#define F8ILL_KULONG_SUPPORTED (sizeof(void *) < sizeof(kernel_ulong_t))
|
|
#define F8ILL_KULONG_MASK ((kernel_ulong_t) 0xffffffff00000000ULL)
|
|
|
|
/*
|
|
* For 64-bit kernel_ulong_t and 32-bit pointer,
|
|
* return a kernel_ulong_t value by filling higher bits.
|
|
* For other architertures, return the original pointer.
|
|
*/
|
|
static inline kernel_ulong_t
|
|
f8ill_ptr_to_kulong(const void *const ptr)
|
|
{
|
|
const unsigned long uptr = (unsigned long) ptr;
|
|
return F8ILL_KULONG_SUPPORTED
|
|
? F8ILL_KULONG_MASK | uptr : (kernel_ulong_t) uptr;
|
|
}
|
|
|
|
# define LENGTH_OF(arg) ((unsigned int) sizeof(arg) - 1)
|
|
|
|
/* Zero-extend a signed integer type to unsigned long long. */
|
|
#define zero_extend_signed_to_ull(v) \
|
|
(sizeof(v) == sizeof(char) ? (unsigned long long) (unsigned char) (v) : \
|
|
sizeof(v) == sizeof(short) ? (unsigned long long) (unsigned short) (v) : \
|
|
sizeof(v) == sizeof(int) ? (unsigned long long) (unsigned int) (v) : \
|
|
sizeof(v) == sizeof(long) ? (unsigned long long) (unsigned long) (v) : \
|
|
(unsigned long long) (v))
|
|
|
|
/* Sign-extend an unsigned integer type to long long. */
|
|
#define sign_extend_unsigned_to_ll(v) \
|
|
(sizeof(v) == sizeof(char) ? (long long) (char) (v) : \
|
|
sizeof(v) == sizeof(short) ? (long long) (short) (v) : \
|
|
sizeof(v) == sizeof(int) ? (long long) (int) (v) : \
|
|
sizeof(v) == sizeof(long) ? (long long) (long) (v) : \
|
|
(long long) (v))
|
|
|
|
#define SKIP_MAIN_UNDEFINED(arg) \
|
|
int main(void) { error_msg_and_skip("undefined: %s", arg); }
|
|
|
|
#if WORDS_BIGENDIAN
|
|
# define LL_PAIR(HI, LO) (HI), (LO)
|
|
#else
|
|
# define LL_PAIR(HI, LO) (LO), (HI)
|
|
#endif
|
|
#define LL_VAL_TO_PAIR(llval) LL_PAIR((long) ((llval) >> 32), (long) (llval))
|
|
|
|
#define _STR(_arg) #_arg
|
|
#define ARG_STR(_arg) (_arg), #_arg
|
|
#define ARG_ULL_STR(_arg) _arg##ULL, #_arg
|
|
|
|
/*
|
|
* Assign an object of type DEST_TYPE at address DEST_ADDR
|
|
* using memcpy to avoid potential unaligned access.
|
|
*/
|
|
#define SET_STRUCT(DEST_TYPE, DEST_ADDR, ...) \
|
|
do { \
|
|
DEST_TYPE dest_type_tmp_var = { __VA_ARGS__ }; \
|
|
memcpy(DEST_ADDR, &dest_type_tmp_var, sizeof(dest_type_tmp_var)); \
|
|
} while (0)
|
|
|
|
#define NLMSG_ATTR(nlh, hdrlen) ((void *)(nlh) + NLMSG_SPACE(hdrlen))
|
|
|
|
#endif /* !STRACE_TESTS_H */
|