| //===-- wrappers_c_checks.h -------------------------------------*- C++ -*-===// |
| // |
| // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. |
| // See https://llvm.org/LICENSE.txt for license information. |
| // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception |
| // |
| //===----------------------------------------------------------------------===// |
| |
| #ifndef SCUDO_CHECKS_H_ |
| #define SCUDO_CHECKS_H_ |
| |
| #include "common.h" |
| |
| #include <errno.h> |
| |
| #ifndef __has_builtin |
| #define __has_builtin(X) 0 |
| #endif |
| |
| namespace scudo { |
| |
| // A common errno setting logic shared by almost all Scudo C wrappers. |
| inline void *setErrnoOnNull(void *Ptr) { |
| if (UNLIKELY(!Ptr)) |
| errno = ENOMEM; |
| return Ptr; |
| } |
| |
| // Checks return true on failure. |
| |
| // Checks aligned_alloc() parameters, verifies that the alignment is a power of |
| // two and that the size is a multiple of alignment. |
| inline bool checkAlignedAllocAlignmentAndSize(uptr Alignment, uptr Size) { |
| return !isPowerOfTwo(Alignment) || !isAligned(Size, Alignment); |
| } |
| |
| // Checks posix_memalign() parameters, verifies that alignment is a power of two |
| // and a multiple of sizeof(void *). |
| inline bool checkPosixMemalignAlignment(uptr Alignment) { |
| return !isPowerOfTwo(Alignment) || !isAligned(Alignment, sizeof(void *)); |
| } |
| |
| // Returns true if calloc(Size, N) overflows on Size*N calculation. Use a |
| // builtin supported by recent clang & GCC if it exists, otherwise fallback to a |
| // costly division. |
| inline bool checkForCallocOverflow(uptr Size, uptr N, uptr *Product) { |
| #if __has_builtin(__builtin_umull_overflow) && (SCUDO_WORDSIZE == 64U) |
| return __builtin_umull_overflow(Size, N, |
| reinterpret_cast<unsigned long *>(Product)); |
| #elif __has_builtin(__builtin_umul_overflow) && (SCUDO_WORDSIZE == 32U) |
| // On, e.g. armv7, uptr/uintptr_t may be defined as unsigned long |
| return __builtin_umul_overflow(Size, N, |
| reinterpret_cast<unsigned int *>(Product)); |
| #else |
| *Product = Size * N; |
| if (!Size) |
| return false; |
| return (*Product / Size) != N; |
| #endif |
| } |
| |
| // Returns true if the size passed to pvalloc overflows when rounded to the next |
| // multiple of PageSize. |
| inline bool checkForPvallocOverflow(uptr Size, uptr PageSize) { |
| return roundUp(Size, PageSize) < Size; |
| } |
| |
| } // namespace scudo |
| |
| #endif // SCUDO_CHECKS_H_ |