Merge pull request #259 from annagrin/dev/annagrin/legacy_zstring_support

Added zstring_span and removed zstring_builder to support legacy strings
This commit is contained in:
Neil MacIntosh 2016-02-07 12:11:54 -08:00
commit 0be53d99ef
2 changed files with 185 additions and 28 deletions

View File

@ -71,17 +71,21 @@ namespace gsl
// type system for these types that will not either incur significant runtime costs or // type system for these types that will not either incur significant runtime costs or
// (sometimes needlessly) break existing programs when introduced. // (sometimes needlessly) break existing programs when introduced.
// //
template<std::ptrdiff_t Extent = dynamic_range>
using czstring = const char*; template<typename CharT, std::ptrdiff_t Extent = dynamic_range>
using basic_zstring = CharT*;
template<std::ptrdiff_t Extent = dynamic_range> template<std::ptrdiff_t Extent = dynamic_range>
using cwzstring = const wchar_t*; using czstring = basic_zstring<const char, Extent>;
template<std::ptrdiff_t Extent = dynamic_range> template<std::ptrdiff_t Extent = dynamic_range>
using zstring = char*; using cwzstring = basic_zstring<const wchar_t, Extent>;
template<std::ptrdiff_t Extent = dynamic_range> template<std::ptrdiff_t Extent = dynamic_range>
using wzstring = wchar_t*; using zstring = basic_zstring<char, Extent>;
template<std::ptrdiff_t Extent = dynamic_range>
using wzstring = basic_zstring<wchar_t, Extent>;
// //
// ensure_sentinel() // ensure_sentinel()
@ -521,43 +525,82 @@ inline std::wstring to_string(wstring_span<> view)
#endif #endif
template<typename CharT, size_t Extent = dynamic_range> // zero-terminated string span, used to convert
class basic_zstring_builder // zero-terminated spans to legacy strings
template<typename CharT, std::ptrdiff_t Extent = dynamic_range>
class basic_zstring_span
{ {
public: public:
using impl_type = span<CharT, Extent>;
using string_span_type = basic_string_span<CharT, Extent>;
using value_type = CharT; using value_type = CharT;
using pointer = CharT*; using const_value_type = std::add_const_t<CharT>;
using size_type = typename string_span_type::size_type;
using iterator = typename string_span_type::iterator;
basic_zstring_builder(CharT* data, size_type length) : sv_(data, length) {} using pointer = std::add_pointer_t<value_type>;
using const_pointer = std::add_pointer_t<const_value_type>;
template<size_t Size> using zstring_type = basic_zstring<value_type, Extent>;
basic_zstring_builder(CharT(&arr)[Size]) : sv_(arr) {} using const_zstring_type = basic_zstring<const_value_type, Extent>;
pointer data() const { return sv_.data(); } using impl_type = span<value_type, Extent>;
string_span_type view() const { return sv_; } using string_span_type = basic_string_span<value_type, Extent>;
size_type length() const { return sv_.length(); } constexpr basic_zstring_span(impl_type span) noexcept
: span_(span)
{
// expects a zero-terminated span
Expects(span[span.size() - 1] == '\0');
}
pointer assume0() const { return data(); } // copy
string_span_type ensure_z() const { return gsl::ensure_z(sv_); } constexpr basic_zstring_span(const basic_zstring_span& other) = default;
iterator begin() const { return sv_.begin(); } // move
iterator end() const { return sv_.end(); } #ifndef GSL_MSVC_NO_DEFAULT_MOVE_CTOR
constexpr basic_zstring_span(basic_zstring_span&& other) = default;
#else
constexpr basic_zstring_span(basic_zstring_span&& other)
: span_(std::move(other.span_))
{}
#endif
// assign
constexpr basic_zstring_span& operator=(const basic_zstring_span& other) = default;
// move assign
#ifndef GSL_MSVC_NO_DEFAULT_MOVE_CTOR
constexpr basic_zstring_span& operator=(basic_zstring_span&& other) = default;
#else
constexpr basic_zstring_span& operator=(basic_zstring_span&& other)
{
span_ = std::move(other.span_);
return *this;
}
#endif
constexpr bool empty() const noexcept { return span_.size() == 0; }
constexpr string_span_type as_string_span() const noexcept { return span_.first(span_.size()-1); }
constexpr string_span_type ensure_z() const noexcept { return gsl::ensure_z(span_); }
constexpr const_zstring_type assume_z() const noexcept { return span_.data(); }
private: private:
impl_type sv_; impl_type span_;
}; };
template <size_t Max = dynamic_range> template <std::ptrdiff_t Max = dynamic_range>
using zstring_builder = basic_zstring_builder<char, Max>; using zstring_span = basic_zstring_span<char, Max>;
template <size_t Max = dynamic_range> template <std::ptrdiff_t Max = dynamic_range>
using wzstring_builder = basic_zstring_builder<wchar_t, Max>; using wzstring_span = basic_zstring_span<wchar_t, Max>;
}
template <std::ptrdiff_t Max = dynamic_range>
using czstring_span = basic_zstring_span<const char, Max>;
template <std::ptrdiff_t Max = dynamic_range>
using cwzstring_span = basic_zstring_span<const wchar_t, Max>;
} // namespace GSL
// operator == // operator ==
template <typename CharT, std::ptrdiff_t Extent = gsl::dynamic_range, typename T, template <typename CharT, std::ptrdiff_t Extent = gsl::dynamic_range, typename T,

View File

@ -830,6 +830,120 @@ SUITE(string_span_tests)
CHECK(wspan.length() == 5); CHECK(wspan.length() == 5);
#endif #endif
} }
czstring_span<> CreateTempName(string_span<> span)
{
Expects(span.size() > 1);
int last = 0;
if (span.size() > 4)
{
span[0] = 't';
span[1] = 'm';
span[2] = 'p';
last = 3;
}
span[last] = '\0';
auto ret = span.subspan(0, 4);
return{ ret };
}
TEST(zstring)
{
// create zspan from zero terminated string
{
char buf[1];
buf[0] = '\0';
zstring_span<> zspan({ buf, 1 });
CHECK(strlen(zspan.assume_z()) == 0);
CHECK(zspan.as_string_span().size() == 0);
CHECK(zspan.ensure_z().size() == 0);
}
// create zspan from non-zero terminated string
{
char buf[1];
buf[0] = 'a';
auto workaround_macro = [&]() { zstring_span<> zspan({ buf, 1 }); };
CHECK_THROW(workaround_macro(), fail_fast);
}
// usage scenario: create zero-terminated temp file name and pass to a legacy API
{
char buf[10];
auto name = CreateTempName({ buf, 10 });
if (!name.empty())
{
czstring<> str = name.assume_z();
CHECK(strlen(str) == 3);
CHECK(*(str+3) == '\0');
}
}
}
cwzstring_span<> CreateTempNameW(wstring_span<> span)
{
Expects(span.size() > 1);
int last = 0;
if (span.size() > 4)
{
span[0] = L't';
span[1] = L'm';
span[2] = L'p';
last = 3;
}
span[last] = L'\0';
auto ret = span.subspan(0, 4);
return{ ret };
}
TEST(wzstring)
{
// create zspan from zero terminated string
{
wchar_t buf[1];
buf[0] = L'\0';
wzstring_span<> zspan({ buf, 1 });
CHECK(wcsnlen(zspan.assume_z(), 1) == 0);
CHECK(zspan.as_string_span().size() == 0);
CHECK(zspan.ensure_z().size() == 0);
}
// create zspan from non-zero terminated string
{
wchar_t buf[1];
buf[0] = L'a';
auto workaround_macro = [&]() { wzstring_span<> zspan({ buf, 1 }); };
CHECK_THROW(workaround_macro(), fail_fast);
}
// usage scenario: create zero-terminated temp file name and pass to a legacy API
{
wchar_t buf[10];
auto name = CreateTempNameW({ buf, 10 });
if (!name.empty())
{
cwzstring<> str = name.assume_z();
CHECK(wcsnlen(str, 10) == 3);
CHECK(*(str + 3) == L'\0');
}
}
}
} }
int main(int, const char *[]) int main(int, const char *[])