// bslmf_ismemberobjectpointer.h                                      -*-C++-*-
#ifndef INCLUDED_BSLMF_ISMEMBEROBJECTPOINTER
#define INCLUDED_BSLMF_ISMEMBEROBJECTPOINTER

#include <bsls_ident.h>
BSLS_IDENT("$Id: $")

//@PURPOSE: Provide a compile-time check for member object pointer types.
//
//@CLASSES:
//  bsl::is_member_object_pointer: standard meta-function
//  bsl::is_member_object_pointer_v: the result value of the meta-function
//
//@SEE_ALSO: bslmf_integralconstant, bslmf_ismemberfunctionpointer
//
//@DESCRIPTION: This component defines a meta-function,
// 'bsl::is_member_object_pointer' and a template variable
// 'bsl::is_member_object_pointer_v', that represents the result value of the
// 'bsl::is_member_object_pointer' meta-function, that may be used to query
// whether a type is a pointer to non-static member object type.
//
// 'bsl::is_member_object_pointer' meets the requirements of the
// 'is_member_object_pointer' template defined in the C++11 standard
// [meta.unary.cat].
//
// Note that the template variable 'is_member_object_pointer_v' is defined in
// the C++17 standard as an inline variable.  If the current compiler supports
// the inline variable C++17 compiler feature,
// 'bsl::is_member_object_pointer_v' is defined as an 'inline constexpr bool'
// variable.  Otherwise, if the compiler supports the variable templates C++14
// compiler feature, 'bsl::is_member_object_pointer_v' is defined as a
// non-inline 'constexpr bool' variable.  See
// 'BSLS_COMPILERFEATURES_SUPPORT_INLINE_VARIABLES' and
// 'BSLS_COMPILERFEATURES_SUPPORT_VARIABLE_TEMPLATES' macros in
// bsls_compilerfeatures component for details.
//
///Usage
///-----
// In this section we show intended use of this component.
//
///Example 1: Verify Member Object Pointer Types
///- - - - - - - - - - - - - - - - - - - - - - -
// Suppose that we want to assert whether a set of types are pointers to member
// object types.
//
// First, we create a user-defined type 'MyStruct':
//..
//  struct MyStruct
//  {
//  };
//..
// Now, we create a 'typedef' for a member object pointer type:
//..
//  typedef int MyStruct::* DataMemPtr;
//..
// Finally, we instantiate the 'bsl::is_member_object_pointer' template for a
// non-member data type and the 'MyStructDataPtr' type, and assert the 'value'
// static data member of each instantiation:
//..
//  assert(false == bsl::is_member_object_pointer<int*>::value);
//  assert(true  == bsl::is_member_object_pointer<DataMemPtr>::value);
//..
// Note that if the current compiler supports the variable templates C++14
// feature then we can re-write the snippet of code above using the
// 'bsl::is_member_object_pointer_v' variable as follows:
//..
//#ifdef BSLS_COMPILERFEATURES_SUPPORT_VARIABLE_TEMPLATES
//  assert(false == bsl::is_member_object_pointer_v<int*>);
//  assert(true  == bsl::is_member_object_pointer_v<DataMemPtr>);
//#endif
//..

#include <bslscm_version.h>

#include <bslmf_integralconstant.h>
#include <bslmf_isconst.h>
#include <bslmf_isfunction.h>

#include <bsls_compilerfeatures.h>
#include <bsls_keyword.h>
#include <bsls_platform.h>

#ifndef BDE_DONT_ALLOW_TRANSITIVE_INCLUDES
#include <bslmf_ismemberfunctionpointer.h>
#endif // BDE_DONT_ALLOW_TRANSITIVE_INCLUDES

#if !defined(BSLS_PLATFORM_CMP_IBM)
namespace bsl {

                       // ===============================
                       // struct is_member_object_pointer
                       // ===============================

template <class t_TYPE>
struct is_member_object_pointer : false_type {
    // This 'struct' template implements the 'is_member_object_pointer'
    // meta-function defined in the C++11 standard [meta.unary.cat] to
    // determine if the (template parameter) 't_TYPE' is a pointer to
    // non-static data member type.  This 'struct' derives from
    // 'bsl::true_type' if the 't_TYPE' is a pointer to non-static data member
    // type, and from 'bsl::false_type' otherwise.
};

#ifdef BSLS_PLATFORM_CMP_MSVC
# pragma warning(push)
# pragma warning(disable: 4180)  // cv-qualifier has no effect on function type
#endif

template <class t_TYPE, class t_CLASS>
struct is_member_object_pointer<t_TYPE t_CLASS::*>
: is_const<const t_TYPE>::type {
    // Partial specialization relies on the principle that only object types
    // can be 'const'-qualified.  Reference-types and function-types do not
    // retain the 'const' qualifier when added in this manner, and there are
    // no 'void' class members.
};

#ifdef BSLS_PLATFORM_CMP_MSVC
# pragma warning(pop)
#endif

template <class t_TYPE, class t_CLASS>
struct is_member_object_pointer<t_TYPE t_CLASS::*const>
: is_const<const t_TYPE>::type {
    // Partial specialization relies on the principle that only object types
    // can be 'const'-qualified.  Reference-types and function-types do not
    // retain the 'const' qualifier when added in this manner, and there are
    // no 'void' class members.
};

template <class t_TYPE, class t_CLASS>
struct is_member_object_pointer<t_TYPE t_CLASS::*volatile>
: is_const<const t_TYPE>::type {
    // Partial specialization relies on the principle that only object types
    // can be 'const'-qualified.  Reference-types and function-types do not
    // retain the 'const' qualifier when added in this manner, and there are
    // no 'void' class members.
};

template <class t_TYPE, class t_CLASS>
struct is_member_object_pointer<t_TYPE t_CLASS::*const volatile>
: is_const<const t_TYPE>::type {
    // Partial specialization relies on the principle that only object types
    // can be 'const'-qualified.  Reference-types and function-types do not
    // retain the 'const' qualifier when added in this manner, and there are
    // no 'void' class members.
};

}  // close namespace bsl
#else
// The IBM xlC compiler produces parses an error when trying to add 'const' to
// a function type, so leans on the original BDE implementation of this trait.
// Note that this implementation fails on all other compilers for member
// functions with a C-style elipsis, erroneously reporting such functions as
// data members.  However, xlC appears to have sufficient compenstating bugs
// that this implementation gives the correct result in such cases too.

namespace BloombergLP {
namespace bslmf {

                       // ================================
                       // struct IsPointerToMemberData_Imp
                       // ================================

template <class t_TYPE>
struct IsPointerToMemberData_Imp : bsl::false_type {
    // This 'struct' template provides a meta-function to determine whether the
    // (template parameter) 't_TYPE' is a pointer to non-static data member
    // type.  This generic default template derives from 'bsl::false_type'.  A
    // template specialization is provided (below) that derives from
    // 'bsl::true_type'.
};

template <class t_TYPE, class t_CLASS>
struct IsPointerToMemberData_Imp<t_TYPE t_CLASS::*>
: bsl::integral_constant<bool, !bsl::is_function<t_TYPE>::value> {
    // This partial specialization of 'IsPointerToMemberData_Imp' derives from
    // 'bsl::true_type' for when the (template parameter) 't_TYPE' is a pointer
    // to non-static data member type.
};

}  // close package namespace
}  // close enterprise namespace

namespace bsl {

                       // ===============================
                       // struct is_member_object_pointer
                       // ===============================

template <class t_TYPE>
struct is_member_object_pointer
: BloombergLP::bslmf::IsPointerToMemberData_Imp<t_TYPE>::type {
    // This 'struct' template implements the 'is_member_object_pointer'
    // meta-function defined in the C++11 standard [meta.unary.cat] to
    // determine if the (template parameter) 't_TYPE' is a pointer to
    // non-static data member type.  This 'struct' derives from
    // 'bsl::true_type' if the 't_TYPE' is a pointer to non-static data member
    // type, and from 'bsl::false_type' otherwise.
};

template <class t_TYPE>
struct is_member_object_pointer<const t_TYPE>
: BloombergLP::bslmf::IsPointerToMemberData_Imp<t_TYPE>::type {
    // Partial specialization to handle 'const'-qualified pointer-to-member
    // objects.
};

template <class t_TYPE>
struct is_member_object_pointer<volatile t_TYPE>
: BloombergLP::bslmf::IsPointerToMemberData_Imp<t_TYPE>::type {
    // Partial specialization to handle 'volatile'-qualified pointer-to-member
    // objects.
};

template <class t_TYPE>
struct is_member_object_pointer<const volatile t_TYPE>
: BloombergLP::bslmf::IsPointerToMemberData_Imp<t_TYPE>::type {
    // Partial specialization to handle 'const volatile'-qualified
    // pointer-to-member objects.
};

}  // close namespace bsl
#endif

#ifdef BSLS_COMPILERFEATURES_SUPPORT_VARIABLE_TEMPLATES
namespace bsl {

template <class t_TYPE>
BSLS_KEYWORD_INLINE_VARIABLE constexpr bool is_member_object_pointer_v =
                                       is_member_object_pointer<t_TYPE>::value;
    // This template variable represents the result value of the
    // 'bsl::is_member_object_pointer' meta-function.
}  // close namespace bsl
#endif

#endif

// ----------------------------------------------------------------------------
// Copyright 2013-2018 Bloomberg Finance L.P.
//
// 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
//
//     http://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.
// ----------------------------- END-OF-FILE ----------------------------------