mirror of
https://github.com/microsoft/GSL.git
synced 2024-11-29 02:51:59 -05:00
194 lines
5.0 KiB
C++
194 lines
5.0 KiB
C++
|
///////////////////////////////////////////////////////////////////////////////
|
||
|
//
|
||
|
// Copyright (c) 2015 Microsoft Corporation. All rights reserved.
|
||
|
//
|
||
|
// This code is licensed under the MIT License (MIT).
|
||
|
//
|
||
|
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||
|
// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||
|
// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||
|
// AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||
|
// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||
|
// OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
||
|
// THE SOFTWARE.
|
||
|
//
|
||
|
///////////////////////////////////////////////////////////////////////////////
|
||
|
|
||
|
#ifdef _MSC_VER
|
||
|
// blanket turn off warnings from CppCoreCheck from catch
|
||
|
// so people aren't annoyed by them when running the tool.
|
||
|
#pragma warning(disable : 26440 26426) // from catch
|
||
|
|
||
|
// Fix VS2015 build breaks in Release
|
||
|
#pragma warning(disable : 4702) // unreachable code
|
||
|
#endif
|
||
|
|
||
|
#include <catch/catch.hpp> // for AssertionHandler, StringRef, CHECK, TEST_...
|
||
|
#include <gsl/pointers> // for not_null, operator<, operator<=, operator>
|
||
|
|
||
|
namespace gsl
|
||
|
{
|
||
|
struct fail_fast;
|
||
|
} // namespace gsl
|
||
|
|
||
|
using namespace gsl;
|
||
|
|
||
|
GSL_SUPPRESS(f.4) // NO-FORMAT: attribute
|
||
|
bool helper(not_null<int*> p) { return *p == 12; }
|
||
|
GSL_SUPPRESS(f.4) // NO-FORMAT: attribute
|
||
|
bool helper_const(not_null<const int*> p) { return *p == 12; }
|
||
|
|
||
|
GSL_SUPPRESS(f.4) // NO-FORMAT: attribute
|
||
|
bool strict_helper(strict_not_null<int*> p) { return *p == 12; }
|
||
|
GSL_SUPPRESS(f.4) // NO-FORMAT: attribute
|
||
|
bool strict_helper_const(strict_not_null<const int*> p) { return *p == 12; }
|
||
|
|
||
|
int* return_pointer() { return nullptr; }
|
||
|
const int* return_pointer_const() { return nullptr; }
|
||
|
|
||
|
GSL_SUPPRESS(con.4) // NO-FORMAT: attribute
|
||
|
TEST_CASE("TestStrictNotNull")
|
||
|
{
|
||
|
{
|
||
|
// raw ptr <-> strict_not_null
|
||
|
int x = 42;
|
||
|
|
||
|
#ifdef CONFIRM_COMPILATION_ERRORS
|
||
|
strict_not_null<int*> snn = &x;
|
||
|
strict_helper(&x);
|
||
|
strict_helper_const(&x);
|
||
|
strict_helper(return_pointer());
|
||
|
strict_helper_const(return_pointer_const());
|
||
|
#endif
|
||
|
|
||
|
const strict_not_null<int*> snn1{&x};
|
||
|
|
||
|
helper(snn1);
|
||
|
helper_const(snn1);
|
||
|
|
||
|
CHECK(*snn1 == 42);
|
||
|
}
|
||
|
|
||
|
{
|
||
|
// strict_not_null -> strict_not_null
|
||
|
int x = 42;
|
||
|
|
||
|
strict_not_null<int*> snn1{&x};
|
||
|
const strict_not_null<int*> snn2{&x};
|
||
|
|
||
|
strict_helper(snn1);
|
||
|
strict_helper_const(snn1);
|
||
|
strict_helper_const(snn2);
|
||
|
|
||
|
CHECK(snn1 == snn2);
|
||
|
}
|
||
|
|
||
|
{
|
||
|
// strict_not_null -> not_null
|
||
|
int x = 42;
|
||
|
|
||
|
strict_not_null<int*> snn{&x};
|
||
|
|
||
|
const not_null<int*> nn1 = snn;
|
||
|
const not_null<int*> nn2{snn};
|
||
|
|
||
|
helper(snn);
|
||
|
helper_const(snn);
|
||
|
|
||
|
CHECK(snn == nn1);
|
||
|
CHECK(snn == nn2);
|
||
|
}
|
||
|
|
||
|
{
|
||
|
// not_null -> strict_not_null
|
||
|
int x = 42;
|
||
|
|
||
|
not_null<int*> nn{&x};
|
||
|
|
||
|
const strict_not_null<int*> snn1{nn};
|
||
|
const strict_not_null<int*> snn2{nn};
|
||
|
|
||
|
strict_helper(nn);
|
||
|
strict_helper_const(nn);
|
||
|
|
||
|
CHECK(snn1 == nn);
|
||
|
CHECK(snn2 == nn);
|
||
|
|
||
|
std::hash<strict_not_null<int*>> hash_snn;
|
||
|
std::hash<not_null<int*>> hash_nn;
|
||
|
|
||
|
CHECK(hash_nn(snn1) == hash_nn(nn));
|
||
|
CHECK(hash_snn(snn1) == hash_nn(nn));
|
||
|
CHECK(hash_nn(snn1) == hash_nn(snn2));
|
||
|
CHECK(hash_snn(snn1) == hash_snn(nn));
|
||
|
}
|
||
|
|
||
|
#ifdef CONFIRM_COMPILATION_ERRORS
|
||
|
{
|
||
|
strict_not_null<int*> p{nullptr};
|
||
|
}
|
||
|
#endif
|
||
|
}
|
||
|
|
||
|
#if defined(__cplusplus) && (__cplusplus >= 201703L)
|
||
|
|
||
|
GSL_SUPPRESS(con.4) // NO-FORMAT: attribute
|
||
|
TEST_CASE("TestStrictNotNullConstructorTypeDeduction")
|
||
|
{
|
||
|
{
|
||
|
int i = 42;
|
||
|
|
||
|
strict_not_null x{&i};
|
||
|
helper(strict_not_null{&i});
|
||
|
helper_const(strict_not_null{&i});
|
||
|
|
||
|
CHECK(*x == 42);
|
||
|
}
|
||
|
|
||
|
{
|
||
|
int i = 42;
|
||
|
int* p = &i;
|
||
|
|
||
|
strict_not_null x{p};
|
||
|
helper(strict_not_null{p});
|
||
|
helper_const(strict_not_null{p});
|
||
|
|
||
|
CHECK(*x == 42);
|
||
|
}
|
||
|
|
||
|
{
|
||
|
auto workaround_macro = []() {
|
||
|
int* p1 = nullptr;
|
||
|
const strict_not_null x{p1};
|
||
|
};
|
||
|
CHECK_THROWS_AS(workaround_macro(), fail_fast);
|
||
|
}
|
||
|
|
||
|
{
|
||
|
auto workaround_macro = []() {
|
||
|
const int* p1 = nullptr;
|
||
|
const strict_not_null x{p1};
|
||
|
};
|
||
|
CHECK_THROWS_AS(workaround_macro(), fail_fast);
|
||
|
}
|
||
|
|
||
|
{
|
||
|
int* p = nullptr;
|
||
|
|
||
|
CHECK_THROWS_AS(helper(strict_not_null{p}), fail_fast);
|
||
|
CHECK_THROWS_AS(helper_const(strict_not_null{p}), fail_fast);
|
||
|
}
|
||
|
|
||
|
#ifdef CONFIRM_COMPILATION_ERRORS
|
||
|
{
|
||
|
strict_not_null x{nullptr};
|
||
|
helper(strict_not_null{nullptr});
|
||
|
helper_const(strict_not_null{nullptr});
|
||
|
}
|
||
|
#endif
|
||
|
}
|
||
|
#endif // #if defined(__cplusplus) && (__cplusplus >= 201703L)
|
||
|
|
||
|
static_assert(std::is_nothrow_move_constructible<strict_not_null<void*>>::value,
|
||
|
"strict_not_null must be no-throw move constructible");
|