Overhaul and separate nullptr_t tests to pass with C++03.

The standard requires that nullptr_t can be reinterpret_cast to an integral type
at least the size of nullptr_t. There is no way to emulate this conversion in
the C++03 nullptr_t implementation. The test for this conversion has been moved
to a new test and marked XFAIL with c++03.

This recommits what was originally r222296.

llvm-svn: 222318
This commit is contained in:
Eric Fiselier 2014-11-19 05:49:03 +00:00
parent 3c8c46efd7
commit 1eb8220aff
3 changed files with 95 additions and 34 deletions

View File

@ -18,42 +18,62 @@ struct A
A(std::nullptr_t) {}
};
template <class T>
void test_conversions()
{
{
T p = 0;
assert(p == nullptr);
}
{
T p = nullptr;
assert(p == nullptr);
assert(nullptr == p);
assert(!(p != nullptr));
assert(!(nullptr != p));
}
}
template <class T>
void test_comparisons()
{
T p = nullptr;
assert(p == nullptr);
assert(p <= nullptr);
assert(p >= nullptr);
assert(!(p != nullptr));
assert(!(p < nullptr));
assert(!(p > nullptr));
assert(nullptr == p);
assert(nullptr <= p);
assert(nullptr >= p);
assert(!(nullptr != p));
assert(!(nullptr < p));
assert(!(nullptr > p));
}
int main()
{
static_assert(sizeof(std::nullptr_t) == sizeof(void*),
"sizeof(std::nullptr_t) == sizeof(void*)");
A* p = 0;
assert(p == nullptr);
void (A::*pmf)() = 0;
#ifdef __clang__
// GCC 4.2 can't handle this
assert(pmf == nullptr);
#endif
int A::*pmd = 0;
assert(pmd == nullptr);
A a1(nullptr);
A a2(0);
bool b = nullptr;
assert(!b);
assert(nullptr == nullptr);
assert(nullptr <= nullptr);
assert(nullptr >= nullptr);
assert(!(nullptr != nullptr));
assert(!(nullptr < nullptr));
assert(!(nullptr > nullptr));
A* a = nullptr;
assert(a == nullptr);
assert(a <= nullptr);
assert(a >= nullptr);
assert(!(a != nullptr));
assert(!(a < nullptr));
assert(!(a > nullptr));
assert(nullptr == a);
assert(nullptr <= a);
assert(nullptr >= a);
assert(!(nullptr != a));
assert(!(nullptr < a));
assert(!(nullptr > a));
std::ptrdiff_t i = reinterpret_cast<std::ptrdiff_t>(nullptr);
assert(i == 0);
{
test_conversions<std::nullptr_t>();
test_conversions<void*>();
test_conversions<A*>();
test_conversions<void(*)()>();
test_conversions<void(A::*)()>();
test_conversions<int A::*>();
}
{
test_comparisons<std::nullptr_t>();
test_comparisons<void*>();
test_comparisons<A*>();
test_comparisons<void(*)()>();
}
{
bool b = nullptr;
assert(!b);
}
}

View File

@ -0,0 +1,17 @@
//===----------------------------------------------------------------------===//
//
// The LLVM Compiler Infrastructure
//
// This file is dual licensed under the MIT and the University of Illinois Open
// Source Licenses. See LICENSE.TXT for details.
//
//===----------------------------------------------------------------------===//
// typedef decltype(nullptr) nullptr_t;
#include <cstddef>
int main()
{
std::ptrdiff_t i = static_cast<std::ptrdiff_t>(nullptr);
}

View File

@ -0,0 +1,24 @@
//===----------------------------------------------------------------------===//
//
// The LLVM Compiler Infrastructure
//
// This file is dual licensed under the MIT and the University of Illinois Open
// Source Licenses. See LICENSE.TXT for details.
//
//===----------------------------------------------------------------------===//
// NOTE: nullptr_t emulation cannot handle a reinterpret_cast to an
// integral type
// XFAIL: c++98, c++03
// typedef decltype(nullptr) nullptr_t;
#include <cstddef>
#include <cassert>
int main()
{
std::ptrdiff_t i = reinterpret_cast<std::ptrdiff_t>(nullptr);
assert(i == 0);
}