Compare commits

..

No commits in common. "main" and "v4.1.0" have entirely different histories.
main ... v4.1.0

26 changed files with 487 additions and 1061 deletions

View File

@ -1,29 +0,0 @@
---
name: Bug report
about: Create a report to help us improve
title: ''
labels: 'Status: Open, Type: Bug'
assignees: ''
---
**Describe the bug**
A clear and concise description of what the bug is.
**To Reproduce**
```c++
#include <gsl>
// your repro here: ...
```
**Expected behavior**
A clear and concise description of what you expected to happen.
**Spec (please complete the following information):**
- OS: [e.g. Windows]
- Compiler: [e.g. MSVC]
- C++ Version: [e.g. C++20]
**Additional context**
Add any other context about the problem here.

View File

@ -38,7 +38,7 @@ jobs:
echo "Emulator starting in background" echo "Emulator starting in background"
- name: Configure - name: Configure
run: cmake -DCMAKE_TOOLCHAIN_FILE=$ANDROID_NDK_LATEST_HOME/build/cmake/android.toolchain.cmake -DANDROID_PLATFORM=16 -DANDROID_ABI=x86_64 -DCMAKE_BUILD_TYPE=Debug .. run: cmake -Werror=dev -DCMAKE_TOOLCHAIN_FILE=$ANDROID_NDK_LATEST_HOME/build/cmake/android.toolchain.cmake -DANDROID_PLATFORM=16 -DANDROID_ABI=x86_64 -DCMAKE_BUILD_TYPE=Debug ..
- name: Build - name: Build
run: cmake --build . --parallel run: cmake --build . --parallel

View File

@ -17,16 +17,11 @@ jobs:
gcc: gcc:
strategy: strategy:
matrix: matrix:
gcc_version: [ 12, 13, 14 ] gcc_version: [ 10, 11, 12 ]
build_type: [ Debug, Release ] build_type: [ Debug, Release ]
cxx_version: [ 14, 17, 20, 23 ] cxx_version: [ 14, 17, 20, 23 ]
exclude: exclude:
# https://github.com/google/googletest/issues/4232 - gcc_version: 10
# Looks like GoogleTest is not interested in making version 1.14
# work with gcc-12.
- gcc_version: 12
cxx_version: 20
- gcc_version: 12
cxx_version: 23 cxx_version: 23
runs-on: ubuntu-latest runs-on: ubuntu-latest
steps: steps:
@ -42,15 +37,9 @@ jobs:
clang: clang:
strategy: strategy:
matrix: matrix:
clang_version: [ 16, 17, 18 ] clang_version: [ 13, 14, 15 ]
build_type: [ Debug, Release ] build_type: [ Debug, Release ]
cxx_version: [ 14, 17, 20, 23 ] cxx_version: [ 14, 17, 20, 23 ]
exclude:
# https://github.com/llvm/llvm-project/issues/93734
# Looks like clang fixed this issue in clang-18, but won't backport
# the fix.
- clang_version: 17
cxx_version: 23
runs-on: ubuntu-latest runs-on: ubuntu-latest
steps: steps:
- uses: actions/checkout@v4 - uses: actions/checkout@v4
@ -65,7 +54,7 @@ jobs:
xcode: xcode:
strategy: strategy:
matrix: matrix:
xcode_version: [ '15.4' ] xcode_version: [ '14.3.1', '15.4' ]
build_type: [ Debug, Release ] build_type: [ Debug, Release ]
cxx_version: [ 14, 17, 20, 23 ] cxx_version: [ 14, 17, 20, 23 ]
runs-on: macos-latest runs-on: macos-latest

View File

@ -1,6 +1,6 @@
cmake_minimum_required(VERSION 3.14...3.16) cmake_minimum_required(VERSION 3.14...3.16)
project(GSL VERSION 4.2.0 LANGUAGES CXX) project(GSL VERSION 4.1.0 LANGUAGES CXX)
add_library(GSL INTERFACE) add_library(GSL INTERFACE)
add_library(Microsoft.GSL::GSL ALIAS GSL) add_library(Microsoft.GSL::GSL ALIAS GSL)

View File

@ -1,6 +1,5 @@
# GSL: Guidelines Support Library # GSL: Guidelines Support Library
[![CI](https://github.com/Microsoft/GSL/actions/workflows/compilers.yml/badge.svg)](https://github.com/microsoft/GSL/actions/workflows/compilers.yml?query=branch%3Amain) [![Build Status](https://dev.azure.com/cppstat/GSL/_apis/build/status/microsoft.GSL?branchName=main)](https://dev.azure.com/cppstat/GSL/_build/latest?definitionId=1&branchName=main)
[![vcpkg](https://img.shields.io/vcpkg/v/ms-gsl)](https://vcpkg.io/en/package/ms-gsl)
The Guidelines Support Library (GSL) contains functions and types that are suggested for use by the The Guidelines Support Library (GSL) contains functions and types that are suggested for use by the
[C++ Core Guidelines](https://github.com/isocpp/CppCoreGuidelines) maintained by the [Standard C++ Foundation](https://isocpp.org). [C++ Core Guidelines](https://github.com/isocpp/CppCoreGuidelines) maintained by the [Standard C++ Foundation](https://isocpp.org).
@ -40,6 +39,8 @@ span_p | &#x26
[u32zstring](docs/headers.md#user-content-H-zstring) | &#x2611; | An alias to `basic_zstring` with dynamic extent and a char type of `char32_t` [u32zstring](docs/headers.md#user-content-H-zstring) | &#x2611; | An alias to `basic_zstring` with dynamic extent and a char type of `char32_t`
[cu32zstring](docs/headers.md#user-content-H-zstring) | &#x2611; | An alias to `basic_zstring` with dynamic extent and a char type of `const char32_t` [cu32zstring](docs/headers.md#user-content-H-zstring) | &#x2611; | An alias to `basic_zstring` with dynamic extent and a char type of `const char32_t`
[**2. Owners**][cg-owners] | | [**2. Owners**][cg-owners] | |
[unique_ptr](docs/headers.md#user-content-H-pointers-unique_ptr) | &#x2611; | An alias to `std::unique_ptr`
[shared_ptr](docs/headers.md#user-content-H-pointers-shared_ptr) | &#x2611; | An alias to `std::shared_ptr`
stack_array | &#x2610; | A stack-allocated array stack_array | &#x2610; | A stack-allocated array
dyn_array | &#x2610; | A heap-allocated array dyn_array | &#x2610; | A heap-allocated array
[**3. Assertions**][cg-assertions] | | [**3. Assertions**][cg-assertions] | |
@ -47,11 +48,13 @@ dyn_array | &#x26
[Ensures](docs/headers.md#user-content-H-assert-ensures) | &#x2611; | A postcondition assertion; on failure it terminates [Ensures](docs/headers.md#user-content-H-assert-ensures) | &#x2611; | A postcondition assertion; on failure it terminates
[**4. Utilities**][cg-utilities] | | [**4. Utilities**][cg-utilities] | |
move_owner | &#x2610; | A helper function that moves one `owner` to the other move_owner | &#x2610; | A helper function that moves one `owner` to the other
[byte](docs/headers.md#user-content-H-byte-byte) | &#x2611; | Either an alias to `std::byte` or a byte type
[final_action](docs/headers.md#user-content-H-util-final_action) | &#x2611; | A RAII style class that invokes a functor on its destruction [final_action](docs/headers.md#user-content-H-util-final_action) | &#x2611; | A RAII style class that invokes a functor on its destruction
[finally](docs/headers.md#user-content-H-util-finally) | &#x2611; | A helper function instantiating [final_action](docs/headers.md#user-content-H-util-final_action) [finally](docs/headers.md#user-content-H-util-finally) | &#x2611; | A helper function instantiating [final_action](docs/headers.md#user-content-H-util-final_action)
[GSL_SUPPRESS](docs/headers.md#user-content-H-assert-gsl_suppress) | &#x2611; | A macro that takes an argument and turns it into `[[gsl::suppress(x)]]` or `[[gsl::suppress("x")]]` [GSL_SUPPRESS](docs/headers.md#user-content-H-assert-gsl_suppress) | &#x2611; | A macro that takes an argument and turns it into `[[gsl::suppress(x)]]` or `[[gsl::suppress("x")]]`
[[implicit]] | &#x2610; | A "marker" to put on single-argument constructors to explicitly make them non-explicit [[implicit]] | &#x2610; | A "marker" to put on single-argument constructors to explicitly make them non-explicit
[index](docs/headers.md#user-content-H-util-index) | &#x2611; | A type to use for all container and array indexing (currently an alias for `std::ptrdiff_t`) [index](docs/headers.md#user-content-H-util-index) | &#x2611; | A type to use for all container and array indexing (currently an alias for `std::ptrdiff_t`)
joining_thread | &#x2610; | A RAII style version of `std::thread` that joins
[narrow](docs/headers.md#user-content-H-narrow-narrow) | &#x2611; | A checked version of `narrow_cast`; it can throw [narrowing_error](docs/headers.md#user-content-H-narrow-narrowing_error) [narrow](docs/headers.md#user-content-H-narrow-narrow) | &#x2611; | A checked version of `narrow_cast`; it can throw [narrowing_error](docs/headers.md#user-content-H-narrow-narrowing_error)
[narrow_cast](docs/headers.md#user-content-H-util-narrow_cast) | &#x2611; | A narrowing cast for values and a synonym for `static_cast` [narrow_cast](docs/headers.md#user-content-H-util-narrow_cast) | &#x2611; | A narrowing cast for values and a synonym for `static_cast`
[narrowing_error](docs/headers.md#user-content-H-narrow-narrowing_error) | &#x2611; | A custom exception type thrown by [narrow](docs/headers.md#user-content-H-narrow-narrow) [narrowing_error](docs/headers.md#user-content-H-narrow-narrowing_error) | &#x2611; | A custom exception type thrown by [narrow](docs/headers.md#user-content-H-narrow-narrow)
@ -73,14 +76,6 @@ cu16string_span | &#x2610; | Deprecated. An alias to `basic
u32string_span | &#x2610; | Deprecated. An alias to `basic_string_span` with a char type of `char32_t` u32string_span | &#x2610; | Deprecated. An alias to `basic_string_span` with a char type of `char32_t`
cu32string_span | &#x2610; | Deprecated. An alias to `basic_string_span` with a char type of `const char32_t` cu32string_span | &#x2610; | Deprecated. An alias to `basic_string_span` with a char type of `const char32_t`
## The following features have been adopted by WG21. They are deprecated in GSL.
Feature | Deprecated Since | Notes
------------------------------------------------------------------|------------------|------
[unique_ptr](docs/headers.md#user-content-H-pointers-unique_ptr) | C++11 | Use std::unique_ptr instead.
[shared_ptr](docs/headers.md#user-content-H-pointers-shared_ptr) | C++11 | Use std::shared_ptr instead.
[byte](docs/headers.md#user-content-H-byte-byte) | C++17 | Use std::byte instead.
joining_thread | C++20 (Note: Not yet implemented in GSL) | Use std::jthread instead.
This is based on [CppCoreGuidelines semi-specification](https://github.com/isocpp/CppCoreGuidelines/blob/master/CppCoreGuidelines.md#gsl-guidelines-support-library). This is based on [CppCoreGuidelines semi-specification](https://github.com/isocpp/CppCoreGuidelines/blob/master/CppCoreGuidelines.md#gsl-guidelines-support-library).
[cg-views]: https://github.com/isocpp/CppCoreGuidelines/blob/master/CppCoreGuidelines.md#gslview-views [cg-views]: https://github.com/isocpp/CppCoreGuidelines/blob/master/CppCoreGuidelines.md#gslview-views
@ -97,9 +92,9 @@ Below is a table showing the versions currently being tested (also see [.github/
Compiler |Toolset Versions Currently Tested Compiler |Toolset Versions Currently Tested
:------- |--: :------- |--:
GCC | 12, 13, 14 GCC | 10, 11, 12
XCode | 14.3.1, 15.4 XCode | 14.3.1, 15.4
Clang | 16, 17, 18 Clang | 13, 14, 15
Visual Studio with MSVC | VS2019, VS2022 Visual Studio with MSVC | VS2019, VS2022
Visual Studio with LLVM | VS2019, VS2022 Visual Studio with LLVM | VS2019, VS2022
@ -111,8 +106,8 @@ If you successfully port GSL to another platform, we would love to hear from you
Target | CI/CD Status Target | CI/CD Status
:------- | -----------: :------- | -----------:
iOS | [![CI_iOS](https://github.com/microsoft/GSL/workflows/CI_iOS/badge.svg?branch=main)](https://github.com/microsoft/GSL/actions/workflows/ios.yml?query=branch%3Amain) iOS | ![CI_iOS](https://github.com/microsoft/GSL/workflows/CI_iOS/badge.svg)
Android | [![CI_Android](https://github.com/microsoft/GSL/workflows/CI_Android/badge.svg?branch=main)](https://github.com/microsoft/GSL/actions/workflows/android.yml?query=branch%3Amain) Android | ![CI_Android](https://github.com/microsoft/GSL/workflows/CI_Android/badge.svg)
Note: These CI/CD steps are run with each pull request, however failures in them are non-blocking. Note: These CI/CD steps are run with each pull request, however failures in them are non-blocking.
@ -202,7 +197,7 @@ include(FetchContent)
FetchContent_Declare(GSL FetchContent_Declare(GSL
GIT_REPOSITORY "https://github.com/microsoft/GSL" GIT_REPOSITORY "https://github.com/microsoft/GSL"
GIT_TAG "v4.2.0" GIT_TAG "v4.1.0"
GIT_SHALLOW ON GIT_SHALLOW ON
) )

View File

@ -91,16 +91,16 @@ See [SL.str.5: Use `std::byte` to refer to byte values that do not necessarily r
### Non-member functions ### Non-member functions
```cpp ```cpp
template <class IntegerType, std::enable_if_t<std::is_integral<IntegerType>::value, bool> = true> template <class IntegerType, class = std::enable_if_t<std::is_integral<IntegerType>::value>>
constexpr byte& operator<<=(byte& b, IntegerType shift) noexcept; constexpr byte& operator<<=(byte& b, IntegerType shift) noexcept;
template <class IntegerType, std::enable_if_t<std::is_integral<IntegerType>::value, bool> = true> template <class IntegerType, class = std::enable_if_t<std::is_integral<IntegerType>::value>>
constexpr byte operator<<(byte b, IntegerType shift) noexcept; constexpr byte operator<<(byte b, IntegerType shift) noexcept;
template <class IntegerType, std::enable_if_t<std::is_integral<IntegerType>::value, bool> = true> template <class IntegerType, class = std::enable_if_t<std::is_integral<IntegerType>::value>>
constexpr byte& operator>>=(byte& b, IntegerType shift) noexcept; constexpr byte& operator>>=(byte& b, IntegerType shift) noexcept;
template <class IntegerType, std::enable_if_t<std::is_integral<IntegerType>::value, bool> = true> template <class IntegerType, class = std::enable_if_t<std::is_integral<IntegerType>::value>>
constexpr byte operator>>(byte b, IntegerType shift) noexcept; constexpr byte operator>>(byte b, IntegerType shift) noexcept;
``` ```
@ -134,7 +134,7 @@ constexpr byte operator~(byte b) noexcept;
Bitwise negation of a `byte`. Flips all bits. Zeroes become ones, ones become zeroes. Bitwise negation of a `byte`. Flips all bits. Zeroes become ones, ones become zeroes.
```cpp ```cpp
template <class IntegerType, std::enable_if_t<std::is_integral<IntegerType>::value, bool> = true> template <class IntegerType, class = std::enable_if_t<std::is_integral<IntegerType>::value>>
constexpr IntegerType to_integer(byte b) noexcept; constexpr IntegerType to_integer(byte b) noexcept;
``` ```
@ -224,14 +224,6 @@ When a nullptr check fails, `std::terminate` is called.
See [F.23: Use a `not_null<T>` to indicate that “null” is not a valid value](https://isocpp.github.io/CppCoreGuidelines/CppCoreGuidelines#Rf-nullptr) See [F.23: Use a `not_null<T>` to indicate that “null” is not a valid value](https://isocpp.github.io/CppCoreGuidelines/CppCoreGuidelines#Rf-nullptr)
#### Member Types
```cpp
using element_type = T;
```
The type of the pointer or smart pointer that is managed by this object.
#### Member functions #### Member functions
##### Construct/Copy ##### Construct/Copy
@ -302,12 +294,6 @@ void operator[](std::ptrdiff_t) const = delete;
Array index operator is explicitly deleted. Pointers point to single objects ([I.13: Do not pass an array as a single pointer](http://isocpp.github.io/CppCoreGuidelines/CppCoreGuidelines#Ri-array)), so don't allow treating them as an array. Array index operator is explicitly deleted. Pointers point to single objects ([I.13: Do not pass an array as a single pointer](http://isocpp.github.io/CppCoreGuidelines/CppCoreGuidelines#Ri-array)), so don't allow treating them as an array.
```cpp
void swap(not_null<T>& other) { std::swap(ptr_, other.ptr_); }
```
Swaps contents with another `gsl::not_null` object.
#### Non-member functions #### Non-member functions
```cpp ```cpp
@ -317,13 +303,6 @@ auto make_not_null(T&& t) noexcept;
Creates a `gsl::not_null` object, deducing the target type from the type of the argument. Creates a `gsl::not_null` object, deducing the target type from the type of the argument.
```cpp
template <typename T, std::enable_if_t<std::is_move_assignable<T>::value && std::is_move_constructible<T>::value, bool> = true>
void swap(not_null<T>& a, not_null<T>& b);
```
Swaps the contents of two `gsl::not_null` objects.
```cpp ```cpp
template <class T, class U> template <class T, class U>
auto operator==(const not_null<T>& lhs, auto operator==(const not_null<T>& lhs,
@ -684,6 +663,10 @@ template <class Container>
constexpr span<typename Container::value_type> make_span(Container& cont); constexpr span<typename Container::value_type> make_span(Container& cont);
template <class Container> template <class Container>
constexpr span<const typename Container::value_type> make_span(const Container& cont); constexpr span<const typename Container::value_type> make_span(const Container& cont);
template <class Ptr>
constexpr span<typename Ptr::element_type> make_span(Ptr& cont, std::size_t count);
template <class Ptr>
constexpr span<typename Ptr::element_type> make_span(Ptr& cont);
``` ```
Utility function for creating a `span` with [`gsl::dynamic_extent`](#user-content-H-span_ext-dynamic_extent) from Utility function for creating a `span` with [`gsl::dynamic_extent`](#user-content-H-span_ext-dynamic_extent) from
@ -875,10 +858,3 @@ constexpr auto at(std::span<T, extent> sp, const index i) -> decltype(sp[sp.size
This overload returns a reference to the `i`s element of the `std::span` `sp`. It [`Expects`](#user-content-H-assert-expects) that the provided index is within the bounds of the array. This overload returns a reference to the `i`s element of the `std::span` `sp`. It [`Expects`](#user-content-H-assert-expects) that the provided index is within the bounds of the array.
For [`gsl::at`](#user-content-H-span_ext-at) for [`gsl::span`](#user-content-H-span-span) see header [`span_ext`](#user-content-H-span_ext). For [`gsl::at`](#user-content-H-span_ext-at) for [`gsl::span`](#user-content-H-span-span) see header [`span_ext`](#user-content-H-span_ext).
```cpp
template <class T, std::enable_if_t<std::is_move_assignable<T>::value && std::is_move_constructible<T>::value>>
void swap(T& a, T& b);
```
Swaps the contents of two objects. Exists only to specialize `gsl::swap<T>(gsl::not_null<T>&, gsl::not_null<T>&)`.

View File

@ -17,8 +17,8 @@
#ifndef GSL_ALGORITHM_H #ifndef GSL_ALGORITHM_H
#define GSL_ALGORITHM_H #define GSL_ALGORITHM_H
#include "./assert" // for Expects #include "assert" // for Expects
#include "./span" // for dynamic_extent, span #include "span" // for dynamic_extent, span
#include <algorithm> // for copy_n #include <algorithm> // for copy_n
#include <cstddef> // for ptrdiff_t #include <cstddef> // for ptrdiff_t

View File

@ -17,8 +17,6 @@
#ifndef GSL_BYTE_H #ifndef GSL_BYTE_H
#define GSL_BYTE_H #define GSL_BYTE_H
#include "./util" // for GSL_DEPRECATED
#include <type_traits> #include <type_traits>
#ifdef _MSC_VER #ifdef _MSC_VER
@ -82,14 +80,7 @@ namespace gsl
{ {
#if GSL_USE_STD_BYTE #if GSL_USE_STD_BYTE
namespace impl { using std::byte;
// impl::byte is used by gsl::as_bytes so our own code does not trigger a deprecation warning as would be the case when we used gsl::byte.
// Users of GSL should only use gsl::byte, not gsl::impl::byte.
using byte = std::byte;
}
using byte GSL_DEPRECATED("Use std::byte instead.") = std::byte;
using std::to_integer; using std::to_integer;
#else // GSL_USE_STD_BYTE #else // GSL_USE_STD_BYTE
@ -100,31 +91,25 @@ enum class byte_may_alias byte : unsigned char
{ {
}; };
namespace impl { template <class IntegerType, class = std::enable_if_t<std::is_integral<IntegerType>::value>>
// impl::byte is used by gsl::as_bytes so our own code does not trigger a deprecation warning as would be the case when we used gsl::byte.
// Users of GSL should only use gsl::byte, not gsl::impl::byte.
using byte = gsl::byte;
}
template <class IntegerType, std::enable_if_t<std::is_integral<IntegerType>::value, bool> = true>
constexpr byte& operator<<=(byte& b, IntegerType shift) noexcept constexpr byte& operator<<=(byte& b, IntegerType shift) noexcept
{ {
return b = byte(static_cast<unsigned char>(b) << shift); return b = byte(static_cast<unsigned char>(b) << shift);
} }
template <class IntegerType, std::enable_if_t<std::is_integral<IntegerType>::value, bool> = true> template <class IntegerType, class = std::enable_if_t<std::is_integral<IntegerType>::value>>
constexpr byte operator<<(byte b, IntegerType shift) noexcept constexpr byte operator<<(byte b, IntegerType shift) noexcept
{ {
return byte(static_cast<unsigned char>(b) << shift); return byte(static_cast<unsigned char>(b) << shift);
} }
template <class IntegerType, std::enable_if_t<std::is_integral<IntegerType>::value, bool> = true> template <class IntegerType, class = std::enable_if_t<std::is_integral<IntegerType>::value>>
constexpr byte& operator>>=(byte& b, IntegerType shift) noexcept constexpr byte& operator>>=(byte& b, IntegerType shift) noexcept
{ {
return b = byte(static_cast<unsigned char>(b) >> shift); return b = byte(static_cast<unsigned char>(b) >> shift);
} }
template <class IntegerType, std::enable_if_t<std::is_integral<IntegerType>::value, bool> = true> template <class IntegerType, class = std::enable_if_t<std::is_integral<IntegerType>::value>>
constexpr byte operator>>(byte b, IntegerType shift) noexcept constexpr byte operator>>(byte b, IntegerType shift) noexcept
{ {
return byte(static_cast<unsigned char>(b) >> shift); return byte(static_cast<unsigned char>(b) >> shift);
@ -162,7 +147,7 @@ constexpr byte operator^(byte l, byte r) noexcept
constexpr byte operator~(byte b) noexcept { return byte(~static_cast<unsigned char>(b)); } constexpr byte operator~(byte b) noexcept { return byte(~static_cast<unsigned char>(b)); }
template <class IntegerType, std::enable_if_t<std::is_integral<IntegerType>::value, bool> = true> template <class IntegerType, class = std::enable_if_t<std::is_integral<IntegerType>::value>>
constexpr IntegerType to_integer(byte b) noexcept constexpr IntegerType to_integer(byte b) noexcept
{ {
return static_cast<IntegerType>(b); return static_cast<IntegerType>(b);
@ -170,24 +155,23 @@ constexpr IntegerType to_integer(byte b) noexcept
#endif // GSL_USE_STD_BYTE #endif // GSL_USE_STD_BYTE
template <typename T> template <typename T>
// NOTE: need suppression since c++14 does not allow "return {t}" // NOTE: need suppression since c++14 does not allow "return {t}"
// GSL_SUPPRESS(type.4) // NO-FORMAT: attribute // TODO: suppression does not work // GSL_SUPPRESS(type.4) // NO-FORMAT: attribute // TODO: suppression does not work
constexpr gsl::impl::byte to_byte(T t) noexcept constexpr byte to_byte(T t) noexcept
{ {
static_assert(std::is_same<T, unsigned char>::value, static_assert(std::is_same<T, unsigned char>::value,
"gsl::to_byte(t) must be provided an unsigned char, otherwise data loss may occur. " "gsl::to_byte(t) must be provided an unsigned char, otherwise data loss may occur. "
"If you are calling to_byte with an integer constant use: gsl::to_byte<t>() version."); "If you are calling to_byte with an integer constant use: gsl::to_byte<t>() version.");
return gsl::impl::byte(t); return byte(t);
} }
template <int I> template <int I>
constexpr gsl::impl::byte to_byte() noexcept constexpr byte to_byte() noexcept
{ {
static_assert(I >= 0 && I <= 255, static_assert(I >= 0 && I <= 255,
"gsl::byte only has 8 bits of storage, values must be in range 0-255"); "gsl::byte only has 8 bits of storage, values must be in range 0-255");
return static_cast<gsl::impl::byte>(I); return static_cast<byte>(I);
} }
} // namespace gsl } // namespace gsl

View File

@ -18,16 +18,16 @@
#define GSL_GSL_H #define GSL_GSL_H
// IWYU pragma: begin_exports // IWYU pragma: begin_exports
#include "./algorithm" // copy #include "algorithm" // copy
#include "./assert" // Ensures/Expects #include "assert" // Ensures/Expects
#include "./byte" // byte #include "byte" // byte
#include "./pointers" // owner, not_null #include "pointers" // owner, not_null
#include "./span" // span #include "span" // span
#include "./zstring" // zstring #include "zstring" // zstring
#include "./util" // finally()/narrow_cast()... #include "util" // finally()/narrow_cast()...
#ifdef __cpp_exceptions #ifdef __cpp_exceptions
#include "./narrow" // narrow() #include "narrow" // narrow()
#endif #endif
// IWYU pragma: end_exports // IWYU pragma: end_exports

View File

@ -16,8 +16,8 @@
#ifndef GSL_NARROW_H #ifndef GSL_NARROW_H
#define GSL_NARROW_H #define GSL_NARROW_H
#include "./assert" // for GSL_SUPPRESS #include "assert" // for GSL_SUPPRESS
#include "./util" // for narrow_cast #include "util" // for narrow_cast
#include <exception> // for std::exception #include <exception> // for std::exception
namespace gsl namespace gsl
{ {

View File

@ -17,8 +17,7 @@
#ifndef GSL_POINTERS_H #ifndef GSL_POINTERS_H
#define GSL_POINTERS_H #define GSL_POINTERS_H
#include "./assert" // for Ensures, Expects #include "assert" // for Ensures, Expects
#include "./util" // for GSL_DEPRECATED
#include <cstddef> // for ptrdiff_t, nullptr_t, size_t #include <cstddef> // for ptrdiff_t, nullptr_t, size_t
#include <functional> // for less, greater #include <functional> // for less, greater
@ -63,11 +62,8 @@ namespace details
// //
// GSL.owner: ownership pointers // GSL.owner: ownership pointers
// //
template <typename... Ts> using std::shared_ptr;
using shared_ptr GSL_DEPRECATED("Use std::shared_ptr instead") = std::shared_ptr<Ts...>; using std::unique_ptr;
template <typename... Ts>
using unique_ptr GSL_DEPRECATED("Use std::unique_ptr instead") = std::unique_ptr<Ts...>;
// //
// owner // owner
@ -79,7 +75,7 @@ using unique_ptr GSL_DEPRECATED("Use std::unique_ptr instead") = std::unique_ptr
// T must be a pointer type // T must be a pointer type
// - disallow construction from any type other than pointer type // - disallow construction from any type other than pointer type
// //
template <class T, std::enable_if_t<std::is_pointer<T>::value, bool> = true> template <class T, class = std::enable_if_t<std::is_pointer<T>::value>>
using owner = T; using owner = T;
// //
@ -102,8 +98,6 @@ class not_null
public: public:
static_assert(details::is_comparable_to_nullptr<T>::value, "T cannot be compared to nullptr."); static_assert(details::is_comparable_to_nullptr<T>::value, "T cannot be compared to nullptr.");
using element_type = T;
template <typename U, typename = std::enable_if_t<std::is_convertible<U, T>::value>> template <typename U, typename = std::enable_if_t<std::is_convertible<U, T>::value>>
constexpr not_null(U&& u) noexcept(std::is_nothrow_move_constructible<T>::value) : ptr_(std::forward<U>(u)) constexpr not_null(U&& u) noexcept(std::is_nothrow_move_constructible<T>::value) : ptr_(std::forward<U>(u))
{ {
@ -123,7 +117,7 @@ public:
not_null(const not_null& other) = default; not_null(const not_null& other) = default;
not_null& operator=(const not_null& other) = default; not_null& operator=(const not_null& other) = default;
constexpr details::value_or_reference_return_t<T> get() const constexpr details::value_or_reference_return_t<T> get() const
noexcept(noexcept(details::value_or_reference_return_t<T>(std::declval<T&>()))) noexcept(noexcept(details::value_or_reference_return_t<T>{std::declval<T&>()}))
{ {
return ptr_; return ptr_;
} }
@ -145,18 +139,10 @@ public:
not_null& operator-=(std::ptrdiff_t) = delete; not_null& operator-=(std::ptrdiff_t) = delete;
void operator[](std::ptrdiff_t) const = delete; void operator[](std::ptrdiff_t) const = delete;
void swap(not_null<T>& other) { std::swap(ptr_, other.ptr_); }
private: private:
T ptr_; T ptr_;
}; };
template <typename T, std::enable_if_t<std::is_move_assignable<T>::value && std::is_move_constructible<T>::value, bool> = true>
void swap(not_null<T>& a, not_null<T>& b)
{
a.swap(b);
}
template <class T> template <class T>
auto make_not_null(T&& t) noexcept auto make_not_null(T&& t) noexcept
{ {
@ -281,19 +267,19 @@ class strict_not_null : public not_null<T>
{ {
public: public:
template <typename U, typename = std::enable_if_t<std::is_convertible<U, T>::value>> template <typename U, typename = std::enable_if_t<std::is_convertible<U, T>::value>>
constexpr explicit strict_not_null(U&& u) noexcept(std::is_nothrow_move_constructible<T>::value) : not_null<T>(std::forward<U>(u)) constexpr explicit strict_not_null(U&& u) : not_null<T>(std::forward<U>(u))
{} {}
template <typename = std::enable_if_t<!std::is_same<std::nullptr_t, T>::value>> template <typename = std::enable_if_t<!std::is_same<std::nullptr_t, T>::value>>
constexpr explicit strict_not_null(T u) noexcept(std::is_nothrow_move_constructible<T>::value) : not_null<T>(std::move(u)) constexpr explicit strict_not_null(T u) : not_null<T>(u)
{} {}
template <typename U, typename = std::enable_if_t<std::is_convertible<U, T>::value>> template <typename U, typename = std::enable_if_t<std::is_convertible<U, T>::value>>
constexpr strict_not_null(const not_null<U>& other) noexcept(std::is_nothrow_move_constructible<T>::value) : not_null<T>(other) constexpr strict_not_null(const not_null<U>& other) : not_null<T>(other)
{} {}
template <typename U, typename = std::enable_if_t<std::is_convertible<U, T>::value>> template <typename U, typename = std::enable_if_t<std::is_convertible<U, T>::value>>
constexpr strict_not_null(const strict_not_null<U>& other) noexcept(std::is_nothrow_move_constructible<T>::value) : not_null<T>(other) constexpr strict_not_null(const strict_not_null<U>& other) : not_null<T>(other)
{} {}
// To avoid invalidating the "not null" invariant, the contained pointer is actually copied // To avoid invalidating the "not null" invariant, the contained pointer is actually copied

View File

@ -17,10 +17,10 @@
#ifndef GSL_SPAN_H #ifndef GSL_SPAN_H
#define GSL_SPAN_H #define GSL_SPAN_H
#include "./assert" // for Expects #include "assert" // for Expects
#include "./byte" // for gsl::impl::byte #include "byte" // for byte
#include "./span_ext" // for span specialization of gsl::at and other span-related extensions #include "span_ext" // for span specialization of gsl::at and other span-related extensions
#include "./util" // for narrow_cast #include "util" // for narrow_cast
#include <array> // for array #include <array> // for array
#include <cstddef> // for ptrdiff_t, size_t, nullptr_t #include <cstddef> // for ptrdiff_t, size_t, nullptr_t
@ -140,9 +140,7 @@ namespace details
constexpr span_iterator(pointer begin, pointer end, pointer current) constexpr span_iterator(pointer begin, pointer end, pointer current)
: begin_(begin), end_(end), current_(current) : begin_(begin), end_(end), current_(current)
{ {}
Expects(begin_ <= current_ && current <= end_);
}
constexpr operator span_iterator<const Type>() const noexcept constexpr operator span_iterator<const Type>() const noexcept
{ {
@ -151,18 +149,21 @@ namespace details
constexpr reference operator*() const noexcept constexpr reference operator*() const noexcept
{ {
Expects(current_ != end_); Expects(begin_ && end_);
Expects(begin_ <= current_ && current_ < end_);
return *current_; return *current_;
} }
constexpr pointer operator->() const noexcept constexpr pointer operator->() const noexcept
{ {
Expects(current_ != end_); Expects(begin_ && end_);
Expects(begin_ <= current_ && current_ < end_);
return current_; return current_;
} }
constexpr span_iterator& operator++() noexcept constexpr span_iterator& operator++() noexcept
{ {
Expects(current_ != end_); Expects(begin_ && current_ && end_);
Expects(current_ < end_);
// clang-format off // clang-format off
GSL_SUPPRESS(bounds.1) // NO-FORMAT: attribute GSL_SUPPRESS(bounds.1) // NO-FORMAT: attribute
// clang-format on // clang-format on
@ -179,7 +180,8 @@ namespace details
constexpr span_iterator& operator--() noexcept constexpr span_iterator& operator--() noexcept
{ {
Expects(begin_ != current_); Expects(begin_ && end_);
Expects(begin_ < current_);
--current_; --current_;
return *this; return *this;
} }
@ -582,8 +584,6 @@ public:
template <std::size_t Count> template <std::size_t Count>
constexpr span<element_type, Count> first() const noexcept constexpr span<element_type, Count> first() const noexcept
{ {
static_assert(Extent == dynamic_extent || Count <= Extent,
"first() cannot extract more elements from a span than it contains.");
Expects(Count <= size()); Expects(Count <= size());
return span<element_type, Count>{data(), Count}; return span<element_type, Count>{data(), Count};
} }
@ -594,8 +594,6 @@ public:
// clang-format on // clang-format on
constexpr span<element_type, Count> last() const noexcept constexpr span<element_type, Count> last() const noexcept
{ {
static_assert(Extent == dynamic_extent || Count <= Extent,
"last() cannot extract more elements from a span than it contains.");
Expects(Count <= size()); Expects(Count <= size());
return span<element_type, Count>{data() + (size() - Count), Count}; return span<element_type, Count>{data() + (size() - Count), Count};
} }
@ -607,9 +605,6 @@ public:
constexpr auto subspan() const noexcept -> constexpr auto subspan() const noexcept ->
typename details::calculate_subspan_type<ElementType, Extent, Offset, Count>::type typename details::calculate_subspan_type<ElementType, Extent, Offset, Count>::type
{ {
static_assert(Extent == dynamic_extent || (Extent >= Offset && (Count == dynamic_extent ||
Count <= Extent - Offset)),
"subspan() cannot extract more elements from a span than it contains.");
Expects((size() >= Offset) && (Count == dynamic_extent || (Count <= size() - Offset))); Expects((size() >= Offset) && (Count == dynamic_extent || (Count <= size() - Offset)));
using type = using type =
typename details::calculate_subspan_type<ElementType, Extent, Offset, Count>::type; typename details::calculate_subspan_type<ElementType, Extent, Offset, Count>::type;
@ -824,28 +819,28 @@ namespace details
// [span.objectrep], views of object representation // [span.objectrep], views of object representation
template <class ElementType, std::size_t Extent> template <class ElementType, std::size_t Extent>
span<const gsl::impl::byte, details::calculate_byte_size<ElementType, Extent>::value> span<const byte, details::calculate_byte_size<ElementType, Extent>::value>
as_bytes(span<ElementType, Extent> s) noexcept as_bytes(span<ElementType, Extent> s) noexcept
{ {
using type = span<const gsl::impl::byte, details::calculate_byte_size<ElementType, Extent>::value>; using type = span<const byte, details::calculate_byte_size<ElementType, Extent>::value>;
// clang-format off // clang-format off
GSL_SUPPRESS(type.1) // NO-FORMAT: attribute GSL_SUPPRESS(type.1) // NO-FORMAT: attribute
// clang-format on // clang-format on
return type{reinterpret_cast<const gsl::impl::byte*>(s.data()), s.size_bytes()}; return type{reinterpret_cast<const byte*>(s.data()), s.size_bytes()};
} }
template <class ElementType, std::size_t Extent, template <class ElementType, std::size_t Extent,
std::enable_if_t<!std::is_const<ElementType>::value, int> = 0> std::enable_if_t<!std::is_const<ElementType>::value, int> = 0>
span<gsl::impl::byte, details::calculate_byte_size<ElementType, Extent>::value> span<byte, details::calculate_byte_size<ElementType, Extent>::value>
as_writable_bytes(span<ElementType, Extent> s) noexcept as_writable_bytes(span<ElementType, Extent> s) noexcept
{ {
using type = span<gsl::impl::byte, details::calculate_byte_size<ElementType, Extent>::value>; using type = span<byte, details::calculate_byte_size<ElementType, Extent>::value>;
// clang-format off // clang-format off
GSL_SUPPRESS(type.1) // NO-FORMAT: attribute GSL_SUPPRESS(type.1) // NO-FORMAT: attribute
// clang-format on // clang-format on
return type{reinterpret_cast<gsl::impl::byte*>(s.data()), s.size_bytes()}; return type{reinterpret_cast<byte*>(s.data()), s.size_bytes()};
} }
} // namespace gsl } // namespace gsl

View File

@ -27,8 +27,8 @@
// //
/////////////////////////////////////////////////////////////////////////////// ///////////////////////////////////////////////////////////////////////////////
#include "./assert" // GSL_KERNEL_MODE #include "assert" // GSL_KERNEL_MODE
#include "./util" // for narrow_cast, narrow #include "util" // for narrow_cast, narrow
#include <cstddef> // for ptrdiff_t, size_t #include <cstddef> // for ptrdiff_t, size_t
#include <utility> #include <utility>
@ -123,14 +123,14 @@ constexpr span<const typename Container::value_type> make_span(const Container&
} }
template <class Ptr> template <class Ptr>
GSL_DEPRECATED("This function is deprecated. See GSL issue #1092.") [[deprecated("This function is deprecated. See GSL issue #1092.")]]
constexpr span<typename Ptr::element_type> make_span(Ptr& cont, std::size_t count) constexpr span<typename Ptr::element_type> make_span(Ptr& cont, std::size_t count)
{ {
return span<typename Ptr::element_type>(cont, count); return span<typename Ptr::element_type>(cont, count);
} }
template <class Ptr> template <class Ptr>
GSL_DEPRECATED("This function is deprecated. See GSL issue #1092.") [[deprecated("This function is deprecated. See GSL issue #1092.")]]
constexpr span<typename Ptr::element_type> make_span(Ptr& cont) constexpr span<typename Ptr::element_type> make_span(Ptr& cont)
{ {
return span<typename Ptr::element_type>(cont); return span<typename Ptr::element_type>(cont);

4
include/gsl/string_span Normal file
View File

@ -0,0 +1,4 @@
#pragma once
#pragma message( \
"This header will soon be removed. Use <gsl/zstring> instead of <gsl/string_span>")
#include "zstring"

View File

@ -17,7 +17,7 @@
#ifndef GSL_UTIL_H #ifndef GSL_UTIL_H
#define GSL_UTIL_H #define GSL_UTIL_H
#include "./assert" // for Expects #include "assert" // for Expects
#include <array> #include <array>
#include <cstddef> // for ptrdiff_t, size_t #include <cstddef> // for ptrdiff_t, size_t
@ -60,32 +60,6 @@
#define GSL_INLINE #define GSL_INLINE
#endif #endif
#if defined(__has_cpp_attribute)
#if __has_cpp_attribute(deprecated)
#define GSL_DEPRECATED(msg) [[deprecated(msg)]]
#endif // __has_cpp_attribute(deprecated)
#endif // defined(__has_cpp_attribute)
#if !defined(GSL_DEPRECATED)
#if defined(__cplusplus)
#if __cplusplus >= 201309L
#define GSL_DEPRECATED(msg) [[deprecated(msg)]]
#endif // __cplusplus >= 201309L
#endif // defined(__cplusplus)
#endif // !defined(GSL_DEPRECATED)
#if !defined(GSL_DEPRECATED)
#if defined(_MSC_VER)
#define GSL_DEPRECATED(msg) __declspec(deprecated(msg))
#elif defined(__GNUC__)
#define GSL_DEPRECATED(msg) __attribute__((deprecated(msg)))
#endif // defined(_MSC_VER)
#endif // !defined(GSL_DEPRECATED)
#if !defined(GSL_DEPRECATED)
#define GSL_DEPRECATED(msg)
#endif // !defined(GSL_DEPRECATED)
namespace gsl namespace gsl
{ {
// //
@ -172,9 +146,6 @@ GSL_SUPPRESS(bounds.1) // NO-FORMAT: attribute
return *(cont.begin() + i); return *(cont.begin() + i);
} }
template <class T, std::enable_if_t<std::is_move_assignable<T>::value && std::is_move_constructible<T>::value>>
void swap(T& a, T& b) { std::swap(a, b); }
#if defined(__cpp_lib_span) && __cpp_lib_span >= 202002L #if defined(__cpp_lib_span) && __cpp_lib_span >= 202002L
template <class T, std::size_t extent = std::dynamic_extent> template <class T, std::size_t extent = std::dynamic_extent>
constexpr auto at(std::span<T, extent> sp, const index i) -> decltype(sp[sp.size()]) constexpr auto at(std::span<T, extent> sp, const index i) -> decltype(sp[sp.size()])

View File

@ -17,7 +17,7 @@
#ifndef GSL_ZSTRING_H #ifndef GSL_ZSTRING_H
#define GSL_ZSTRING_H #define GSL_ZSTRING_H
#include "./span_ext" // for dynamic_extent #include "span_ext" // for dynamic_extent
#include <cstddef> // for size_t, nullptr_t #include <cstddef> // for size_t, nullptr_t

View File

@ -110,7 +110,6 @@ if(MSVC) # MSVC or simulating MSVC
-Wno-shift-sign-overflow # GTest gtest-port.h -Wno-shift-sign-overflow # GTest gtest-port.h
-Wno-undef # GTest -Wno-undef # GTest
-Wno-used-but-marked-unused # GTest EXPECT_DEATH -Wno-used-but-marked-unused # GTest EXPECT_DEATH
-Wno-switch-default # GTest EXPECT_DEATH
$<$<EQUAL:${GSL_CXX_STANDARD},14>: # no support for [[maybe_unused]] $<$<EQUAL:${GSL_CXX_STANDARD},14>: # no support for [[maybe_unused]]
-Wno-unused-member-function -Wno-unused-member-function
-Wno-unused-variable -Wno-unused-variable
@ -206,7 +205,6 @@ add_executable(gsl_tests
byte_tests.cpp byte_tests.cpp
notnull_tests.cpp notnull_tests.cpp
owner_tests.cpp owner_tests.cpp
pointers_tests.cpp
span_compatibility_tests.cpp span_compatibility_tests.cpp
span_ext_tests.cpp span_ext_tests.cpp
span_tests.cpp span_tests.cpp

View File

@ -1,10 +1,10 @@
cmake_minimum_required(VERSION 3.13) cmake_minimum_required(VERSION 3.0.2)
project(googletest-download NONE) project(googletest-download NONE)
include(ExternalProject) include(ExternalProject)
ExternalProject_Add(googletest ExternalProject_Add(googletest
GIT_REPOSITORY https://github.com/google/googletest.git GIT_REPOSITORY https://github.com/google/googletest.git
GIT_TAG v1.14.0 GIT_TAG 1b18723e874b256c1e39378c6774a90701d70f7a
SOURCE_DIR "${CMAKE_CURRENT_BINARY_DIR}/googletest-src" SOURCE_DIR "${CMAKE_CURRENT_BINARY_DIR}/googletest-src"
BINARY_DIR "${CMAKE_CURRENT_BINARY_DIR}/googletest-build" BINARY_DIR "${CMAKE_CURRENT_BINARY_DIR}/googletest-build"
CONFIGURE_COMMAND "" CONFIGURE_COMMAND ""

View File

@ -188,7 +188,7 @@ TEST(algorithm_tests, incompatible_type)
span<int> src_span_dyn(src); span<int> src_span_dyn(src);
span<int, 4> src_span_static(src); span<int, 4> src_span_static(src);
span<int*> dst_span_dyn(dst); span<int*> dst_span_dyn(dst);
span<int*, 4> dst_span_static(gsl::make_span(dst)); span<int*, 4> dst_span_static(dst);
// every line should produce a compilation error // every line should produce a compilation error
copy(src_span_dyn, dst_span_dyn); copy(src_span_dyn, dst_span_dyn);

View File

@ -16,12 +16,8 @@
#include <gtest/gtest.h> #include <gtest/gtest.h>
#define GSL_USE_STD_BYTE 0
#include <gsl/byte> // for to_byte, to_integer, byte, operator&, ope... #include <gsl/byte> // for to_byte, to_integer, byte, operator&, ope...
#include <type_traits>
#include <utility>
using namespace std; using namespace std;
using namespace gsl; using namespace gsl;
@ -37,28 +33,28 @@ int modify_both(gsl::byte& b, int& i)
TEST(byte_tests, construction) TEST(byte_tests, construction)
{ {
{ {
const gsl::byte b = static_cast<gsl::byte>(4); const byte b = static_cast<byte>(4);
EXPECT_TRUE(static_cast<unsigned char>(b) == 4); EXPECT_TRUE(static_cast<unsigned char>(b) == 4);
} }
{ {
const gsl::byte b = gsl::byte(12); const byte b = byte(12);
EXPECT_TRUE(static_cast<unsigned char>(b) == 12); EXPECT_TRUE(static_cast<unsigned char>(b) == 12);
} }
{ {
const gsl::byte b = to_byte<12>(); const byte b = to_byte<12>();
EXPECT_TRUE(static_cast<unsigned char>(b) == 12); EXPECT_TRUE(static_cast<unsigned char>(b) == 12);
} }
{ {
const unsigned char uc = 12; const unsigned char uc = 12;
const gsl::byte b = to_byte(uc); const byte b = to_byte(uc);
EXPECT_TRUE(static_cast<unsigned char>(b) == 12); EXPECT_TRUE(static_cast<unsigned char>(b) == 12);
} }
#if defined(__cplusplus) && (__cplusplus >= 201703L) #if defined(__cplusplus) && (__cplusplus >= 201703L)
{ {
const gsl::byte b{14}; const byte b{14};
EXPECT_TRUE(static_cast<unsigned char>(b) == 14); EXPECT_TRUE(static_cast<unsigned char>(b) == 14);
} }
#endif #endif
@ -67,16 +63,14 @@ TEST(byte_tests, construction)
to_byte(char{}); to_byte(char{});
to_byte(3); to_byte(3);
to_byte(3u); to_byte(3u);
to_byte<-1>();
to_byte<256u>();
#endif #endif
} }
TEST(byte_tests, bitwise_operations) TEST(byte_tests, bitwise_operations)
{ {
const gsl::byte b = to_byte<0xFF>(); const byte b = to_byte<0xFF>();
gsl::byte a = to_byte<0x00>(); byte a = to_byte<0x00>();
EXPECT_TRUE((b | a) == to_byte<0xFF>()); EXPECT_TRUE((b | a) == to_byte<0xFF>());
EXPECT_TRUE(a == to_byte<0x00>()); EXPECT_TRUE(a == to_byte<0x00>());
@ -110,7 +104,7 @@ TEST(byte_tests, bitwise_operations)
TEST(byte_tests, to_integer) TEST(byte_tests, to_integer)
{ {
const gsl::byte b = to_byte<0x12>(); const byte b = to_byte<0x12>();
EXPECT_TRUE(0x12 == gsl::to_integer<char>(b)); EXPECT_TRUE(0x12 == gsl::to_integer<char>(b));
EXPECT_TRUE(0x12 == gsl::to_integer<short>(b)); EXPECT_TRUE(0x12 == gsl::to_integer<short>(b));
@ -129,50 +123,12 @@ TEST(byte_tests, to_integer)
TEST(byte_tests, aliasing) TEST(byte_tests, aliasing)
{ {
int i{0}; int i{0};
const int res = modify_both(reinterpret_cast<gsl::byte&>(i), i); const int res = modify_both(reinterpret_cast<byte&>(i), i);
EXPECT_TRUE(res == i); EXPECT_TRUE(res == i);
} }
#if __cplusplus >= 201703l
using std::void_t;
#else // __cplusplus >= 201703l
template <class...>
using void_t = void;
#endif // __cplusplus < 201703l
template <typename U, typename = void>
static constexpr bool LShiftCompilesFor = false;
template <typename U>
static constexpr bool LShiftCompilesFor<
U, void_t<decltype(gsl::operator<< <float>(declval<gsl::byte>(), declval<U>()))>> = true;
static_assert(!LShiftCompilesFor<float>, "!LShiftCompilesFor<float>");
template <typename U, typename = void>
static constexpr bool RShiftCompilesFor = false;
template <typename U>
static constexpr bool RShiftCompilesFor<
U, void_t<decltype(gsl::operator>> <U>(declval<gsl::byte>(), declval<U>()))>> = true;
static_assert(!RShiftCompilesFor<float>, "!RShiftCompilesFor<float>");
template <typename U, typename = void>
static constexpr bool LShiftAssignCompilesFor = false;
template <typename U>
static constexpr bool LShiftAssignCompilesFor<
U, void_t<decltype(gsl::operator<<= <U>(declval<gsl::byte&>(), declval<U>()))>> = true;
static_assert(!LShiftAssignCompilesFor<float>, "!LShiftAssignCompilesFor<float>");
template <typename U, typename = void>
static constexpr bool RShiftAssignCompilesFor = false;
template <typename U>
static constexpr bool RShiftAssignCompilesFor<
U, void_t<decltype(gsl::operator>>= <U>(declval<gsl::byte&>(), declval<U>()))>> = true;
static_assert(!RShiftAssignCompilesFor<float>, "!RShiftAssignCompilesFor<float>");
template <typename U, typename = void>
static constexpr bool ToIntegerCompilesFor = false;
template <typename U>
static constexpr bool
ToIntegerCompilesFor<U, void_t<decltype(gsl::to_integer<U>(gsl::byte{}))>> = true;
static_assert(!ToIntegerCompilesFor<float>, "!ToIntegerCompilesFor<float>");
} // namespace } // namespace
#ifdef CONFIRM_COMPILATION_ERRORS
copy(src_span_static, dst_span_static);
#endif

View File

@ -18,25 +18,17 @@
#include <gsl/pointers> // for not_null, operator<, operator<=, operator> #include <gsl/pointers> // for not_null, operator<, operator<=, operator>
#include <algorithm> // for addressof #include <algorithm> // for addressof
#include <cstdint> // for uint16_t #include <cstdint> // for uint16_t
#include <memory> // for shared_ptr, make_shared, operator<, opera... #include <memory> // for shared_ptr, make_shared, operator<, opera...
#include <sstream> // for operator<<, ostringstream, basic_ostream:... #include <sstream> // for operator<<, ostringstream, basic_ostream:...
#include <string> // for basic_string, operator==, string, operator<< #include <string> // for basic_string, operator==, string, operator<<
#include <type_traits> // for declval #include <typeinfo> // for type_info
#include <typeinfo> // for type_info #include <variant> // for variant, monostate, get
#include <variant> // for variant, monostate, get
#include "deathTestCommon.h" #include "deathTestCommon.h"
using namespace gsl; using namespace gsl;
#if __cplusplus >= 201703l
using std::void_t;
#else // __cplusplus >= 201703l
template <class...>
using void_t = void;
#endif // __cplusplus < 201703l
struct MyBase struct MyBase
{ {
}; };
@ -149,39 +141,16 @@ bool helper_const(not_null<const int*> p) { return *p == 12; }
int* return_pointer() { return nullptr; } int* return_pointer() { return nullptr; }
} // namespace } // namespace
template <typename U, typename = void>
static constexpr bool CtorCompilesFor_A = false;
template <typename U>
static constexpr bool
CtorCompilesFor_A<U, void_t<decltype(gsl::not_null<void*>{std::declval<U>()})>> = true;
template <typename U, int N, typename = void>
static constexpr bool CtorCompilesFor_B = false;
template <typename U, int N>
static constexpr bool CtorCompilesFor_B<U, N, void_t<decltype(gsl::not_null<U>{N})>> = true;
template <typename U, typename = void>
static constexpr bool DefaultCtorCompilesFor = false;
template <typename U>
static constexpr bool DefaultCtorCompilesFor<U, void_t<decltype(gsl::not_null<U>{})>> = true;
template <typename U, typename = void>
static constexpr bool CtorCompilesFor_C = false;
template <typename U>
static constexpr bool
CtorCompilesFor_C<U, void_t<decltype(gsl::not_null<U*>{std::declval<std::unique_ptr<U>>()})>> =
true;
TEST(notnull_tests, TestNotNullConstructors) TEST(notnull_tests, TestNotNullConstructors)
{ {
{ {
static_assert(CtorCompilesFor_A<void*>, "CtorCompilesFor_A<void*>");
static_assert(!CtorCompilesFor_A<std::nullptr_t>, "!CtorCompilesFor_A<std::nullptr_t>");
static_assert(!CtorCompilesFor_B<void*, 0>, "!CtorCompilesFor_B<void*, 0>");
static_assert(!DefaultCtorCompilesFor<void*>, "!DefaultCtorCompilesFor<void*>");
static_assert(!CtorCompilesFor_C<int>, "CtorCompilesFor_C<int>");
#ifdef CONFIRM_COMPILATION_ERRORS #ifdef CONFIRM_COMPILATION_ERRORS
not_null<int*> p = nullptr; // yay...does not compile!
not_null<std::vector<char>*> p1 = 0; // yay...does not compile!
not_null<int*> p2; // yay...does not compile!
std::unique_ptr<int> up = std::make_unique<int>(120);
not_null<int*> p3 = up;
// Forbid non-nullptr assignable types // Forbid non-nullptr assignable types
not_null<std::vector<int>> f(std::vector<int>{1}); not_null<std::vector<int>> f(std::vector<int>{1});
not_null<int> z(10); not_null<int> z(10);
@ -209,14 +178,6 @@ TEST(notnull_tests, TestNotNullConstructors)
EXPECT_DEATH((not_null<decltype(pi)>(pi)), expected); EXPECT_DEATH((not_null<decltype(pi)>(pi)), expected);
} }
{
// from unique pointer
not_null<std::unique_ptr<int>> x(
std::make_unique<int>(10)); // unique_ptr<int> is nullptr assignable
EXPECT_DEATH((not_null<std::unique_ptr<int>>(std::unique_ptr<int>{})), expected);
}
{ {
// from pointer to local // from pointer to local
int t = 42; int t = 42;
@ -307,27 +268,6 @@ TEST(notnull_tests, TestNotNullostream)
ostream_helper<std::string>("string"); ostream_helper<std::string>("string");
} }
template <typename U, typename V, typename = void>
static constexpr bool AssignmentCompilesFor = false;
template <typename U, typename V>
static constexpr bool
AssignmentCompilesFor<U, V,
void_t<decltype(std::declval<gsl::not_null<U*>&>().operator=(
std::declval<gsl::not_null<V*>&>()))>> = true;
template <typename U, typename V, typename = void>
static constexpr bool SCastCompilesFor = false;
template <typename U, typename V>
static constexpr bool
SCastCompilesFor<U, V, void_t<decltype(static_cast<U*>(std::declval<gsl::not_null<V*>&>()))>> =
true;
template <typename U, typename V, typename = void>
static constexpr bool RCastCompilesFor = false;
template <typename U, typename V>
static constexpr bool RCastCompilesFor<
U, V, void_t<decltype(reinterpret_cast<U*>(std::declval<gsl::not_null<V*>&>()))>> = true;
TEST(notnull_tests, TestNotNullCasting) TEST(notnull_tests, TestNotNullCasting)
{ {
MyBase base; MyBase base;
@ -340,30 +280,15 @@ TEST(notnull_tests, TestNotNullCasting)
q = p; // allowed with heterogeneous copy ctor q = p; // allowed with heterogeneous copy ctor
EXPECT_TRUE(q == p); EXPECT_TRUE(q == p);
static_assert(AssignmentCompilesFor<MyBase, MyDerived>, #ifdef CONFIRM_COMPILATION_ERRORS
"AssignmentCompilesFor<MyBase, MyDerived>"); q = u; // no viable conversion possible between MyBase* and Unrelated*
static_assert(!AssignmentCompilesFor<MyBase, Unrelated>, p = q; // not possible to implicitly convert MyBase* to MyDerived*
"!AssignmentCompilesFor<MyBase, Unrelated>");
static_assert(!AssignmentCompilesFor<Unrelated, MyDerived>,
"!AssignmentCompilesFor<Unrelated, MyDerived>");
static_assert(!AssignmentCompilesFor<MyDerived, MyBase>,
"!AssignmentCompilesFor<MyDerived, MyBase>");
static_assert(SCastCompilesFor<MyDerived, MyDerived>, "SCastCompilesFor<MyDerived, MyDerived>");
static_assert(SCastCompilesFor<MyBase, MyDerived>, "SCastCompilesFor<MyBase, MyDerived>");
static_assert(!SCastCompilesFor<MyDerived, MyBase>, "!SCastCompilesFor<MyDerived, MyBase>");
static_assert(!SCastCompilesFor<Unrelated, MyDerived>,
"!SCastCompilesFor<Unrelated, MyDerived>");
static_assert(!RCastCompilesFor<MyDerived, MyDerived>,
"!SCastCompilesFor<MyDerived, MyDerived>");
static_assert(!RCastCompilesFor<Unrelated, MyDerived>,
"!SCastCompilesFor<Unrelated, MyDerived>");
not_null<Unrelated*> r = p;
not_null<Unrelated*> s = reinterpret_cast<Unrelated*>(p);
#endif
not_null<Unrelated*> t(reinterpret_cast<Unrelated*>(p.get())); not_null<Unrelated*> t(reinterpret_cast<Unrelated*>(p.get()));
EXPECT_TRUE(reinterpret_cast<void*>(p.get()) == reinterpret_cast<void*>(t.get())); EXPECT_TRUE(reinterpret_cast<void*>(p.get()) == reinterpret_cast<void*>(t.get()));
(void) static_cast<MyDerived*>(p);
(void) static_cast<MyBase*>(p);
} }
TEST(notnull_tests, TestNotNullAssignment) TEST(notnull_tests, TestNotNullAssignment)
@ -505,18 +430,6 @@ TEST(notnull_tests, TestNotNullCustomPtrComparison)
#if defined(__cplusplus) && (__cplusplus >= 201703L) #if defined(__cplusplus) && (__cplusplus >= 201703L)
template <typename U, typename = void>
static constexpr bool TypeDeductionCtorCompilesFor = false;
template <typename U>
static constexpr bool
TypeDeductionCtorCompilesFor<U, void_t<decltype(not_null{std::declval<U>()})>> = true;
template <typename U, typename = void>
static constexpr bool TypeDeductionHelperCompilesFor = false;
template <typename U>
static constexpr bool
TypeDeductionHelperCompilesFor<U, void_t<decltype(helper(not_null{std::declval<U>()}))>> = true;
TEST(notnull_tests, TestNotNullConstructorTypeDeduction) TEST(notnull_tests, TestNotNullConstructorTypeDeduction)
{ {
{ {
@ -533,9 +446,9 @@ TEST(notnull_tests, TestNotNullConstructorTypeDeduction)
const int i = 42; const int i = 42;
not_null x{&i}; not_null x{&i};
static_assert(TypeDeductionHelperCompilesFor<int*>, "TypeDeductionHelperCompilesFor<int*>"); #ifdef CONFIRM_COMPILATION_ERRORS
static_assert(!TypeDeductionHelperCompilesFor<const int*>, helper(not_null{&i});
"!TypeDeductionHelperCompilesFor<const int*>"); #endif
helper_const(not_null{&i}); helper_const(not_null{&i});
EXPECT_TRUE(*x == 42); EXPECT_TRUE(*x == 42);
@ -557,6 +470,9 @@ TEST(notnull_tests, TestNotNullConstructorTypeDeduction)
const int* p = &i; const int* p = &i;
not_null x{p}; not_null x{p};
#ifdef CONFIRM_COMPILATION_ERRORS
helper(not_null{p});
#endif
helper_const(not_null{p}); helper_const(not_null{p});
EXPECT_TRUE(*x == 42); EXPECT_TRUE(*x == 42);
@ -591,15 +507,12 @@ TEST(notnull_tests, TestNotNullConstructorTypeDeduction)
EXPECT_DEATH(helper_const(not_null{p}), expected); EXPECT_DEATH(helper_const(not_null{p}), expected);
} }
static_assert(TypeDeductionCtorCompilesFor<void*>, "TypeDeductionCtorCompilesFor<void*>"); #ifdef CONFIRM_COMPILATION_ERRORS
#if defined(_MSC_VER) && !defined(__clang__) {
// Fails on gcc, clang, xcode, VS clang with not_null x{nullptr};
// "error : no type named 'type' in 'std::enable_if<false>'; 'enable_if' cannot be used to helper(not_null{nullptr});
// disable this declaration" helper_const(not_null{nullptr});
static_assert(!TypeDeductionCtorCompilesFor<std::nullptr_t>, }
"!TypeDeductionCtorCompilesFor<std::nullptr_t>");
static_assert(!TypeDeductionHelperCompilesFor<std::nullptr_t>,
"!TypeDeductionHelperCompilesFor<std::nullptr_t>");
#endif #endif
} }
@ -615,11 +528,6 @@ TEST(notnull_tests, TestVariantEmplace)
} }
#endif // #if defined(__cplusplus) && (__cplusplus >= 201703L) #endif // #if defined(__cplusplus) && (__cplusplus >= 201703L)
template <typename U, typename = void>
static constexpr bool HelperCompilesFor = false;
template <typename U>
static constexpr bool HelperCompilesFor<U, void_t<decltype(helper(std::declval<U>()))>> = true;
TEST(notnull_tests, TestMakeNotNull) TEST(notnull_tests, TestMakeNotNull)
{ {
{ {
@ -636,8 +544,9 @@ TEST(notnull_tests, TestMakeNotNull)
const int i = 42; const int i = 42;
const auto x = make_not_null(&i); const auto x = make_not_null(&i);
static_assert(HelperCompilesFor<gsl::not_null<int*>>, #ifdef CONFIRM_COMPILATION_ERRORS
"HelperCompilesFor<gsl::not_null<int*>>"); helper(make_not_null(&i));
#endif
helper_const(make_not_null(&i)); helper_const(make_not_null(&i));
EXPECT_TRUE(*x == 42); EXPECT_TRUE(*x == 42);
@ -659,8 +568,9 @@ TEST(notnull_tests, TestMakeNotNull)
const int* p = &i; const int* p = &i;
const auto x = make_not_null(p); const auto x = make_not_null(p);
static_assert(!HelperCompilesFor<gsl::not_null<const int*>>, #ifdef CONFIRM_COMPILATION_ERRORS
"!HelperCompilesFor<gsl::not_null<const int*>>"); helper(make_not_null(p));
#endif
helper_const(make_not_null(p)); helper_const(make_not_null(p));
EXPECT_TRUE(*x == 42); EXPECT_TRUE(*x == 42);

View File

@ -17,7 +17,6 @@
#include <gtest/gtest.h> #include <gtest/gtest.h>
#include <gsl/pointers> // for owner #include <gsl/pointers> // for owner
#include <type_traits> // for declval
using namespace gsl; using namespace gsl;
@ -33,18 +32,12 @@ TEST(owner_tests, basic_test)
delete p; delete p;
} }
#if __cplusplus >= 201703l TEST(owner_tests, check_pointer_constraint)
using std::void_t; {
#else // __cplusplus >= 201703l #ifdef CONFIRM_COMPILATION_ERRORS
template <class...> {
using void_t = void; owner<int> integerTest = 10;
#endif // __cplusplus < 201703l owner<std::shared_ptr<int>> sharedPtrTest(new int(10));
}
template <typename U, typename = void> #endif
static constexpr bool OwnerCompilesFor = false; }
template <typename U>
static constexpr bool OwnerCompilesFor<U, void_t<decltype(gsl::owner<U>{})>> =
true;
static_assert(OwnerCompilesFor<int*>, "OwnerCompilesFor<int*>");
static_assert(!OwnerCompilesFor<int>, "!OwnerCompilesFor<int>");
static_assert(!OwnerCompilesFor<std::shared_ptr<int>>, "!OwnerCompilesFor<std::shared_ptr<int>>");

View File

@ -1,97 +0,0 @@
#include <gtest/gtest.h>
#include <gsl/pointers>
#include <memory>
#include <type_traits>
#include <utility>
#if __cplusplus >= 201703l
using std::void_t;
#else // __cplusplus >= 201703l
template <class...>
using void_t = void;
#endif // __cplusplus < 201703l
namespace
{
// Custom pointer type that can be used for gsl::not_null, but for which these cannot be swapped.
struct NotMoveAssignableCustomPtr
{
NotMoveAssignableCustomPtr() = default;
NotMoveAssignableCustomPtr(const NotMoveAssignableCustomPtr&) = default;
NotMoveAssignableCustomPtr& operator=(const NotMoveAssignableCustomPtr&) = default;
NotMoveAssignableCustomPtr(NotMoveAssignableCustomPtr&&) = default;
NotMoveAssignableCustomPtr& operator=(NotMoveAssignableCustomPtr&&) = delete;
bool operator!=(std::nullptr_t) const { return true; }
int dummy{}; // Without this clang warns, that NotMoveAssignableCustomPtr() is unneeded
};
template <typename U, typename = void>
static constexpr bool SwapCompilesFor = false;
template <typename U>
static constexpr bool
SwapCompilesFor<U, void_t<decltype(gsl::swap<U>(std::declval<gsl::not_null<U>&>(),
std::declval<gsl::not_null<U>&>()))>> = true;
TEST(pointers_test, swap)
{
// taken from gh-1129:
{
gsl::not_null<std::unique_ptr<int>> a(std::make_unique<int>(0));
gsl::not_null<std::unique_ptr<int>> b(std::make_unique<int>(1));
EXPECT_TRUE(*a == 0);
EXPECT_TRUE(*b == 1);
gsl::swap(a, b);
EXPECT_TRUE(*a == 1);
EXPECT_TRUE(*b == 0);
// Make sure our custom ptr can be used with not_null. The shared_pr is to prevent "unused"
// compiler warnings.
const auto shared_custom_ptr{std::make_shared<NotMoveAssignableCustomPtr>()};
gsl::not_null<NotMoveAssignableCustomPtr> c{*shared_custom_ptr};
EXPECT_TRUE(c.get() != nullptr);
}
{
gsl::strict_not_null<std::unique_ptr<int>> a{std::make_unique<int>(0)};
gsl::strict_not_null<std::unique_ptr<int>> b{std::make_unique<int>(1)};
EXPECT_TRUE(*a == 0);
EXPECT_TRUE(*b == 1);
gsl::swap(a, b);
EXPECT_TRUE(*a == 1);
EXPECT_TRUE(*b == 0);
}
{
gsl::not_null<std::unique_ptr<int>> a{std::make_unique<int>(0)};
gsl::strict_not_null<std::unique_ptr<int>> b{std::make_unique<int>(1)};
EXPECT_TRUE(*a == 0);
EXPECT_TRUE(*b == 1);
gsl::swap(a, b);
EXPECT_TRUE(*a == 1);
EXPECT_TRUE(*b == 0);
}
static_assert(!SwapCompilesFor<NotMoveAssignableCustomPtr>,
"!SwapCompilesFor<NotMoveAssignableCustomPtr>");
}
TEST(pointers_test, member_types)
{
static_assert(std::is_same<gsl::not_null<int*>::element_type, int*>::value,
"check member type: element_type");
}
} // namespace

View File

@ -1005,18 +1005,12 @@ static_assert(std::is_convertible<const std::array<int, 3>&, gsl::span<const int
"std::is_convertible<const std::array<int, 3>&, gsl::span<const int>>"); "std::is_convertible<const std::array<int, 3>&, gsl::span<const int>>");
#if __cplusplus >= 201703l #if __cplusplus >= 201703l
using std::void_t;
#else // __cplusplus >= 201703l
template <class...>
using void_t = void;
#endif // __cplusplus < 201703l
template <typename U, typename = void> template <typename U, typename = void>
static constexpr bool AsWritableBytesCompilesFor = false; static constexpr bool AsWritableBytesCompilesFor = false;
template <typename U> template <typename U>
static constexpr bool static constexpr bool
AsWritableBytesCompilesFor<U, ::void_t<decltype(as_writable_bytes(declval<U>()))>> = true; AsWritableBytesCompilesFor<U, void_t<decltype(as_writable_bytes(declval<U>()))>> = true;
static_assert(AsWritableBytesCompilesFor<gsl::span<int>>, static_assert(AsWritableBytesCompilesFor<gsl::span<int>>,
"AsWritableBytesCompilesFor<gsl::span<int>>"); "AsWritableBytesCompilesFor<gsl::span<int>>");
@ -1026,3 +1020,4 @@ static_assert(!AsWritableBytesCompilesFor<gsl::span<const int>>,
"!AsWritableBytesCompilesFor<gsl::span<const int>>"); "!AsWritableBytesCompilesFor<gsl::span<const int>>");
static_assert(!AsWritableBytesCompilesFor<gsl::span<const int, 9>>, static_assert(!AsWritableBytesCompilesFor<gsl::span<const int, 9>>,
"!AsWritableBytesCompilesFor<gsl::span<const int, 9>>"); "!AsWritableBytesCompilesFor<gsl::span<const int, 9>>");
#endif // __cplusplus >= 201703l

View File

@ -48,13 +48,6 @@
using namespace gsl; using namespace gsl;
#if __cplusplus >= 201703l
using std::void_t;
#else // __cplusplus >= 201703l
template <class...>
using void_t = void;
#endif // __cplusplus < 201703l
namespace namespace
{ {
@ -69,7 +62,8 @@ struct AddressOverloaded
#if (__cplusplus > 201402L) #if (__cplusplus > 201402L)
[[maybe_unused]] [[maybe_unused]]
#endif #endif
AddressOverloaded operator&() const AddressOverloaded
operator&() const
{ {
return {}; return {};
} }
@ -222,12 +216,6 @@ TEST(span_test, from_pointer_length_constructor)
TEST(span_test, from_pointer_pointer_construction) TEST(span_test, from_pointer_pointer_construction)
{ {
// const auto terminateHandler = std::set_terminate([] {
// std::cerr << "Expected Death. from_pointer_pointer_construction";
// std::abort();
// });
// const auto expected = GetExpectedDeathString(terminateHandler);
int arr[4] = {1, 2, 3, 4}; int arr[4] = {1, 2, 3, 4};
{ {
@ -257,11 +245,19 @@ TEST(span_test, from_pointer_pointer_construction)
EXPECT_TRUE(s.data() == &arr[0]); EXPECT_TRUE(s.data() == &arr[0]);
} }
//{ // this test succeeds on all platforms, gsl::span is more relaxed than std::span where this would be UB // this will fail the std::distance() precondition, which asserts on MSVC debug builds
//{
// auto workaround_macro = [&]() { span<int> s{&arr[1], &arr[0]}; }; // auto workaround_macro = [&]() { span<int> s{&arr[1], &arr[0]}; };
// EXPECT_DEATH(workaround_macro(), expected); // EXPECT_DEATH(workaround_macro(), expected);
//} //}
// this will fail the std::distance() precondition, which asserts on MSVC debug builds
//{
// int* p = nullptr;
// auto workaround_macro = [&]() { span<int> s{&arr[0], p}; };
// EXPECT_DEATH(workaround_macro(), expected);
//}
{ {
int* p = nullptr; int* p = nullptr;
span<int> s{p, p}; span<int> s{p, p};
@ -275,21 +271,19 @@ TEST(span_test, from_pointer_pointer_construction)
EXPECT_TRUE(s.size() == 0); EXPECT_TRUE(s.size() == 0);
EXPECT_TRUE(s.data() == nullptr); EXPECT_TRUE(s.data() == nullptr);
} }
}
template <typename U, typename V, typename = void> // this will fail the std::distance() precondition, which asserts on MSVC debug builds
static constexpr bool CtorCompilesFor = false; //{
template <typename U, typename V> // int* p = nullptr;
static constexpr bool CtorCompilesFor<U, V, void_t<decltype(U{std::declval<V>()})>> = true; // auto workaround_macro = [&]() { span<int> s{&arr[0], p}; };
// EXPECT_DEATH(workaround_macro(), expected);
//}
}
TEST(span_test, from_array_constructor) TEST(span_test, from_array_constructor)
{ {
int arr[5] = {1, 2, 3, 4, 5}; int arr[5] = {1, 2, 3, 4, 5};
static_assert(!CtorCompilesFor<span<int, 6>, int[5]>, "!CtorCompilesFor<span<int, 6>, int[5]>");
static_assert(!CtorCompilesFor<span<int, 0>, int[5]>, "!CtorCompilesFor<span<int, 0>, int[5]>");
static_assert(!CtorCompilesFor<span<int>, int[2][3]>, "!CtorCompilesFor<span<int>, int[2][3]>");
{ {
const span<int> s{arr}; const span<int> s{arr};
EXPECT_TRUE(s.size() == 5); EXPECT_TRUE(s.size() == 5);
@ -304,28 +298,70 @@ TEST(span_test, from_array_constructor)
int arr2d[2][3] = {1, 2, 3, 4, 5, 6}; int arr2d[2][3] = {1, 2, 3, 4, 5, 6};
static_assert(!CtorCompilesFor<span<int, 0>, int[2][3]>, #ifdef CONFIRM_COMPILATION_ERRORS
"!CtorCompilesFor<span<int, 0>, int[2][3]>"); {
static_assert(!CtorCompilesFor<span<int, 6>, int[2][3]>, span<int, 6> s{arr};
"!CtorCompilesFor<span<int, 6>, int[2][3]>"); }
{
span<int, 0> s{arr};
EXPECT_TRUE(s.size() == 0);
EXPECT_TRUE(s.data() == &arr[0]);
}
{
span<int> s{arr2d};
EXPECT_TRUE(s.size() == 6);
EXPECT_TRUE(s.data() == &arr2d[0][0]);
EXPECT_TRUE(s[0] == 1);
EXPECT_TRUE(s[5] == 6);
}
{
span<int, 0> s{arr2d};
EXPECT_TRUE(s.size() == 0);
EXPECT_TRUE(s.data() == &arr2d[0][0]);
}
{
span<int, 6> s{arr2d};
}
#endif
{ {
const span<int[3]> s{std::addressof(arr2d[0]), 1}; const span<int[3]> s{std::addressof(arr2d[0]), 1};
EXPECT_TRUE(s.size() == 1); EXPECT_TRUE(s.size() == 1);
EXPECT_TRUE(s.data() == std::addressof(arr2d[0])); EXPECT_TRUE(s.data() == std::addressof(arr2d[0]));
} }
int arr3d[2][3][2] = {{{1, 2}, {3, 4}, {5, 6}}, {{7, 8}, {9, 10}, {11, 12}}}; int arr3d[2][3][2] = { { {1, 2}, {3, 4}, {5, 6} }, { {7, 8}, {9, 10}, {11, 12} } };
static_assert(!CtorCompilesFor<span<int>, int[2][3][2]>, #ifdef CONFIRM_COMPILATION_ERRORS
"!CtorCompilesFor<span<int>, int[2][3][2]>"); {
static_assert(!CtorCompilesFor<span<int, 0>, int[2][3][2]>, span<int> s{arr3d};
"!CtorCompilesFor<span<int, 0>, int[2][3][2]>"); EXPECT_TRUE(s.size() == 12);
static_assert(!CtorCompilesFor<span<int, 11>, int[2][3][2]>, EXPECT_TRUE(s.data() == &arr3d[0][0][0]);
"!CtorCompilesFor<span<int, 11>, int[2][3][2]>"); EXPECT_TRUE(s[0] == 1);
static_assert(!CtorCompilesFor<span<int, 12>, int[2][3][2]>, EXPECT_TRUE(s[11] == 12);
"!CtorCompilesFor<span<int, 12>, int[2][3][2]>"); }
{
span<int, 0> s{arr3d};
EXPECT_TRUE(s.size() == 0);
EXPECT_TRUE(s.data() == &arr3d[0][0][0]);
}
{
span<int, 11> s{arr3d};
}
{
span<int, 12> s{arr3d};
EXPECT_TRUE(s.size() == 12);
EXPECT_TRUE(s.data() == &arr3d[0][0][0]);
EXPECT_TRUE(s[0] == 1);
EXPECT_TRUE(s[5] == 6);
}
#endif
{ {
const span<int[3][2]> s{std::addressof(arr3d[0]), 1}; const span<int[3][2]> s{std::addressof(arr3d[0]), 1};
EXPECT_TRUE(s.size() == 1); EXPECT_TRUE(s.size() == 1);
@ -353,13 +389,6 @@ TEST(span_test, from_dynamic_array_constructor)
delete[] arr; delete[] arr;
} }
template <typename U, typename V, typename = void>
static constexpr bool ConversionCompilesFor = false;
template <typename U, typename V>
static constexpr bool
ConversionCompilesFor<U, V, void_t<decltype(std::declval<void (*)(U)>()(std::declval<V>()))>> =
true;
TEST(span_test, from_std_array_constructor) TEST(span_test, from_std_array_constructor)
{ {
std::array<int, 4> arr = {1, 2, 3, 4}; std::array<int, 4> arr = {1, 2, 3, 4};
@ -399,31 +428,43 @@ TEST(span_test, from_std_array_constructor)
EXPECT_TRUE(ao_arr.data() == fs.data()); EXPECT_TRUE(ao_arr.data() == fs.data());
} }
static_assert(!CtorCompilesFor<span<int, 2>, std::array<int, 4>&>, #ifdef CONFIRM_COMPILATION_ERRORS
"!CtorCompilesFor<span<int, 2>, std::array<int, 4>&>"); {
static_assert(!CtorCompilesFor<span<const int, 2>, std::array<int, 4>&>, span<int, 2> s{arr};
"!CtorCompilesFor<span<const int, 2>, std::array<int, 4>&>"); EXPECT_TRUE(s.size() == 2);
EXPECT_TRUE(s.data() == arr.data());
static_assert(!CtorCompilesFor<span<int, 0>, std::array<int, 4>&>, span<const int, 2> cs{arr};
"!CtorCompilesFor<span<int, 0>, std::array<int, 4>&>"); EXPECT_TRUE(cs.size() == 2);
static_assert(!CtorCompilesFor<span<const int, 0>, std::array<int, 4>&>, EXPECT_TRUE(cs.data() == arr.data());
"!CtorCompilesFor<span<const int, 0>, std::array<int, 4>&>"); }
static_assert(!CtorCompilesFor<span<int, 5>, std::array<int, 4>&>, {
"!CtorCompilesFor<span<int, 5>, std::array<int, 4>&>"); span<int, 0> s{arr};
EXPECT_TRUE(s.size() == 0);
EXPECT_TRUE(s.data() == arr.data());
#if !defined(_MSC_VER) || (_MSC_VER > 1943) || (__cplusplus >= 201703L) span<const int, 0> cs{arr};
// Fails on "Visual Studio 16 2019/Visual Studio 17 2022, windows-2019/2022, Debug/Release, 14". EXPECT_TRUE(cs.size() == 0);
static_assert(!ConversionCompilesFor<span<int>, std::array<int, 4>>, EXPECT_TRUE(cs.data() == arr.data());
"!ConversionCompilesFor<span<int>, std::array<int, 4>>"); }
{
span<int, 5> s{arr};
}
{
auto get_an_array = []() -> std::array<int, 4> { return {1, 2, 3, 4}; };
auto take_a_span = [](span<int> s) { static_cast<void>(s); };
// try to take a temporary std::array
take_a_span(get_an_array());
}
#endif #endif
{ {
auto get_an_array = []() -> std::array<int, 4> { return {1, 2, 3, 4}; }; auto get_an_array = []() -> std::array<int, 4> { return {1, 2, 3, 4}; };
auto take_a_span = [](span<const int>) {}; auto take_a_span = [](span<const int> s) { static_cast<void>(s); };
// try to take a temporary std::array // try to take a temporary std::array
static_assert(ConversionCompilesFor<span<const int>, std::array<int, 4>>,
"ConversionCompilesFor<span<const int>, std::array<int, 4>>");
take_a_span(get_an_array()); take_a_span(get_an_array());
} }
} }
@ -452,12 +493,23 @@ TEST(span_test, from_const_std_array_constructor)
EXPECT_TRUE(s.data() == ao_arr.data()); EXPECT_TRUE(s.data() == ao_arr.data());
} }
static_assert(!CtorCompilesFor<span<const int, 2>, std::array<int, 4>&>, #ifdef CONFIRM_COMPILATION_ERRORS
"!CtorCompilesFor<span<const int, 2>, std::array<int, 4>&>"); {
static_assert(!CtorCompilesFor<span<const int, 0>, std::array<int, 4>&>, span<const int, 2> s{arr};
"!CtorCompilesFor<span<const int, 0>, std::array<int, 4>&>"); EXPECT_TRUE(s.size() == 2);
static_assert(!CtorCompilesFor<span<int, 5>, std::array<int, 4>&>, EXPECT_TRUE(s.data() == arr.data());
"!CtorCompilesFor<span<int, 5>, std::array<int, 4>&>"); }
{
span<const int, 0> s{arr};
EXPECT_TRUE(s.size() == 0);
EXPECT_TRUE(s.data() == arr.data());
}
{
span<const int, 5> s{arr};
}
#endif
{ {
auto get_an_array = []() -> const std::array<int, 4> { return {1, 2, 3, 4}; }; auto get_an_array = []() -> const std::array<int, 4> { return {1, 2, 3, 4}; };
@ -483,14 +535,27 @@ TEST(span_test, from_std_array_const_constructor)
EXPECT_TRUE(s.data() == arr.data()); EXPECT_TRUE(s.data() == arr.data());
} }
static_assert(!CtorCompilesFor<span<const int, 2>, const std::array<int, 4>&>, #ifdef CONFIRM_COMPILATION_ERRORS
"!CtorCompilesFor<span<const int, 2>, const std::array<int, 4>&>"); {
static_assert(!CtorCompilesFor<span<const int, 0>, const std::array<int, 4>&>, span<const int, 2> s{arr};
"!CtorCompilesFor<span<const int, 0>, const std::array<int, 4>&>"); EXPECT_TRUE(s.size() == 2);
static_assert(!CtorCompilesFor<span<const int, 5>, const std::array<int, 4>&>, EXPECT_TRUE(s.data() == arr.data());
"!CtorCompilesFor<span<const int, 5>, const std::array<int, 4>&>"); }
static_assert(!CtorCompilesFor<span<int, 5>, const std::array<int, 4>&>,
"!CtorCompilesFor<span<int, 5>, const std::array<int, 4>&>"); {
span<const int, 0> s{arr};
EXPECT_TRUE(s.size() == 0);
EXPECT_TRUE(s.data() == arr.data());
}
{
span<const int, 5> s{arr};
}
{
span<int, 4> s{arr};
}
#endif
} }
TEST(span_test, from_container_constructor) TEST(span_test, from_container_constructor)
@ -512,28 +577,32 @@ TEST(span_test, from_container_constructor)
const std::string cstr = "hello"; const std::string cstr = "hello";
{ {
static_assert(CtorCompilesFor<span<char>, std::string&> == (__cplusplus >= 201703L), #ifdef CONFIRM_COMPILATION_ERRORS
"CtorCompilesFor<span<char>, std::string&> == (__cplusplus >= 201703L)"); span<char> s{str};
EXPECT_TRUE(s.size() == str.size());
span<const char> cs{str}; EXPECT_TRUE(s.data() == str.data()));
EXPECT_TRUE(cs.size() == str.size()); #endif
EXPECT_TRUE(cs.data() == str.data()); span<const char> cs{str};
EXPECT_TRUE(cs.size() == str.size());
EXPECT_TRUE(cs.data() == str.data());
} }
{ {
static_assert(!CtorCompilesFor<span<char>, const std::string&>, #ifdef CONFIRM_COMPILATION_ERRORS
"!CtorCompilesFor<span<char>, const std::string&>"); span<char> s{cstr};
#endif
span<const char> cs{cstr}; span<const char> cs{cstr};
EXPECT_TRUE(cs.size() == cstr.size()); EXPECT_TRUE(cs.size() == cstr.size());
EXPECT_TRUE(cs.data() == cstr.data()); EXPECT_TRUE(cs.data() == cstr.data());
} }
#if !defined(_MSC_VER) || (_MSC_VER > 1943) || (__cplusplus >= 201703L) {
// Fails on "Visual Studio 16 2019/Visual Studio 17 2022, windows-2019/2022, Debug/Release, 14". #ifdef CONFIRM_COMPILATION_ERRORS
static_assert(!ConversionCompilesFor<span<int>, std::vector<int>>, auto get_temp_vector = []() -> std::vector<int> { return {}; };
"!ConversionCompilesFor<span<int>, std::vector<int>>"); auto use_span = [](span<int> s) { static_cast<void>(s); };
#endif // !defined(_MSC_VER) || (_MSC_VER > 1942) || (__cplusplus >= 201703L) use_span(get_temp_vector());
#endif
}
{ {
auto get_temp_vector = []() -> std::vector<int> { return {}; }; auto get_temp_vector = []() -> std::vector<int> { return {}; };
@ -541,8 +610,13 @@ TEST(span_test, from_container_constructor)
use_span(get_temp_vector()); use_span(get_temp_vector());
} }
static_assert(!ConversionCompilesFor<span<char>, std::string>, {
"!ConversionCompilesFor<span<char>, std::string>"); #ifdef CONFIRM_COMPILATION_ERRORS
auto get_temp_string = []() -> std::string { return {}; };
auto use_span = [](span<char> s) { static_cast<void>(s); };
use_span(get_temp_string());
#endif
}
{ {
auto get_temp_string = []() -> std::string { return {}; }; auto get_temp_string = []() -> std::string { return {}; };
@ -550,10 +624,13 @@ TEST(span_test, from_container_constructor)
use_span(get_temp_string()); use_span(get_temp_string());
} }
static_assert(!ConversionCompilesFor<span<const char>, const std::vector<int>>, {
"!ConversionCompilesFor<span<const char>, const std::vector<int>>"); #ifdef CONFIRM_COMPILATION_ERRORS
static_assert(!ConversionCompilesFor<span<char>, const std::string>, auto get_temp_vector = []() -> const std::vector<int> { return {}; };
"!ConversionCompilesFor<span<char>, const std::string>"); auto use_span = [](span<const char> s) { static_cast<void>(s); };
use_span(get_temp_vector());
#endif
}
{ {
auto get_temp_string = []() -> const std::string { return {}; }; auto get_temp_string = []() -> const std::string { return {}; };
@ -561,8 +638,12 @@ TEST(span_test, from_container_constructor)
use_span(get_temp_string()); use_span(get_temp_string());
} }
static_assert(!CtorCompilesFor<span<int>, std::map<int, int>&>, {
"!CtorCompilesFor<span<int>, std::map<int, int>&>"); #ifdef CONFIRM_COMPILATION_ERRORS
std::map<int, int> m;
span<int> s{m};
#endif
}
} }
TEST(span_test, from_convertible_span_constructor) TEST(span_test, from_convertible_span_constructor)
@ -614,20 +695,52 @@ TEST(span_test, from_convertible_span_constructor)
EXPECT_DEATH(T{avd}, expected); EXPECT_DEATH(T{avd}, expected);
} }
static_assert(!ConversionCompilesFor<span<const DerivedClass, 2>, span<DerivedClass>&>, #ifdef CONFIRM_COMPILATION_ERRORS
"!ConversionCompilesFor<span<const DerivedClass, 2>, span<DerivedClass>&>"); {
static_assert(!ConversionCompilesFor<span<const DerivedClass, 1>, span<DerivedClass, 2>&>, std::array<DerivedClass, 2> arr{};
"!ConversionCompilesFor<span<const DerivedClass, 1>, span<DerivedClass, 2>&>"); span<DerivedClass> avd{arr};
static_assert(!ConversionCompilesFor<span<const DerivedClass, 3>, span<DerivedClass, 2>&>, span<const DerivedClass, 2> avcd = avd;
"!ConversionCompilesFor<span<const DerivedClass, 3>, span<DerivedClass, 2>&>"); static_cast<void>(avcd);
static_assert(!ConversionCompilesFor<span<BaseClass, 3>, span<DerivedClass>&>, }
"!ConversionCompilesFor<span<BaseClass, 3>, span<DerivedClass>&>");
static_assert(!ConversionCompilesFor<span<unsigned int>, span<int>&>, {
"!ConversionCompilesFor<span<unsigned int>, span<int>&>"); std::array<DerivedClass, 2> arr{};
static_assert(!ConversionCompilesFor<span<const unsigned int>, span<int>&>, span<DerivedClass, 2> avd{arr};
"!ConversionCompilesFor<span<const unsigned int>, span<int>&>"); span<const DerivedClass, 1> avcd = avd;
static_assert(!ConversionCompilesFor<span<short>, span<int>&>, static_cast<void>(avcd);
"!ConversionCompilesFor<span<short>, span<int>&>"); }
{
std::array<DerivedClass, 2> arr{};
span<DerivedClass, 2> avd{arr};
span<const DerivedClass, 3> avcd = avd;
static_cast<void>(avcd);
}
{
span<DerivedClass> avd;
span<BaseClass> avb = avd;
static_cast<void>(avb);
}
{
span<int> s;
span<unsigned int> s2 = s;
static_cast<void>(s2);
}
{
span<int> s;
span<const unsigned int> s2 = s;
static_cast<void>(s2);
}
{
span<int> s;
span<short> s2 = s;
static_cast<void>(s2);
}
#endif
} }
TEST(span_test, copy_move_and_assignment) TEST(span_test, copy_move_and_assignment)
@ -687,9 +800,10 @@ TEST(span_test, first)
{ {
span<int, 5> av = arr; span<int, 5> av = arr;
#ifdef CONFIRM_COMPILATION_ERRORS #ifdef CONFIRM_COMPILATION_ERRORS
(void) av.first<6>(); EXPECT_TRUE(av.first<6>().size() == 6);
EXPECT_TRUE(av.first<-1>().size() == -1);
#endif #endif
EXPECT_DEATH(av.first(6), expected); EXPECT_DEATH(av.first(6).size(), expected);
} }
{ {
@ -730,9 +844,9 @@ TEST(span_test, last)
{ {
span<int, 5> av = arr; span<int, 5> av = arr;
#ifdef CONFIRM_COMPILATION_ERRORS #ifdef CONFIRM_COMPILATION_ERRORS
(void) av.last<6>(); EXPECT_TRUE(av.last<6>().size() == 6);
#endif #endif
EXPECT_DEATH(av.last(6), expected); EXPECT_DEATH(av.last(6).size(), expected);
} }
{ {
@ -757,9 +871,6 @@ TEST(span_test, subspan)
EXPECT_TRUE((av.subspan<2, 2>().size()) == 2); EXPECT_TRUE((av.subspan<2, 2>().size()) == 2);
EXPECT_TRUE(decltype(av.subspan<2, 2>())::extent == 2); EXPECT_TRUE(decltype(av.subspan<2, 2>())::extent == 2);
EXPECT_TRUE(av.subspan(2, 2).size() == 2); EXPECT_TRUE(av.subspan(2, 2).size() == 2);
EXPECT_TRUE((av.subspan<2, 3>().size()) == 3);
EXPECT_TRUE(decltype(av.subspan<2, 3>())::extent == 3);
EXPECT_TRUE(av.subspan(2, 3).size() == 3); EXPECT_TRUE(av.subspan(2, 3).size() == 3);
} }
@ -776,12 +887,8 @@ TEST(span_test, subspan)
EXPECT_TRUE(decltype(av.subspan<0, 5>())::extent == 5); EXPECT_TRUE(decltype(av.subspan<0, 5>())::extent == 5);
EXPECT_TRUE(av.subspan(0, 5).size() == 5); EXPECT_TRUE(av.subspan(0, 5).size() == 5);
#ifdef CONFIRM_COMPILATION_ERRORS EXPECT_DEATH(av.subspan(0, 6).size(), expected);
(void) av.subspan<0, 6>(); EXPECT_DEATH(av.subspan(1, 5).size(), expected);
(void) av.subspan<1, 5>();
#endif
EXPECT_DEATH(av.subspan(0, 6), expected);
EXPECT_DEATH(av.subspan(1, 5), expected);
} }
{ {
@ -789,22 +896,14 @@ TEST(span_test, subspan)
EXPECT_TRUE((av.subspan<4, 0>().size()) == 0); EXPECT_TRUE((av.subspan<4, 0>().size()) == 0);
EXPECT_TRUE(decltype(av.subspan<4, 0>())::extent == 0); EXPECT_TRUE(decltype(av.subspan<4, 0>())::extent == 0);
EXPECT_TRUE(av.subspan(4, 0).size() == 0); EXPECT_TRUE(av.subspan(4, 0).size() == 0);
EXPECT_TRUE((av.subspan<5, 0>().size()) == 0);
EXPECT_TRUE(decltype(av.subspan<5, 0>())::extent == 0);
EXPECT_TRUE(av.subspan(5, 0).size() == 0); EXPECT_TRUE(av.subspan(5, 0).size() == 0);
EXPECT_DEATH(av.subspan(6, 0).size(), expected);
#ifdef CONFIRM_COMPILATION_ERRORS
(void) av.subspan<6, 0>();
#endif
EXPECT_DEATH(av.subspan(6, 0), expected);
} }
{ {
span<int, 5> av = arr; span<int, 5> av = arr;
EXPECT_TRUE(av.subspan<1>().size() == 4); EXPECT_TRUE(av.subspan<1>().size() == 4);
EXPECT_TRUE(decltype(av.subspan<1>())::extent == 4); EXPECT_TRUE(decltype(av.subspan<1>())::extent == 4);
EXPECT_TRUE(av.subspan(1).size() == 4);
} }
{ {
@ -812,58 +911,35 @@ TEST(span_test, subspan)
EXPECT_TRUE((av.subspan<0, 0>().size()) == 0); EXPECT_TRUE((av.subspan<0, 0>().size()) == 0);
EXPECT_TRUE(decltype(av.subspan<0, 0>())::extent == 0); EXPECT_TRUE(decltype(av.subspan<0, 0>())::extent == 0);
EXPECT_TRUE(av.subspan(0, 0).size() == 0); EXPECT_TRUE(av.subspan(0, 0).size() == 0);
EXPECT_DEATH((av.subspan<1, 0>().size()), expected);
EXPECT_DEATH((av.subspan<1, 0>()), expected);
EXPECT_DEATH((av.subspan(1, 0)), expected);
} }
{ {
span<int> av; span<int> av;
EXPECT_TRUE((av.subspan<0>().size()) == 0);
EXPECT_TRUE(decltype(av.subspan<0>())::extent == dynamic_extent);
EXPECT_TRUE(av.subspan(0).size() == 0); EXPECT_TRUE(av.subspan(0).size() == 0);
EXPECT_DEATH(av.subspan(1).size(), expected);
EXPECT_DEATH(av.subspan<1>(), expected);
EXPECT_TRUE(decltype(av.subspan<1>())::extent == dynamic_extent);
EXPECT_DEATH(av.subspan(1), expected);
} }
{ {
span<int> av = arr; span<int> av = arr;
EXPECT_TRUE(av.subspan(0).size() == 5); EXPECT_TRUE(av.subspan(0).size() == 5);
EXPECT_TRUE(av.subspan<0>().size() == 5);
EXPECT_TRUE(av.subspan(1).size() == 4); EXPECT_TRUE(av.subspan(1).size() == 4);
EXPECT_TRUE(av.subspan<1>().size() == 4);
EXPECT_TRUE(av.subspan(4).size() == 1); EXPECT_TRUE(av.subspan(4).size() == 1);
EXPECT_TRUE(av.subspan<4>().size() == 1);
EXPECT_TRUE(av.subspan(5).size() == 0); EXPECT_TRUE(av.subspan(5).size() == 0);
EXPECT_TRUE(av.subspan<5>().size() == 0); EXPECT_DEATH(av.subspan(6).size(), expected);
EXPECT_DEATH(av.subspan(6), expected);
EXPECT_DEATH(av.subspan<6>(), expected);
const auto av2 = av.subspan(1); const auto av2 = av.subspan(1);
for (std::size_t i = 0; i < 4; ++i) EXPECT_TRUE(av2[i] == static_cast<int>(i) + 2); for (std::size_t i = 0; i < 4; ++i) EXPECT_TRUE(av2[i] == static_cast<int>(i) + 2);
const auto av3 = av.subspan<1>();
for (std::size_t i = 0; i < 4; ++i) EXPECT_TRUE(av3[i] == static_cast<int>(i) + 2);
} }
{ {
span<int, 5> av = arr; span<int, 5> av = arr;
EXPECT_TRUE(av.subspan(0).size() == 5); EXPECT_TRUE(av.subspan(0).size() == 5);
EXPECT_TRUE(av.subspan<0>().size() == 5);
EXPECT_TRUE(av.subspan(1).size() == 4); EXPECT_TRUE(av.subspan(1).size() == 4);
EXPECT_TRUE(av.subspan<1>().size() == 4);
EXPECT_TRUE(av.subspan(4).size() == 1); EXPECT_TRUE(av.subspan(4).size() == 1);
EXPECT_TRUE(av.subspan<4>().size() == 1);
EXPECT_TRUE(av.subspan(5).size() == 0); EXPECT_TRUE(av.subspan(5).size() == 0);
EXPECT_TRUE(av.subspan<5>().size() == 0); EXPECT_DEATH(av.subspan(6).size(), expected);
EXPECT_DEATH(av.subspan(6), expected);
#ifdef CONFIRM_COMPILATION_ERRORS
EXPECT_DEATH(av.subspan<6>(), expected);
#endif
const auto av2 = av.subspan(1); const auto av2 = av.subspan(1);
for (std::size_t i = 0; i < 4; ++i) EXPECT_TRUE(av2[i] == static_cast<int>(i) + 2); for (std::size_t i = 0; i < 4; ++i) EXPECT_TRUE(av2[i] == static_cast<int>(i) + 2);
const auto av3 = av.subspan<1>();
for (std::size_t i = 0; i < 4; ++i) EXPECT_TRUE(av3[i] == static_cast<int>(i) + 2);
} }
} }
@ -1038,21 +1114,9 @@ TEST(span_test, rbegin_rend)
} }
} }
template <typename U, typename = void>
static constexpr bool AsWritableBytesCompilesFor = false;
template <typename U>
static constexpr bool
AsWritableBytesCompilesFor<U, void_t<decltype(as_writable_bytes(std::declval<U>()))>> = true;
TEST(span_test, as_bytes) TEST(span_test, as_bytes)
{ {
int a[] = {1, 2, 3, 4}; int a[] = {1, 2, 3, 4};
static_assert(AsWritableBytesCompilesFor<span<int>>, "AsWriteableBytesCompilesFor<span<int>>");
// you should not be able to get writeable bytes for const objects
static_assert(!AsWritableBytesCompilesFor<span<const int>>,
"!AsWriteableBytesCompilesFor<span<const int>>");
{ {
const span<const int> s = a; const span<const int> s = a;
EXPECT_TRUE(s.size() == 4); EXPECT_TRUE(s.size() == 4);
@ -1083,6 +1147,17 @@ TEST(span_test, as_writable_bytes)
{ {
int a[] = {1, 2, 3, 4}; int a[] = {1, 2, 3, 4};
{
#ifdef CONFIRM_COMPILATION_ERRORS
// you should not be able to get writeable bytes for const objects
span<const int> s = a;
EXPECT_TRUE(s.size() == 4);
span<const byte> bs = as_writable_bytes(s);
EXPECT_TRUE(static_cast<void*>(bs.data()) == static_cast<void*>(s.data()));
EXPECT_TRUE(bs.size() == s.size_bytes());
#endif
}
{ {
span<int> s; span<int> s;
const auto bs = as_writable_bytes(s); const auto bs = as_writable_bytes(s);
@ -1122,15 +1197,30 @@ TEST(span_test, fixed_size_conversions)
static_cast<void>(s); static_cast<void>(s);
} }
// initialization or assignment to static span that REDUCES size is NOT ok // initialization or assignment to static span that REDUCES size is NOT ok
static_assert(!ConversionCompilesFor<span<int, 2>, int[4]>, #ifdef CONFIRM_COMPILATION_ERRORS
"!ConversionCompilesFor<span<int, 2>, int[4]>"); {
static_assert(!ConversionCompilesFor<span<int, 2>, span<int, 4>>, span<int, 2> s = arr;
"!ConversionCompilesFor<span<int, 2>, span<int, 4>>"); }
{
span<int, 2> s2 = s4;
static_cast<void>(s2);
}
#endif
// even when done dynamically // even when done dynamically
static_assert(!ConversionCompilesFor<span<int, 2>, span<int>>, {
"!ConversionCompilesFor<span<int, 2>, span<int>>"); /*
// this now results in a compile-time error, rather than runtime.
// There is no suitable conversion from dynamic span to fixed span.
span<int> s = arr;
auto f = [&]() {
const span<int, 2> s2 = s;
static_cast<void>(s2);
};
EXPECT_DEATH(f(), expected);
*/
}
// but doing so explicitly is ok // but doing so explicitly is ok
@ -1144,24 +1234,32 @@ TEST(span_test, fixed_size_conversions)
static_cast<void>(s1); static_cast<void>(s1);
} }
// this is not a legal operation in std::span, so we are no longer supporting it /*
// conversion from span<int, 4> to span<int, dynamic_extent> via call to `first` // this is not a legal operation in std::span, so we are no longer supporting it
// then convert from span<int, dynamic_extent> to span<int, 1> // conversion from span<int, 4> to span<int, dynamic_extent> via call to `first`
// The dynamic to fixed extents are not supported in the standard // then convert from span<int, dynamic_extent> to span<int, 1>
// to make this work, span<int, 1> would need to be span<int>. // The dynamic to fixed extents are not supported in the standard
static_assert(!ConversionCompilesFor<span<int, 1>, span<int>>, // to make this work, span<int, 1> would need to be span<int>.
"!ConversionCompilesFor<span<int, 1>, span<int>>"); {
// NB: implicit conversion to span<int,1> from span<int>
span<int, 1> s1 = s4.first(1);
static_cast<void>(s1);
}
*/
// initialization or assignment to static span that requires size INCREASE is not ok. // initialization or assignment to static span that requires size INCREASE is not ok.
int arr2[2] = {1, 2}; int arr2[2] = {1, 2};
static_assert(!ConversionCompilesFor<span<int, 4>, int[2]>, #ifdef CONFIRM_COMPILATION_ERRORS
"!ConversionCompilesFor<span<int, 4>, int[2]>"); {
static_assert(!ConversionCompilesFor<span<int, 2>, int[2]>, span<int, 4> s3 = arr2;
"!ConversionCompilesFor<span<int, 2>, int[2]>"); }
static_assert(!ConversionCompilesFor<span<int, 4>, span<int, 2>>, {
"!ConversionCompilesFor<span<int, 4>, span<int, 2>>"); span<int, 2> s2 = arr2;
span<int, 4> s4a = s2;
}
#endif
{ {
auto f = [&]() { auto f = [&]() {
const span<int, 4> _s4{arr2, 2}; const span<int, 4> _s4{arr2, 2};
@ -1170,11 +1268,16 @@ TEST(span_test, fixed_size_conversions)
EXPECT_DEATH(f(), expected); EXPECT_DEATH(f(), expected);
} }
// This no longer compiles. There is no suitable conversion from dynamic span to a fixed size /*
// span. // This no longer compiles. There is no suitable conversion from dynamic span to a fixed size
// this should fail - we are trying to assign a small dynamic span to a fixed_size larger one span.
static_assert(!ConversionCompilesFor<span<int, 4>, span<int>>, // this should fail - we are trying to assign a small dynamic span to a fixed_size larger one
"!ConversionCompilesFor<span<int, 4>, span<int>>"); span<int> av = arr2; auto f = [&]() {
const span<int, 4> _s4 = av;
static_cast<void>(_s4);
};
EXPECT_DEATH(f(), expected);
*/
} }
TEST(span_test, interop_with_std_regex) TEST(span_test, interop_with_std_regex)
@ -1273,154 +1376,8 @@ TEST(span_test, msvc_compile_error_PR1100)
int arr[]{1, 7, 2, 9}; int arr[]{1, 7, 2, 9};
gsl::span sp{arr, std::size(arr)}; gsl::span sp{arr, std::size(arr)};
std::ranges::sort(sp); std::ranges::sort(sp);
for (const auto& e : sp) { (void) e; } for (const auto& e : sp) {
(void)e;
}
} }
#endif // defined(__cpp_lib_span) && defined(__cpp_lib_ranges) #endif // defined(__cpp_lib_span) && defined(__cpp_lib_ranges)
TEST(span_test, empty_span)
{
span<int> s{};
EXPECT_TRUE(s.empty());
EXPECT_TRUE(s.size() == 0);
EXPECT_TRUE(s.data() == nullptr);
span<const int> cs{};
EXPECT_TRUE(cs.empty());
EXPECT_TRUE(cs.size() == 0);
EXPECT_TRUE(cs.data() == nullptr);
}
TEST(span_test, conversions)
{
int arr[5] = {1, 2, 3, 4, 5};
#if defined(__cpp_deduction_guides) && (__cpp_deduction_guides >= 201611L)
span s = arr;
span cs = s;
#else
span<int, 5> s = arr;
span<int, 5> cs = s;
#endif
EXPECT_TRUE(cs.size() == s.size());
EXPECT_TRUE(cs.data() == s.data());
span<int, 5> fs = s;
EXPECT_TRUE(fs.size() == s.size());
EXPECT_TRUE(fs.data() == s.data());
span<const int, 5> cfs = s;
EXPECT_TRUE(cfs.size() == s.size());
EXPECT_TRUE(cfs.data() == s.data());
}
TEST(span_test, comparison_operators)
{
int arr1[3] = {1, 2, 3};
int arr2[3] = {1, 2, 3};
int arr3[3] = {4, 5, 6};
span<int> s1 = arr1;
span<int> s2 = arr2;
span<int> s3 = arr3;
EXPECT_TRUE(s1 == s2);
EXPECT_FALSE(s1 != s2);
EXPECT_FALSE(s1 == s3);
EXPECT_TRUE(s1 != s3);
EXPECT_TRUE(s1 < s3);
EXPECT_FALSE(s3 < s1);
EXPECT_TRUE(s1 <= s2);
EXPECT_TRUE(s1 <= s3);
EXPECT_FALSE(s3 <= s1);
EXPECT_TRUE(s3 > s1);
EXPECT_FALSE(s1 > s3);
EXPECT_TRUE(s3 >= s1);
EXPECT_TRUE(s1 >= s2);
EXPECT_FALSE(s1 >= s3);
}
// ...existing code...
#if defined(__cpp_lib_span) && __cpp_lib_span >= 202002L
#include <span> // for std::span
TEST(span_test, compare_empty_span)
{
gsl::span<int> gsl_s{};
std::span<int> std_s{};
EXPECT_TRUE(gsl_s.empty());
EXPECT_TRUE(std_s.empty());
EXPECT_EQ(gsl_s.size(), std_s.size());
EXPECT_EQ(gsl_s.data(), std_s.data());
}
TEST(span_test, compare_subspan)
{
int arr[5] = {1, 2, 3, 4, 5};
gsl::span gsl_s = arr;
std::span std_s = arr;
auto gsl_sub1 = gsl_s.subspan(1);
auto std_sub1 = std_s.subspan(1);
EXPECT_EQ(gsl_sub1.size(), std_sub1.size());
EXPECT_EQ(gsl_sub1.data(), std_sub1.data());
auto gsl_sub2 = gsl_s.subspan(1, 2);
auto std_sub2 = std_s.subspan(1, 2);
EXPECT_EQ(gsl_sub2.size(), std_sub2.size());
EXPECT_EQ(gsl_sub2.data(), std_sub2.data());
}
TEST(span_test, compare_conversions)
{
int arr[5] = {1, 2, 3, 4, 5};
gsl::span gsl_s = arr;
std::span std_s = arr;
gsl::span gsl_cs = gsl_s;
std::span std_cs = std_s;
EXPECT_EQ(gsl_cs.size(), std_cs.size());
EXPECT_EQ(gsl_cs.data(), std_cs.data());
gsl::span<int, 5> gsl_fs = gsl_s;
std::span<int, 5> std_fs = std_s;
EXPECT_EQ(gsl_fs.size(), std_fs.size());
EXPECT_EQ(gsl_fs.data(), std_fs.data());
gsl::span<const int, 5> gsl_cfs = gsl_s;
std::span<const int, 5> std_cfs = std_s;
EXPECT_EQ(gsl_cfs.size(), std_cfs.size());
EXPECT_EQ(gsl_cfs.data(), std_cfs.data());
}
TEST(span_test, deduction_guides)
{
int arr[5] = {1, 2, 3, 4, 5};
std::array<int, 5> std_arr = {1, 2, 3, 4, 5};
std::vector<int> vec = {1, 2, 3, 4, 5};
// Test deduction guides for gsl::span
gsl::span gsl_s1 = arr;
gsl::span gsl_s2 = std_arr;
gsl::span gsl_s3 = vec;
// Test deduction guides for std::span (for sanity checks)
std::span std_s1 = arr;
std::span std_s2 = std_arr;
std::span std_s3 = vec;
// Compare sizes
EXPECT_EQ(gsl_s1.size(), std_s1.size());
EXPECT_EQ(gsl_s2.size(), std_s2.size());
EXPECT_EQ(gsl_s3.size(), std_s3.size());
// Compare data pointers
EXPECT_EQ(gsl_s1.data(), std_s1.data());
EXPECT_EQ(gsl_s2.data(), std_s2.data());
EXPECT_EQ(gsl_s3.data(), std_s3.data());
}
#endif // defined(__cpp_lib_span) && __cpp_lib_span >= 202002L

View File

@ -17,28 +17,10 @@
#include <gsl/pointers> // for not_null, operator<, operator<=, operator> #include <gsl/pointers> // for not_null, operator<, operator<=, operator>
#include <gtest/gtest.h> #include <gtest/gtest.h>
#include <type_traits> // for declval
#include "deathTestCommon.h" #include "deathTestCommon.h"
using namespace gsl; using namespace gsl;
#if __cplusplus >= 201703l
using std::void_t;
#else // __cplusplus >= 201703l
template <class...>
using void_t = void;
#endif // __cplusplus < 201703l
// stand-in for a user-defined ref-counted class
template <typename T>
struct RefCounted
{
RefCounted(T* p) : p_(p) {}
operator T*() { return p_; }
T* p_;
};
namespace namespace
{ {
// clang-format off // clang-format off
@ -61,153 +43,11 @@ GSL_SUPPRESS(f.4) // NO-FORMAT: attribute
// clang-format on // clang-format on
bool strict_helper_const(strict_not_null<const int*> p) { return *p == 12; } bool strict_helper_const(strict_not_null<const int*> p) { return *p == 12; }
int* return_pointer() { return nullptr; }
} // namespace
template <typename U, typename = void>
static constexpr bool CtorCompilesFor_A = false;
template <typename U>
static constexpr bool
CtorCompilesFor_A<U, void_t<decltype(gsl::strict_not_null<void*>{std::declval<U>()})>> = true;
template <typename U, int N, typename = void>
static constexpr bool CtorCompilesFor_B = false;
template <typename U, int N>
static constexpr bool CtorCompilesFor_B<U, N, void_t<decltype(gsl::strict_not_null<U>{N})>> = true;
template <typename U, typename = void>
static constexpr bool DefaultCtorCompilesFor = false;
template <typename U>
static constexpr bool DefaultCtorCompilesFor<U, void_t<decltype(gsl::strict_not_null<U>{})>> = true;
template <typename U, typename = void>
static constexpr bool CtorCompilesFor_C = false;
template <typename U>
static constexpr bool CtorCompilesFor_C<
U, void_t<decltype(gsl::strict_not_null<U*>{std::declval<std::unique_ptr<U>>()})>> = true;
TEST(strict_notnull_tests, TestStrictNotNullConstructors)
{
{
static_assert(CtorCompilesFor_A<void*>, "CtorCompilesFor_A<void*>");
static_assert(!CtorCompilesFor_A<std::nullptr_t>, "!CtorCompilesFor_A<std::nullptr_t>");
static_assert(!CtorCompilesFor_B<void*, 0>, "!CtorCompilesFor_B<void*, 0>");
static_assert(!DefaultCtorCompilesFor<void*>, "!DefaultCtorCompilesFor<void*>");
static_assert(!CtorCompilesFor_C<int>, "CtorCompilesFor_C<int>");
#ifdef CONFIRM_COMPILATION_ERRORS #ifdef CONFIRM_COMPILATION_ERRORS
// Forbid non-nullptr assignable types int* return_pointer() { return nullptr; }
strict_not_null<std::vector<int>> f(std::vector<int>{1}); const int* return_pointer_const() { return nullptr; }
strict_not_null<int> z(10);
strict_not_null<std::vector<int>> y({1, 2});
#endif #endif
} } // namespace
const auto terminateHandler = std::set_terminate([] {
std::cerr << "Expected Death. TestNotNullConstructors";
std::abort();
});
const auto expected = GetExpectedDeathString(terminateHandler);
{
// from shared pointer
int i = 12;
auto rp = RefCounted<int>(&i);
strict_not_null<int*> p(rp);
EXPECT_TRUE(p.get() == &i);
strict_not_null<std::shared_ptr<int>> x(
std::make_shared<int>(10)); // shared_ptr<int> is nullptr assignable
int* pi = nullptr;
EXPECT_DEATH((strict_not_null<decltype(pi)>(pi)), expected);
}
{
// from unique pointer
strict_not_null<std::unique_ptr<int>> x(
std::make_unique<int>(10)); // unique_ptr<int> is nullptr assignable
EXPECT_DEATH((strict_not_null<std::unique_ptr<int>>(std::unique_ptr<int>{})), expected);
}
{
// from pointer to local
int t = 42;
strict_not_null<int*> x{&t};
helper(&t);
helper_const(&t);
EXPECT_TRUE(*x == 42);
}
{
// from raw pointer
// from strict_not_null pointer
int t = 42;
int* p = &t;
strict_not_null<int*> x{p};
helper(p);
helper_const(p);
helper(x);
helper_const(x);
EXPECT_TRUE(*x == 42);
}
{
// from raw const pointer
// from strict_not_null const pointer
int t = 42;
const int* cp = &t;
strict_not_null<const int*> x{cp};
helper_const(cp);
helper_const(x);
EXPECT_TRUE(*x == 42);
}
{
// from strict_not_null const pointer, using auto
int t = 42;
const int* cp = &t;
auto x = strict_not_null<const int*>{cp};
EXPECT_TRUE(*x == 42);
}
{
// from returned pointer
EXPECT_DEATH(helper(return_pointer()), expected);
EXPECT_DEATH(helper_const(return_pointer()), expected);
}
}
template <typename U, typename = void>
static constexpr bool StrictHelperCompilesFor = false;
template <typename U>
static constexpr bool
StrictHelperCompilesFor<U, void_t<decltype(strict_helper(std::declval<U>()))>> = true;
template <typename U, typename = void>
static constexpr bool StrictHelperConstCompilesFor = false;
template <typename U>
static constexpr bool
StrictHelperConstCompilesFor<U, void_t<decltype(strict_helper_const(std::declval<U>()))>> =
true;
template <typename U, typename = void>
static constexpr bool HelperCompilesFor = false;
template <typename U>
static constexpr bool HelperCompilesFor<U, void_t<decltype(helper(std::declval<U>()))>> = true;
TEST(strict_notnull_tests, TestStrictNotNull) TEST(strict_notnull_tests, TestStrictNotNull)
{ {
@ -217,15 +57,14 @@ TEST(strict_notnull_tests, TestStrictNotNull)
#ifdef CONFIRM_COMPILATION_ERRORS #ifdef CONFIRM_COMPILATION_ERRORS
strict_not_null<int*> snn = &x; strict_not_null<int*> snn = &x;
strict_helper(&x);
strict_helper_const(&x);
strict_helper(return_pointer());
strict_helper_const(return_pointer_const());
#endif #endif
static_assert(!StrictHelperCompilesFor<int*>, "!StrictHelperCompilesFor<int*>");
static_assert(!StrictHelperConstCompilesFor<int*>,
"!StrictHelperCompilesFor<int*>");
const strict_not_null<int*> snn1{&x}; const strict_not_null<int*> snn1{&x};
static_assert(StrictHelperCompilesFor<const strict_not_null<int*>>,
"StrictHelperCompilesFor<const strict_not_null<int*>>");
helper(snn1); helper(snn1);
helper_const(snn1); helper_const(snn1);
@ -238,17 +77,17 @@ TEST(strict_notnull_tests, TestStrictNotNull)
#ifdef CONFIRM_COMPILATION_ERRORS #ifdef CONFIRM_COMPILATION_ERRORS
strict_not_null<int*> snn = &x; strict_not_null<int*> snn = &x;
strict_helper(&x);
strict_helper_const(&x);
strict_helper(return_pointer());
strict_helper_const(return_pointer_const());
#endif #endif
static_assert(!StrictHelperCompilesFor<const int*>, "!StrictHelperFor<const int*>");
static_assert(!StrictHelperConstCompilesFor<const int*>,
"!StrictHelperCompilesFor<const int*>");
const strict_not_null<const int*> snn1{&x}; const strict_not_null<const int*> snn1{&x};
static_assert(!HelperCompilesFor<const strict_not_null<const int*>>, #ifdef CONFIRM_COMPILATION_ERRORS
"!HelperCompilesFor<const strict_not_null<const int*>>"); helper(snn1);
static_assert(StrictHelperConstCompilesFor<const strict_not_null<const int*>>, #endif
"StrictHelperCompilesFor<const strict_not_null<const int*>>");
helper_const(snn1); helper_const(snn1);
EXPECT_TRUE(*snn1 == 42); EXPECT_TRUE(*snn1 == 42);
@ -275,8 +114,9 @@ TEST(strict_notnull_tests, TestStrictNotNull)
strict_not_null<const int*> snn1{&x}; strict_not_null<const int*> snn1{&x};
const strict_not_null<const int*> snn2{&x}; const strict_not_null<const int*> snn2{&x};
static_assert(!StrictHelperCompilesFor<strict_not_null<const int*>>, #ifdef CONFIRM_COMPILATION_ERRORS
"!StrictHelperCompilesFor<strict_not_null<const int*>>"); strict_helper(snn1);
#endif
strict_helper_const(snn1); strict_helper_const(snn1);
strict_helper_const(snn2); strict_helper_const(snn2);
@ -308,8 +148,9 @@ TEST(strict_notnull_tests, TestStrictNotNull)
const not_null<const int*> nn1 = snn; const not_null<const int*> nn1 = snn;
const not_null<const int*> nn2{snn}; const not_null<const int*> nn2{snn};
static_assert(!HelperCompilesFor<strict_not_null<const int*>>, #ifdef CONFIRM_COMPILATION_ERRORS
"!HelperCompilesFor<strict_not_null<const int*>>"); helper(snn);
#endif
helper_const(snn); helper_const(snn);
EXPECT_TRUE(snn == nn1); EXPECT_TRUE(snn == nn1);
@ -349,8 +190,9 @@ TEST(strict_notnull_tests, TestStrictNotNull)
const strict_not_null<const int*> snn1{nn}; const strict_not_null<const int*> snn1{nn};
const strict_not_null<const int*> snn2{nn}; const strict_not_null<const int*> snn2{nn};
static_assert(!StrictHelperCompilesFor<not_null<const int*>>, #ifdef CONFIRM_COMPILATION_ERRORS
"!StrictHelperCompilesFor<not_null<const int*>>"); strict_helper(nn);
#endif
strict_helper_const(nn); strict_helper_const(nn);
EXPECT_TRUE(snn1 == nn); EXPECT_TRUE(snn1 == nn);
@ -364,13 +206,12 @@ TEST(strict_notnull_tests, TestStrictNotNull)
EXPECT_TRUE(hash_nn(snn1) == hash_nn(snn2)); EXPECT_TRUE(hash_nn(snn1) == hash_nn(snn2));
EXPECT_TRUE(hash_snn(snn1) == hash_snn(nn)); EXPECT_TRUE(hash_snn(snn1) == hash_snn(nn));
} }
}
TEST(pointers_test, member_types) #ifdef CONFIRM_COMPILATION_ERRORS
{ {
// make sure `element_type` is inherited from `gsl::not_null` strict_not_null<int*> p{nullptr};
static_assert(std::is_same<gsl::strict_not_null<int*>::element_type, int*>::value, }
"check member type: element_type"); #endif
} }
#if defined(__cplusplus) && (__cplusplus >= 201703L) #if defined(__cplusplus) && (__cplusplus >= 201703L)
@ -397,8 +238,9 @@ TEST(strict_notnull_tests, TestStrictNotNullConstructorTypeDeduction)
const int i = 42; const int i = 42;
strict_not_null x{&i}; strict_not_null x{&i};
static_assert(!HelperCompilesFor<strict_not_null<const int*>>, #ifdef CONFIRM_COMPILATION_ERRORS
"!HelperCompilesFor<strict_not_null<const int*>>"); helper(strict_not_null{&i});
#endif
helper_const(strict_not_null{&i}); helper_const(strict_not_null{&i});
EXPECT_TRUE(*x == 42); EXPECT_TRUE(*x == 42);
@ -420,8 +262,9 @@ TEST(strict_notnull_tests, TestStrictNotNullConstructorTypeDeduction)
const int* p = &i; const int* p = &i;
strict_not_null x{p}; strict_not_null x{p};
static_assert(!HelperCompilesFor<strict_not_null<const int*>>, #ifdef CONFIRM_COMPILATION_ERRORS
"!HelperCompilesFor<strict_not_null<const int*>>"); helper(strict_not_null{p});
#endif
helper_const(strict_not_null{p}); helper_const(strict_not_null{p});
EXPECT_TRUE(*x == 42); EXPECT_TRUE(*x == 42);