From e63a76f895bcd75d7cd7c7c8bcf07649c9c9a3f1 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Konstantin=20K=C3=A4fer?= Date: Mon, 11 Jun 2018 14:02:15 +0200 Subject: [core] fix undefined behavior in LineAtlas::addDash --- cmake/test-files.cmake | 1 + src/mbgl/geometry/line_atlas.cpp | 7 +++++++ test/geometry/line_atlas.test.cpp | 31 +++++++++++++++++++++++++++++++ 3 files changed, 39 insertions(+) create mode 100644 test/geometry/line_atlas.test.cpp diff --git a/cmake/test-files.cmake b/cmake/test-files.cmake index 8105cbf5ce..a93f199d2a 100644 --- a/cmake/test-files.cmake +++ b/cmake/test-files.cmake @@ -23,6 +23,7 @@ set(MBGL_TEST_FILES # geometry test/geometry/dem_data.test.cpp + test/geometry/line_atlas.test.cpp # gl test/gl/bucket.test.cpp diff --git a/src/mbgl/geometry/line_atlas.cpp b/src/mbgl/geometry/line_atlas.cpp index 71a855b943..1129bd0b20 100644 --- a/src/mbgl/geometry/line_atlas.cpp +++ b/src/mbgl/geometry/line_atlas.cpp @@ -41,6 +41,10 @@ LinePatternPos LineAtlas::addDash(const std::vector& dasharray, LinePatte const uint8_t dashheight = 2 * n + 1; const uint8_t offset = 128; + if (dasharray.size() < 2) { + return LinePatternPos(); + } + if (nextRow + dashheight > image.size.height) { Log::Warning(Event::OpenGL, "line atlas bitmap overflow"); return LinePatternPos(); @@ -73,6 +77,9 @@ LinePatternPos LineAtlas::addDash(const std::vector& dasharray, LinePatte while (right < x / stretch) { left = right; + if (partIndex >= dasharray.size()) { + return LinePatternPos(); + } right = right + dasharray[partIndex]; if (oddLength && partIndex == dasharray.size() - 1) { diff --git a/test/geometry/line_atlas.test.cpp b/test/geometry/line_atlas.test.cpp new file mode 100644 index 0000000000..960e4ad7ad --- /dev/null +++ b/test/geometry/line_atlas.test.cpp @@ -0,0 +1,31 @@ +#include + +#include + +#include + +using namespace mbgl; + +TEST(LineAtlas, Random) { + std::mt19937 generator(42); // Use the same seed for reproducible tests. + std::uniform_int_distribution countDistribution(0, 12); + std::uniform_int_distribution capStyleDistribution(0, 1); + std::normal_distribution lengthDistribution(3, 5); + + for (size_t it = 0; it < 100; it++) { + LineAtlas atlas{ Size{ 128, 1024 } }; + std::vector dasharray; + dasharray.reserve(8); + for (size_t j = 0; j < 100; j++) { + dasharray.resize(0); + const size_t count = countDistribution(generator); + for (size_t i = 0; i < count; i++) { + dasharray.push_back(lengthDistribution(generator)); + } + const LinePatternCap patternCap = + capStyleDistribution(generator) > 0 ? LinePatternCap::Round : LinePatternCap::Square; + + atlas.addDash(dasharray, patternCap); + } + } +} -- cgit v1.2.1