
Summary: This is a re-application of r357533 and r357531. They had been reverted because we thought the commits broke the LLDB data formatters, but it turns out this was because only r357531 had been included in the CI run. Before this patch, we would only ever throw an exception if the badbit was set on the stream. The Standard is currently very unclear on how exceptions should be propagated and what error flags should be set by the input stream operations. This commit changes libc++ to behave under a different (but valid) interpretation of the Standard. This interpretation of the Standard matches what other implementations are doing. This effectively implements the wording in p1264r0. It hasn't been voted into the Standard yet, however there is wide agreement that the fix is correct and it's just a matter of time before the fix is standardized. PR21586 PR15949 rdar://problem/15347558 Reviewers: mclow.lists, EricWF Subscribers: christof, dexonsmith, cfe-commits Differential Revision: https://reviews.llvm.org/D49863 llvm-svn: 357775
118 lines
2.7 KiB
C++
118 lines
2.7 KiB
C++
//===----------------------------------------------------------------------===//
|
|
//
|
|
// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
|
|
// See https://llvm.org/LICENSE.txt for license information.
|
|
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
// <istream>
|
|
|
|
// template <class charT, class traits>
|
|
// basic_istream<charT,traits>&
|
|
// ws(basic_istream<charT,traits>& is);
|
|
|
|
#include <istream>
|
|
#include <cassert>
|
|
#include "test_macros.h"
|
|
|
|
template <class CharT>
|
|
struct testbuf
|
|
: public std::basic_streambuf<CharT>
|
|
{
|
|
typedef std::basic_string<CharT> string_type;
|
|
typedef std::basic_streambuf<CharT> base;
|
|
private:
|
|
string_type str_;
|
|
public:
|
|
|
|
testbuf() {}
|
|
testbuf(const string_type& str)
|
|
: str_(str)
|
|
{
|
|
base::setg(const_cast<CharT*>(str_.data()),
|
|
const_cast<CharT*>(str_.data()),
|
|
const_cast<CharT*>(str_.data()) + str_.size());
|
|
}
|
|
|
|
CharT* eback() const {return base::eback();}
|
|
CharT* gptr() const {return base::gptr();}
|
|
CharT* egptr() const {return base::egptr();}
|
|
};
|
|
|
|
int main(int, char**)
|
|
{
|
|
{
|
|
testbuf<char> sb(" 123");
|
|
std::istream is(&sb);
|
|
ws(is);
|
|
assert(is.good());
|
|
assert(is.peek() == '1');
|
|
}
|
|
{
|
|
testbuf<wchar_t> sb(L" 123");
|
|
std::wistream is(&sb);
|
|
ws(is);
|
|
assert(is.good());
|
|
assert(is.peek() == L'1');
|
|
}
|
|
{
|
|
testbuf<char> sb(" ");
|
|
std::istream is(&sb);
|
|
ws(is);
|
|
assert(!is.fail());
|
|
assert(is.eof());
|
|
ws(is);
|
|
assert(is.eof());
|
|
assert(is.fail());
|
|
}
|
|
{
|
|
testbuf<wchar_t> sb(L" ");
|
|
std::wistream is(&sb);
|
|
ws(is);
|
|
assert(!is.fail());
|
|
assert(is.eof());
|
|
ws(is);
|
|
assert(is.eof());
|
|
assert(is.fail());
|
|
}
|
|
#ifndef TEST_HAS_NO_EXCEPTIONS
|
|
{
|
|
testbuf<char> sb(" ");
|
|
std::basic_istream<char> is(&sb);
|
|
is.exceptions(std::ios_base::eofbit);
|
|
|
|
bool threw = false;
|
|
try {
|
|
std::ws(is);
|
|
} catch (std::ios_base::failure const&) {
|
|
threw = true;
|
|
}
|
|
|
|
assert(!is.bad());
|
|
assert(!is.fail());
|
|
assert( is.eof());
|
|
assert(threw);
|
|
}
|
|
{
|
|
testbuf<wchar_t> sb(L" ");
|
|
std::basic_istream<wchar_t> is(&sb);
|
|
is.exceptions(std::ios_base::eofbit);
|
|
|
|
bool threw = false;
|
|
try {
|
|
std::ws(is);
|
|
} catch (std::ios_base::failure const&) {
|
|
threw = true;
|
|
}
|
|
|
|
assert(!is.bad());
|
|
assert(!is.fail());
|
|
assert( is.eof());
|
|
assert(threw);
|
|
}
|
|
#endif
|
|
|
|
return 0;
|
|
}
|