//===----------------------------------------------------------------------===//
//
// 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
//
//===----------------------------------------------------------------------===//
// type_traits
// is_function
#include <type_traits>
#include <cstddef> // for std::nullptr_t
#include "test_macros.h"
// NOTE: On Windows the function `test_is_function<void()>` and
// `test_is_function<void() noexcept> has the same mangled despite being
// a distinct instantiation. This causes Clang to emit an error. However
// structs do not have this problem.
template <class T>
struct test_is_function {
static_assert( std::is_function<T>::value, "");
static_assert( std::is_function<const T>::value, "");
static_assert( std::is_function<volatile T>::value, "");
static_assert( std::is_function<const volatile T>::value, "");
#if TEST_STD_VER > 14
static_assert( std::is_function_v<T>, "");
static_assert( std::is_function_v<const T>, "");
static_assert( std::is_function_v<volatile T>, "");
static_assert( std::is_function_v<const volatile T>, "");
#endif
};
template <class T>
struct test_is_not_function {
static_assert(!std::is_function<T>::value, "");
static_assert(!std::is_function<const T>::value, "");
static_assert(!std::is_function<volatile T>::value, "");
static_assert(!std::is_function<const volatile T>::value, "");
#if TEST_STD_VER > 14
static_assert(!std::is_function_v<T>, "");
static_assert(!std::is_function_v<const T>, "");
static_assert(!std::is_function_v<volatile T>, "");
static_assert(!std::is_function_v<const volatile T>, "");
#endif
};
class Empty
{
};
class NotEmpty
{
virtual ~NotEmpty();
};
union Union {};
struct bit_zero
{
int : 0;
};
class Abstract
{
virtual ~Abstract() = 0;
};
enum Enum {zero, one};
struct incomplete_type;
typedef void (*FunctionPtr)();
int main(int, char**)
{
test_is_function<void(void)>();
test_is_function<int(int)>();
test_is_function<int(int, double)>();
test_is_function<int(Abstract *)>();
test_is_function<void(...)>();
test_is_not_function<std::nullptr_t>();
test_is_not_function<void>();
test_is_not_function<int>();
test_is_not_function<int&>();
test_is_not_function<int&&>();
test_is_not_function<int*>();
test_is_not_function<double>();
test_is_not_function<char[3]>();
test_is_not_function<char[]>();
test_is_not_function<Union>();
test_is_not_function<Enum>();
test_is_not_function<FunctionPtr>(); // function pointer is not a function
test_is_not_function<Empty>();
test_is_not_function<bit_zero>();
test_is_not_function<NotEmpty>();
test_is_not_function<Abstract>();
test_is_not_function<Abstract*>();
test_is_not_function<incomplete_type>();
#if TEST_STD_VER >= 11
test_is_function<void() noexcept>();
test_is_function<void() const && noexcept>();
#endif
return 0;
}