//===----------------------------------------------------------------------===// // // The LLVM Compiler Infrastructure // // This file is dual licensed under the MIT and the University of Illinois Open // Source Licenses. See LICENSE.TXT for details. // //===----------------------------------------------------------------------===// // In macosx10.9 to macosx10.14, streams are provided in the dylib AND they // have a bug in how they handle null-termination in case of errors (see D40677). // XFAIL: with_system_cxx_lib=macosx10.14 // XFAIL: with_system_cxx_lib=macosx10.13 // XFAIL: with_system_cxx_lib=macosx10.12 // XFAIL: with_system_cxx_lib=macosx10.11 // XFAIL: with_system_cxx_lib=macosx10.10 // XFAIL: with_system_cxx_lib=macosx10.9 // // basic_istream& get(char_type* s, streamsize n, char_type delim); #include #include #include "test_macros.h" template struct testbuf : public std::basic_streambuf { typedef std::basic_string string_type; typedef std::basic_streambuf base; private: string_type str_; public: testbuf() {} testbuf(const string_type& str) : str_(str) { base::setg(const_cast(str_.data()), const_cast(str_.data()), const_cast(str_.data()) + str_.size()); } CharT* eback() const {return base::eback();} CharT* gptr() const {return base::gptr();} CharT* egptr() const {return base::egptr();} }; int main() { { testbuf sb(" * * "); std::istream is(&sb); char s[5]; is.get(s, 5, '*'); assert(!is.eof()); assert(!is.fail()); assert(std::string(s) == " "); assert(is.gcount() == 2); is.get(s, 5, '*'); assert(!is.eof()); assert( is.fail()); assert(std::string(s) == ""); assert(is.gcount() == 0); is.clear(); assert(is.get() == '*'); is.get(s, 5, '*'); assert(!is.eof()); assert(!is.fail()); assert(std::string(s) == " "); assert(is.gcount() == 4); assert(is.get() == '*'); is.get(s, 5, '*'); assert( is.eof()); assert(!is.fail()); assert(std::string(s) == " "); assert(is.gcount() == 1); // Check that even in error case the buffer is properly 0-terminated. is.get(s, 5, '*'); assert( is.eof()); assert( is.fail()); assert(std::string(s) == ""); assert(is.gcount() == 0); } #ifndef TEST_HAS_NO_EXCEPTIONS { testbuf sb(" "); std::istream is(&sb); char s[5] = "test"; is.exceptions(std::istream::eofbit | std::istream::badbit); try { is.get(s, 5, '*'); assert(false); } catch (std::ios_base::failure&) { } assert( is.eof()); assert( is.fail()); assert(std::string(s) == " "); assert(is.gcount() == 1); } #endif { testbuf sb(L" * * "); std::wistream is(&sb); wchar_t s[5]; is.get(s, 5, L'*'); assert(!is.eof()); assert(!is.fail()); assert(std::wstring(s) == L" "); assert(is.gcount() == 2); is.get(s, 5, L'*'); assert(!is.eof()); assert( is.fail()); assert(std::wstring(s) == L""); assert(is.gcount() == 0); is.clear(); assert(is.get() == L'*'); is.get(s, 5, L'*'); assert(!is.eof()); assert(!is.fail()); assert(std::wstring(s) == L" "); assert(is.gcount() == 4); assert(is.get() == L'*'); is.get(s, 5, L'*'); assert( is.eof()); assert(!is.fail()); assert(std::wstring(s) == L" "); assert(is.gcount() == 1); // Check that even in error case the buffer is properly 0-terminated. is.get(s, 5, L'*'); assert( is.eof()); assert( is.fail()); assert(std::wstring(s) == L""); assert(is.gcount() == 0); } #ifndef TEST_HAS_NO_EXCEPTIONS { testbuf sb(L" "); std::wistream is(&sb); wchar_t s[5] = L"test"; is.exceptions(std::wistream::eofbit | std::wistream::badbit); try { is.get(s, 5, L'*'); assert(false); } catch (std::ios_base::failure&) { } assert( is.eof()); assert( is.fail()); assert(std::wstring(s) == L" "); assert(is.gcount() == 1); } #endif }