blob: 790ec909643e88c6cf391f7d1ae3660cec758945 [file] [log] [blame]
// Copyright 2025 The Abseil Authors
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// https://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
#ifndef ABSL_BASE_INTERNAL_NULLABILITY_TRAITS_H_
#define ABSL_BASE_INTERNAL_NULLABILITY_TRAITS_H_
#include <type_traits>
#include "absl/base/config.h"
#include "absl/base/nullability.h"
namespace absl {
ABSL_NAMESPACE_BEGIN
namespace base_internal {
// `value` is true if the type `T` is compatible with nullability annotations
// (is a raw pointer, a smart pointer, or marked with
// ABSL_NULLABILITY_COMPATIBLE). Prefer to use the higher-level
// `AddNonnullIfCompatible` if that is sufficient.
//
// NOTE: This should not be used to detect if the compiler is Clang (since
// Clang is the only compiler that supports nullability annotations).
#if defined(__clang__) && !defined(__OBJC__) && \
ABSL_HAVE_FEATURE(nullability_on_classes)
template <class T, class = void>
struct IsNullabilityCompatibleType {
constexpr static bool value = false;
};
template <class T>
struct IsNullabilityCompatibleType<T, std::void_t<absl_nullable T>> {
constexpr static bool value = true;
};
#else
// False when absl_nullable is a no-op (for non-Clang compilers or Objective-C.)
template <class T, class = void>
struct IsNullabilityCompatibleType {
constexpr static bool value = false;
};
#endif
// A trait to add `absl_nonnull` to a type if it is compatible with nullability
// annotations.
template <typename T, bool ShouldAdd = IsNullabilityCompatibleType<T>::value>
struct AddNonnullIfCompatible;
template <typename T>
struct AddNonnullIfCompatible<T, false> {
using type = T;
};
template <typename T>
struct AddNonnullIfCompatible<T, true> {
using type = absl_nonnull T;
};
} // namespace base_internal
ABSL_NAMESPACE_END
} // namespace absl
#endif // ABSL_BASE_INTERNAL_NULLABILITY_TRAITS_H_