mirror of
https://github.com/llvm-mirror/libcxx.git
synced 2025-10-24 12:02:36 +08:00

Because path can be constructed from a ton of different types, including string and wide strings, this caused it's streaming operators to suck up all sorts of silly types via silly conversions. For example: using namespace std::experimental::filesystem::v1; std::wstring w(L"wide"); std::cout << w; // converts to path. This patch tentatively adopts the resolution to LWG2989 and fixes the issue by making the streaming operators friends of path. git-svn-id: https://llvm.org/svn/llvm-project/libcxx/trunk@324189 91177308-0d34-0410-b5e6-96231b3b80d8
99 lines
2.7 KiB
C++
99 lines
2.7 KiB
C++
//===----------------------------------------------------------------------===//
|
|
//
|
|
// 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.
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
// UNSUPPORTED: c++98, c++03
|
|
|
|
// <experimental/filesystem>
|
|
|
|
// class path
|
|
|
|
// template <class charT, class traits>
|
|
// basic_ostream<charT, traits>&
|
|
// operator<<(basic_ostream<charT, traits>& os, const path& p);
|
|
//
|
|
// template <class charT, class traits>
|
|
// basic_istream<charT, traits>&
|
|
// operator>>(basic_istream<charT, traits>& is, path& p)
|
|
//
|
|
|
|
#include <experimental/filesystem>
|
|
#include <type_traits>
|
|
#include <sstream>
|
|
#include <cassert>
|
|
#include <iostream>
|
|
|
|
#include "test_macros.h"
|
|
#include "test_iterators.h"
|
|
#include "count_new.hpp"
|
|
#include "filesystem_test_helper.hpp"
|
|
|
|
MultiStringType InStr = MKSTR("abcdefg/\"hijklmnop\"/qrstuvwxyz/123456789");
|
|
MultiStringType OutStr = MKSTR("\"abcdefg/\\\"hijklmnop\\\"/qrstuvwxyz/123456789\"");
|
|
|
|
|
|
|
|
template <class CharT>
|
|
void doIOTest() {
|
|
using namespace fs;
|
|
using Ptr = const CharT*;
|
|
using StrStream = std::basic_stringstream<CharT>;
|
|
const Ptr E = OutStr;
|
|
const path p((const char*)InStr);
|
|
StrStream ss;
|
|
{ // test output
|
|
auto& ret = (ss << p);
|
|
assert(ss.str() == E);
|
|
assert(&ret == &ss);
|
|
}
|
|
{ // test input
|
|
path p_in;
|
|
auto& ret = ss >> p_in;
|
|
assert(p_in.native() == (const char*)InStr);
|
|
assert(&ret == &ss);
|
|
}
|
|
}
|
|
|
|
namespace impl {
|
|
using namespace fs;
|
|
|
|
template <class Stream, class Tp, class = decltype(std::declval<Stream&>() << std::declval<Tp&>())>
|
|
std::true_type is_ostreamable_imp(int);
|
|
|
|
template <class Stream, class Tp>
|
|
std::false_type is_ostreamable_imp(long);
|
|
|
|
template <class Stream, class Tp, class = decltype(std::declval<Stream&>() >> std::declval<Tp&>())>
|
|
std::true_type is_istreamable_imp(int);
|
|
|
|
template <class Stream, class Tp>
|
|
std::false_type is_istreamable_imp(long);
|
|
|
|
|
|
} // namespace impl
|
|
|
|
template <class Stream, class Tp>
|
|
struct is_ostreamable : decltype(impl::is_ostreamable_imp<Stream, Tp>(0)) {};
|
|
template <class Stream, class Tp>
|
|
struct is_istreamable : decltype(impl::is_istreamable_imp<Stream, Tp>(0)) {};
|
|
|
|
void test_LWG2989() {
|
|
static_assert(!is_ostreamable<decltype(std::cout), std::wstring>::value, "");
|
|
static_assert(!is_ostreamable<decltype(std::wcout), std::string>::value, "");
|
|
static_assert(!is_istreamable<decltype(std::cin), std::wstring>::value, "");
|
|
static_assert(!is_istreamable<decltype(std::wcin), std::string>::value, "");
|
|
}
|
|
|
|
int main() {
|
|
doIOTest<char>();
|
|
doIOTest<wchar_t>();
|
|
//doIOTest<char16_t>();
|
|
//doIOTest<char32_t>();
|
|
test_LWG2989();
|
|
}
|