mirror of
https://github.com/microsoft/GSL.git
synced 2025-01-19 02:04:59 -05:00
cea0d0ac2b
* Added c++17 test configurations for clang5.0 and clang6.0 * Fixed CppCoreCheck warnings in GSL and tests - Added CMakeSettings.json for VS Open Folder configuration - So we can easily run CppCoreCheck in VS - Fixed CppCorecheck warnings where it made sense - Suppressed the rest - Some suppression does not work due to compiler/tool bugs, so replaced by #pragma disable - CppCoreCheck has noise, suppressed those with comments - Catch produces many warnings, blanket-supressed them all - Had to fix clang formatting to keep attributes in place - clang-format does not support attributes, so I am using - "CommentPragmas: '^ NO-FORMAT:'" to skip formatiting on them - Removed GSL_NOEXCEPT macro, removed incorred noexcepts * Ignore unknown attributes * ignore unknown attributes in noexception mode tests * fixed C26472 in at() * created GSL_SUPPRESS macro to allow all compilers to parse suppression attributes * try to fix gcc compilation problems with attributes * ignore gsl::suppress for gcc * move suppression to function level on return statements clang5.0 and up does not allow attributes on return statemets in constexpr functions * move suppression to function level on return statements * use GSL_SUPPRESS in algorithm_tests * Addressed PR comments
222 lines
6.3 KiB
C++
222 lines
6.3 KiB
C++
///////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// Copyright (c) 2015 Microsoft Corporation. All rights reserved.
|
|
//
|
|
// This code is licensed under the MIT License (MIT).
|
|
//
|
|
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
// AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
// OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
|
// THE SOFTWARE.
|
|
//
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
|
|
#ifdef _MSC_VER
|
|
// blanket turn off warnings from CppCoreCheck from catch
|
|
// so people aren't annoyed by them when running the tool.
|
|
#pragma warning(disable : 26440 26426) // from catch
|
|
#endif
|
|
|
|
#include <catch/catch.hpp> // for AssertionHandler, StringRef, CHECK, CHE...
|
|
|
|
#include <gsl/gsl_algorithm> // for copy
|
|
#include <gsl/span> // for span
|
|
|
|
#include <array> // for array
|
|
#include <cstddef> // for size_t
|
|
|
|
namespace gsl {
|
|
struct fail_fast;
|
|
} // namespace gsl
|
|
|
|
using namespace std;
|
|
using namespace gsl;
|
|
|
|
GSL_SUPPRESS(bounds.4) // NO-FORMAT: attribute
|
|
GSL_SUPPRESS(bounds.2) // NO-FORMAT: attribute
|
|
TEST_CASE("same_type")
|
|
{
|
|
// dynamic source and destination span
|
|
{
|
|
std::array<int, 5> src{1, 2, 3, 4, 5};
|
|
std::array<int, 10> dst{};
|
|
|
|
const span<int> src_span(src);
|
|
const span<int> dst_span(dst);
|
|
|
|
copy(src_span, dst_span);
|
|
copy(src_span, dst_span.subspan(src_span.size()));
|
|
|
|
for (std::size_t i = 0; i < src.size(); ++i) {
|
|
CHECK(dst[i] == src[i]);
|
|
CHECK(dst[i + src.size()] == src[i]);
|
|
}
|
|
}
|
|
|
|
// static source and dynamic destination span
|
|
{
|
|
std::array<int, 5> src{1, 2, 3, 4, 5};
|
|
std::array<int, 10> dst{};
|
|
|
|
const span<int, 5> src_span(src);
|
|
const span<int> dst_span(dst);
|
|
|
|
copy(src_span, dst_span);
|
|
copy(src_span, dst_span.subspan(src_span.size()));
|
|
|
|
for (std::size_t i = 0; i < src.size(); ++i) {
|
|
CHECK(dst[i] == src[i]);
|
|
CHECK(dst[i + src.size()] == src[i]);
|
|
}
|
|
}
|
|
|
|
// dynamic source and static destination span
|
|
{
|
|
std::array<int, 5> src{1, 2, 3, 4, 5};
|
|
std::array<int, 10> dst{};
|
|
|
|
const span<int> src_span(src);
|
|
const span<int, 10> dst_span(dst);
|
|
|
|
copy(src_span, dst_span);
|
|
copy(src_span, dst_span.subspan(src_span.size()));
|
|
|
|
for (std::size_t i = 0; i < src.size(); ++i) {
|
|
CHECK(dst[i] == src[i]);
|
|
CHECK(dst[i + src.size()] == src[i]);
|
|
}
|
|
}
|
|
|
|
// static source and destination span
|
|
{
|
|
std::array<int, 5> src{1, 2, 3, 4, 5};
|
|
std::array<int, 10> dst{};
|
|
|
|
const span<int, 5> src_span(src);
|
|
const span<int, 10> dst_span(dst);
|
|
|
|
copy(src_span, dst_span);
|
|
copy(src_span, dst_span.subspan(src_span.size()));
|
|
|
|
for (std::size_t i = 0; i < src.size(); ++i) {
|
|
CHECK(dst[i] == src[i]);
|
|
CHECK(dst[i + src.size()] == src[i]);
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
GSL_SUPPRESS(bounds.4) // NO-FORMAT: attribute
|
|
GSL_SUPPRESS(bounds.2) // NO-FORMAT: attribute
|
|
TEST_CASE("compatible_type")
|
|
{
|
|
// dynamic source and destination span
|
|
{
|
|
std::array<short, 5> src{1, 2, 3, 4, 5};
|
|
std::array<int, 10> dst{};
|
|
|
|
const span<short> src_span(src);
|
|
const span<int> dst_span(dst);
|
|
|
|
copy(src_span, dst_span);
|
|
copy(src_span, dst_span.subspan(src_span.size()));
|
|
|
|
for (std::size_t i = 0; i < src.size(); ++i) {
|
|
CHECK(dst[i] == src[i]);
|
|
CHECK(dst[i + src.size()] == src[i]);
|
|
}
|
|
}
|
|
|
|
// static source and dynamic destination span
|
|
{
|
|
std::array<short, 5> src{1, 2, 3, 4, 5};
|
|
std::array<int, 10> dst{};
|
|
|
|
const span<short, 5> src_span(src);
|
|
const span<int> dst_span(dst);
|
|
|
|
copy(src_span, dst_span);
|
|
copy(src_span, dst_span.subspan(src_span.size()));
|
|
|
|
for (std::size_t i = 0; i < src.size(); ++i) {
|
|
CHECK(dst[i] == src[i]);
|
|
CHECK(dst[i + src.size()] == src[i]);
|
|
}
|
|
}
|
|
|
|
// dynamic source and static destination span
|
|
{
|
|
std::array<short, 5> src{1, 2, 3, 4, 5};
|
|
std::array<int, 10> dst{};
|
|
|
|
const span<short> src_span(src);
|
|
const span<int, 10> dst_span(dst);
|
|
|
|
copy(src_span, dst_span);
|
|
copy(src_span, dst_span.subspan(src_span.size()));
|
|
|
|
for (std::size_t i = 0; i < src.size(); ++i) {
|
|
CHECK(dst[i] == src[i]);
|
|
CHECK(dst[i + src.size()] == src[i]);
|
|
}
|
|
}
|
|
|
|
// static source and destination span
|
|
{
|
|
std::array<short, 5> src{1, 2, 3, 4, 5};
|
|
std::array<int, 10> dst{};
|
|
|
|
const span<short, 5> src_span(src);
|
|
const span<int, 10> dst_span(dst);
|
|
|
|
copy(src_span, dst_span);
|
|
copy(src_span, dst_span.subspan(src_span.size()));
|
|
|
|
for (std::size_t i = 0; i < src.size(); ++i) {
|
|
CHECK(dst[i] == src[i]);
|
|
CHECK(dst[i + src.size()] == src[i]);
|
|
}
|
|
}
|
|
}
|
|
|
|
#ifdef CONFIRM_COMPILATION_ERRORS
|
|
TEST_CASE("incompatible_type")
|
|
{
|
|
std::array<int, 4> src{1, 2, 3, 4};
|
|
std::array<int*, 12> dst{};
|
|
|
|
span<int> src_span_dyn(src);
|
|
span<int, 4> src_span_static(src);
|
|
span<int*> dst_span_dyn(dst);
|
|
span<int*, 4> dst_span_static(dst);
|
|
|
|
// every line should produce a compilation error
|
|
copy(src_span_dyn, dst_span_dyn);
|
|
copy(src_span_dyn, dst_span_static);
|
|
copy(src_span_static, dst_span_dyn);
|
|
copy(src_span_static, dst_span_static);
|
|
}
|
|
#endif
|
|
|
|
TEST_CASE("small_destination_span")
|
|
{
|
|
std::array<int, 12> src{1, 2, 3, 4};
|
|
std::array<int, 4> dst{};
|
|
|
|
const span<int> src_span_dyn(src);
|
|
const span<int, 12> src_span_static(src);
|
|
const span<int> dst_span_dyn(dst);
|
|
const span<int, 4> dst_span_static(dst);
|
|
|
|
CHECK_THROWS_AS(copy(src_span_dyn, dst_span_dyn), fail_fast);
|
|
CHECK_THROWS_AS(copy(src_span_dyn, dst_span_static), fail_fast);
|
|
CHECK_THROWS_AS(copy(src_span_static, dst_span_dyn), fail_fast);
|
|
|
|
#ifdef CONFIRM_COMPILATION_ERRORS
|
|
copy(src_span_static, dst_span_static);
|
|
#endif
|
|
}
|