Blame lib/xalloc-oversized.h

Packit 709fb3
/* xalloc-oversized.h -- memory allocation size checking
Packit 709fb3
Packit 709fb3
   Copyright (C) 1990-2000, 2003-2004, 2006-2017 Free Software Foundation, Inc.
Packit 709fb3
Packit 709fb3
   This program is free software: you can redistribute it and/or modify
Packit 709fb3
   it under the terms of the GNU General Public License as published by
Packit 709fb3
   the Free Software Foundation; either version 3 of the License, or
Packit 709fb3
   (at your option) any later version.
Packit 709fb3
Packit 709fb3
   This program is distributed in the hope that it will be useful,
Packit 709fb3
   but WITHOUT ANY WARRANTY; without even the implied warranty of
Packit 709fb3
   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
Packit 709fb3
   GNU General Public License for more details.
Packit 709fb3
Packit 709fb3
   You should have received a copy of the GNU General Public License
Packit 709fb3
   along with this program.  If not, see <http://www.gnu.org/licenses/>.  */
Packit 709fb3
Packit 709fb3
#ifndef XALLOC_OVERSIZED_H_
Packit 709fb3
#define XALLOC_OVERSIZED_H_
Packit 709fb3
Packit 709fb3
#include <stddef.h>
Packit 709fb3
#include <stdint.h>
Packit 709fb3
Packit 709fb3
/* True if N * S would overflow in a size_t calculation,
Packit 709fb3
   or would generate a value larger than PTRDIFF_MAX.
Packit 709fb3
   This expands to a constant expression if N and S are both constants.
Packit 709fb3
   By gnulib convention, SIZE_MAX represents overflow in size
Packit 709fb3
   calculations, so the conservative size_t-based dividend to use here
Packit 709fb3
   is SIZE_MAX - 1.  */
Packit 709fb3
#define __xalloc_oversized(n, s) \
Packit 709fb3
  ((size_t) (PTRDIFF_MAX < SIZE_MAX ? PTRDIFF_MAX : SIZE_MAX - 1) / (s) < (n))
Packit 709fb3
Packit 709fb3
#if PTRDIFF_MAX < SIZE_MAX
Packit 709fb3
typedef ptrdiff_t __xalloc_count_type;
Packit 709fb3
#else
Packit 709fb3
typedef size_t __xalloc_count_type;
Packit 709fb3
#endif
Packit 709fb3
Packit 709fb3
/* Return 1 if an array of N objects, each of size S, cannot exist
Packit 709fb3
   reliably due to size or ptrdiff_t arithmetic overflow.  S must be
Packit 709fb3
   positive and N must be nonnegative.  This is a macro, not a
Packit 709fb3
   function, so that it works correctly even when SIZE_MAX < N.  */
Packit 709fb3
Packit 709fb3
#if 7 <= __GNUC__
Packit 709fb3
# define xalloc_oversized(n, s) \
Packit 709fb3
   __builtin_mul_overflow_p (n, s, (__xalloc_count_type) 1)
Packit 709fb3
#elif 5 <= __GNUC__ && !defined __ICC && !__STRICT_ANSI__
Packit 709fb3
# define xalloc_oversized(n, s) \
Packit 709fb3
   (__builtin_constant_p (n) && __builtin_constant_p (s) \
Packit 709fb3
    ? __xalloc_oversized (n, s) \
Packit 709fb3
    : ({ __xalloc_count_type __xalloc_count; \
Packit 709fb3
         __builtin_mul_overflow (n, s, &__xalloc_count); }))
Packit 709fb3
Packit 709fb3
/* Other compilers use integer division; this may be slower but is
Packit 709fb3
   more portable.  */
Packit 709fb3
#else
Packit 709fb3
# define xalloc_oversized(n, s) __xalloc_oversized (n, s)
Packit 709fb3
#endif
Packit 709fb3
Packit 709fb3
#endif /* !XALLOC_OVERSIZED_H_ */