mirror of
https://github.com/llvm-mirror/libcxx.git
synced 2025-10-23 18:38:30 +08:00
Add Filesystem TS -- Complete
Add the completed std::experimental::filesystem implementation and tests. The implementation supports C++11 or newer. The TS is built as part of 'libc++experimental.a'. Users of the TS need to manually link this library. Building and testing the TS can be disabled using the CMake option '-DLIBCXX_ENABLE_FILESYSTEM=OFF'. Currently 'libc++experimental.a' is not installed by default. To turn on the installation of the library use '-DLIBCXX_INSTALL_EXPERIMENTAL_LIBRARY=ON'. git-svn-id: https://llvm.org/svn/llvm-project/libcxx/trunk@273034 91177308-0d34-0410-b5e6-96231b3b80d8
This commit is contained in:
@@ -0,0 +1,277 @@
|
||||
//===----------------------------------------------------------------------===//
|
||||
//
|
||||
// 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
|
||||
|
||||
// path& operator+=(const path& x);
|
||||
// path& operator+=(const string_type& x); // Implemented as Source template
|
||||
// path& operator+=(const value_type* x); // Implemented as Source template
|
||||
// path& operator+=(value_type x);
|
||||
// template <class Source>
|
||||
// path& operator+=(const Source& x);
|
||||
// template <class EcharT>
|
||||
// path& operator+=(EcharT x);
|
||||
// template <class Source>
|
||||
// path& concat(const Source& x);
|
||||
// template <class InputIterator>
|
||||
// path& concat(InputIterator first, InputIterator last);
|
||||
|
||||
|
||||
#include <experimental/filesystem>
|
||||
#include <type_traits>
|
||||
#include <cassert>
|
||||
|
||||
#include "test_macros.h"
|
||||
#include "test_iterators.h"
|
||||
#include "count_new.hpp"
|
||||
#include "filesystem_test_helper.hpp"
|
||||
|
||||
namespace fs = std::experimental::filesystem;
|
||||
|
||||
struct ConcatOperatorTestcase {
|
||||
MultiStringType lhs;
|
||||
MultiStringType rhs;
|
||||
MultiStringType expect;
|
||||
};
|
||||
|
||||
#define LONGSTR "LONGSTR_LONGSTR_LONGSTR_LONGSTR_LONGSTR_LONGSTR_LONGSTR_LONGSTR_LONGSTR_LONGSTR_LONGSTR_LONGSTR"
|
||||
#define S(Str) MKSTR(Str)
|
||||
const ConcatOperatorTestcase Cases[] =
|
||||
{
|
||||
{S(""), S(""), S("")}
|
||||
, {S("p1"), S("p2"), S("p1p2")}
|
||||
, {S("p1/"), S("/p2"), S("p1//p2")}
|
||||
, {S(""), S("\\foo/bar/baz"), S("\\foo/bar/baz")}
|
||||
, {S("c:\\foo"), S(""), S("c:\\foo")}
|
||||
, {S(LONGSTR), S("foo"), S(LONGSTR "foo")}
|
||||
, {S("abcdefghijklmnopqrstuvwxyz/\\"), S("/\\123456789"), S("abcdefghijklmnopqrstuvwxyz/\\/\\123456789")}
|
||||
};
|
||||
const ConcatOperatorTestcase LongLHSCases[] =
|
||||
{
|
||||
{S(""), S(LONGSTR), S(LONGSTR)}
|
||||
, {S("p1/"), S(LONGSTR), S("p1/" LONGSTR)}
|
||||
};
|
||||
const ConcatOperatorTestcase CharTestCases[] =
|
||||
{
|
||||
{S(""), S("P"), S("P")}
|
||||
, {S("/fooba"), S("r"), S("/foobar")}
|
||||
};
|
||||
#undef S
|
||||
#undef LONGSTR
|
||||
|
||||
// The concat operator may need to allocate a temporary buffer before a code_cvt
|
||||
// conversion. Test if this allocation occurs by:
|
||||
// 1. Create a path, `LHS`, and reserve enough space to append `RHS`.
|
||||
// This prevents `LHS` from allocating during the actual appending.
|
||||
// 2. Create a `Source` object `RHS`, which represents a "large" string.
|
||||
// (The string must not trigger the SSO)
|
||||
// 3. Concat `RHS` to `LHS` and check for the expected allocation behavior.
|
||||
template <class CharT>
|
||||
void doConcatSourceAllocTest(ConcatOperatorTestcase const& TC)
|
||||
{
|
||||
using namespace fs;
|
||||
using Ptr = CharT const*;
|
||||
using Str = std::basic_string<CharT>;
|
||||
using InputIter = input_iterator<Ptr>;
|
||||
|
||||
const Ptr L = TC.lhs;
|
||||
const Ptr R = TC.rhs;
|
||||
const Ptr E = TC.expect;
|
||||
std::size_t ReserveSize = StrLen(E) + 1;
|
||||
// basic_string
|
||||
{
|
||||
path LHS(L); PathReserve(LHS, ReserveSize);
|
||||
Str RHS(R);
|
||||
{
|
||||
DisableAllocationGuard g;
|
||||
LHS += RHS;
|
||||
}
|
||||
assert(LHS == E);
|
||||
}
|
||||
// CharT*
|
||||
{
|
||||
path LHS(L); PathReserve(LHS, ReserveSize);
|
||||
Ptr RHS(R);
|
||||
{
|
||||
DisableAllocationGuard g;
|
||||
LHS += RHS;
|
||||
}
|
||||
assert(LHS == E);
|
||||
}
|
||||
{
|
||||
path LHS(L); PathReserve(LHS, ReserveSize);
|
||||
Ptr RHS(R);
|
||||
{
|
||||
DisableAllocationGuard g;
|
||||
LHS.concat(RHS, StrEnd(RHS));
|
||||
}
|
||||
assert(LHS == E);
|
||||
}
|
||||
// input iterator - For non-native char types, appends needs to copy the
|
||||
// iterator range into a contigious block of memory before it can perform the
|
||||
// code_cvt conversions.
|
||||
// For "char" no allocations will be performed because no conversion is
|
||||
// required.
|
||||
bool DisableAllocations = std::is_same<CharT, char>::value;
|
||||
{
|
||||
path LHS(L); PathReserve(LHS, ReserveSize);
|
||||
InputIter RHS(R);
|
||||
{
|
||||
RequireAllocationGuard g; // requires 1 or more allocations occur by default
|
||||
if (DisableAllocations) g.requireExactly(0);
|
||||
LHS += RHS;
|
||||
}
|
||||
assert(LHS == E);
|
||||
}
|
||||
{
|
||||
path LHS(L); PathReserve(LHS, ReserveSize);
|
||||
InputIter RHS(R);
|
||||
InputIter REnd(StrEnd(R));
|
||||
{
|
||||
RequireAllocationGuard g;
|
||||
if (DisableAllocations) g.requireExactly(0);
|
||||
LHS.concat(RHS, REnd);
|
||||
}
|
||||
assert(LHS == E);
|
||||
}
|
||||
}
|
||||
|
||||
template <class CharT>
|
||||
void doConcatSourceTest(ConcatOperatorTestcase const& TC)
|
||||
{
|
||||
using namespace fs;
|
||||
using Ptr = CharT const*;
|
||||
using Str = std::basic_string<CharT>;
|
||||
using InputIter = input_iterator<Ptr>;
|
||||
const Ptr L = TC.lhs;
|
||||
const Ptr R = TC.rhs;
|
||||
const Ptr E = TC.expect;
|
||||
// basic_string
|
||||
{
|
||||
path LHS(L);
|
||||
Str RHS(R);
|
||||
path& Ref = (LHS += RHS);
|
||||
assert(LHS == E);
|
||||
assert(&Ref == &LHS);
|
||||
}
|
||||
{
|
||||
path LHS(L);
|
||||
Str RHS(R);
|
||||
path& Ref = LHS.concat(RHS);
|
||||
assert(LHS == E);
|
||||
assert(&Ref == &LHS);
|
||||
}
|
||||
// Char*
|
||||
{
|
||||
path LHS(L);
|
||||
Str RHS(R);
|
||||
path& Ref = (LHS += RHS);
|
||||
assert(LHS == E);
|
||||
assert(&Ref == &LHS);
|
||||
}
|
||||
{
|
||||
path LHS(L);
|
||||
Ptr RHS(R);
|
||||
path& Ref = LHS.concat(RHS);
|
||||
assert(LHS == E);
|
||||
assert(&Ref == &LHS);
|
||||
}
|
||||
{
|
||||
path LHS(L);
|
||||
Ptr RHS(R);
|
||||
path& Ref = LHS.concat(RHS, StrEnd(RHS));
|
||||
assert(LHS == E);
|
||||
assert(&Ref == &LHS);
|
||||
}
|
||||
// iterators
|
||||
{
|
||||
path LHS(L);
|
||||
InputIter RHS(R);
|
||||
path& Ref = (LHS += RHS);
|
||||
assert(LHS == E);
|
||||
assert(&Ref == &LHS);
|
||||
}
|
||||
{
|
||||
path LHS(L); InputIter RHS(R);
|
||||
path& Ref = LHS.concat(RHS);
|
||||
assert(LHS == E);
|
||||
assert(&Ref == &LHS);
|
||||
}
|
||||
{
|
||||
path LHS(L);
|
||||
InputIter RHS(R);
|
||||
InputIter REnd(StrEnd(R));
|
||||
path& Ref = LHS.concat(RHS, REnd);
|
||||
assert(LHS == E);
|
||||
assert(&Ref == &LHS);
|
||||
}
|
||||
}
|
||||
|
||||
template <class CharT>
|
||||
void doConcatECharTest(ConcatOperatorTestcase const& TC)
|
||||
{
|
||||
using namespace fs;
|
||||
using Ptr = CharT const*;
|
||||
const Ptr RStr = TC.rhs;
|
||||
assert(StrLen(RStr) == 1);
|
||||
const Ptr L = TC.lhs;
|
||||
const CharT R = RStr[0];
|
||||
const Ptr E = TC.expect;
|
||||
{
|
||||
path LHS(L);
|
||||
path& Ref = (LHS += R);
|
||||
assert(LHS == E);
|
||||
assert(&Ref == &LHS);
|
||||
}
|
||||
}
|
||||
|
||||
int main()
|
||||
{
|
||||
using namespace fs;
|
||||
for (auto const & TC : Cases) {
|
||||
{
|
||||
path LHS((const char*)TC.lhs);
|
||||
path RHS((const char*)TC.rhs);
|
||||
path& Ref = (LHS += RHS);
|
||||
assert(LHS == (const char*)TC.expect);
|
||||
assert(&Ref == &LHS);
|
||||
}
|
||||
doConcatSourceTest<char> (TC);
|
||||
doConcatSourceTest<wchar_t> (TC);
|
||||
doConcatSourceTest<char16_t>(TC);
|
||||
doConcatSourceTest<char32_t>(TC);
|
||||
}
|
||||
for (auto const & TC : LongLHSCases) {
|
||||
// Do path test
|
||||
{
|
||||
path LHS((const char*)TC.lhs);
|
||||
path RHS((const char*)TC.rhs);
|
||||
const char* E = TC.expect;
|
||||
PathReserve(LHS, StrLen(E) + 5);
|
||||
{
|
||||
DisableAllocationGuard g;
|
||||
path& Ref = (LHS += RHS);
|
||||
assert(&Ref == &LHS);
|
||||
}
|
||||
assert(LHS == E);
|
||||
}
|
||||
doConcatSourceAllocTest<char>(TC);
|
||||
doConcatSourceAllocTest<wchar_t>(TC);
|
||||
}
|
||||
for (auto const& TC : CharTestCases) {
|
||||
doConcatECharTest<char>(TC);
|
||||
doConcatECharTest<wchar_t>(TC);
|
||||
doConcatECharTest<char16_t>(TC);
|
||||
doConcatECharTest<char32_t>(TC);
|
||||
}
|
||||
}
|
Reference in New Issue
Block a user