mirror of
				https://github.com/llvm-mirror/libcxx.git
				synced 2025-10-23 01:18:52 +08:00 
			
		
		
		
	 1e34c76d33
			
		
	
	1e34c76d33
	
	
	
		
			
			This is a fairly large patch that implements all of the filesystem NB comments
and the relative paths changes (ex. adding weakly_canonical). These issues
and papers are all interrelated so their implementation couldn't be split up
nicely.
This patch upgrades <experimental/filesystem> to match the C++17 spec and not
the published experimental TS spec. Some of the changes in this patch are both
API and ABI breaking, however libc++ makes no guarantee about stability for
experimental implementations.
The major changes in this patch are:
* Implement NB comments for filesystem (P0492R2), including:
  * Implement `perm_options` enum as part of NB comments, and update the
    `permissions` function to match.
  * Implement changes to `remove_filename` and `replace_filename`
  * Implement changes to `path::stem()` and `path::extension()` which support
    splitting examples like `.profile`.
  * Change path iteration to return an empty path instead of '.' for trailing
    separators.
  * Change `operator/=` to handle absolute paths on the RHS.
  * Change `absolute` to no longer accept a current path argument.
* Implement relative paths according to NB comments (P0219r1)
* Combine `path.cpp` and `operations.cpp` since some path functions require
  access to the operations internals, and some fs operations require access
  to the path parser.
git-svn-id: https://llvm.org/svn/llvm-project/libcxx/trunk@329028 91177308-0d34-0410-b5e6-96231b3b80d8
		
	
		
			
				
	
	
		
			106 lines
		
	
	
		
			3.1 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			106 lines
		
	
	
		
			3.1 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 Source>
 | |
| //      path(const Source& source);
 | |
| // template <class InputIterator>
 | |
| //      path(InputIterator first, InputIterator last);
 | |
| 
 | |
| 
 | |
| #include "filesystem_include.hpp"
 | |
| #include <iterator>
 | |
| #include <type_traits>
 | |
| #include <cassert>
 | |
| 
 | |
| #include "test_macros.h"
 | |
| #include "filesystem_test_helper.hpp"
 | |
| 
 | |
| 
 | |
| 
 | |
| template <class It>
 | |
| std::reverse_iterator<It> mkRev(It it) {
 | |
|   return std::reverse_iterator<It>(it);
 | |
| }
 | |
| 
 | |
| void checkIteratorConcepts() {
 | |
|   using namespace fs;
 | |
|   using It = path::iterator;
 | |
|   using Traits = std::iterator_traits<It>;
 | |
|   ASSERT_SAME_TYPE(Traits::iterator_category, std::bidirectional_iterator_tag);
 | |
|   ASSERT_SAME_TYPE(Traits::value_type, path);
 | |
|   ASSERT_SAME_TYPE(Traits::pointer,   path const*);
 | |
|   ASSERT_SAME_TYPE(Traits::reference, path const&);
 | |
|   {
 | |
|     It it;
 | |
|     ASSERT_SAME_TYPE(It&, decltype(++it));
 | |
|     ASSERT_SAME_TYPE(It, decltype(it++));
 | |
|     ASSERT_SAME_TYPE(It&, decltype(--it));
 | |
|     ASSERT_SAME_TYPE(It, decltype(it--));
 | |
|     ASSERT_SAME_TYPE(Traits::reference, decltype(*it));
 | |
|     ASSERT_SAME_TYPE(Traits::pointer, decltype(it.operator->()));
 | |
|     ASSERT_SAME_TYPE(std::string const&, decltype(it->native()));
 | |
|     ASSERT_SAME_TYPE(bool, decltype(it == it));
 | |
|     ASSERT_SAME_TYPE(bool, decltype(it != it));
 | |
|   }
 | |
|   {
 | |
|     path const p;
 | |
|     ASSERT_SAME_TYPE(It, decltype(p.begin()));
 | |
|     ASSERT_SAME_TYPE(It, decltype(p.end()));
 | |
|     assert(p.begin() == p.end());
 | |
|   }
 | |
| }
 | |
| 
 | |
| void checkBeginEndBasic() {
 | |
|   using namespace fs;
 | |
|   using It = path::iterator;
 | |
|   {
 | |
|     path const p;
 | |
|     ASSERT_SAME_TYPE(It, decltype(p.begin()));
 | |
|     ASSERT_SAME_TYPE(It, decltype(p.end()));
 | |
|     assert(p.begin() == p.end());
 | |
|   }
 | |
|   {
 | |
|     path const p("foo");
 | |
|     It default_constructed;
 | |
|     default_constructed = p.begin();
 | |
|     assert(default_constructed == p.begin());
 | |
|     assert(default_constructed != p.end());
 | |
|     default_constructed = p.end();
 | |
|     assert(default_constructed == p.end());
 | |
|     assert(default_constructed != p.begin());
 | |
|   }
 | |
|   {
 | |
|     path p("//root_name//first_dir////second_dir");
 | |
|     const path expect[] = {"/", "root_name", "first_dir", "second_dir"};
 | |
|     assert(checkCollectionsEqual(p.begin(), p.end(), std::begin(expect), std::end(expect)));
 | |
|     assert(checkCollectionsEqualBackwards(p.begin(), p.end(), std::begin(expect), std::end(expect)));
 | |
| 
 | |
|   }
 | |
|   {
 | |
|     path p("////foo/bar/baz///");
 | |
|     const path expect[] = {"/", "foo", "bar", "baz", ""};
 | |
|     assert(checkCollectionsEqual(p.begin(), p.end(), std::begin(expect), std::end(expect)));
 | |
|     assert(checkCollectionsEqualBackwards(p.begin(), p.end(), std::begin(expect), std::end(expect)));
 | |
| 
 | |
|   }
 | |
| 
 | |
| }
 | |
| 
 | |
| int main() {
 | |
|   using namespace fs;
 | |
|   checkIteratorConcepts();
 | |
|   checkBeginEndBasic(); // See path.decompose.pass.cpp for more tests.
 | |
| }
 |