Home | History | Annotate | Download | only in include
      1 // -*- C++ -*-
      2 //===--------------------------- filesystem -------------------------------===//
      3 //
      4 //                     The LLVM Compiler Infrastructure
      5 //
      6 // This file is dual licensed under the MIT and the University of Illinois Open
      7 // Source Licenses. See LICENSE.TXT for details.
      8 //
      9 //===----------------------------------------------------------------------===//
     10 #ifndef _LIBCPP_FILESYSTEM
     11 #define _LIBCPP_FILESYSTEM
     12 /*
     13     filesystem synopsis
     14 
     15     namespace std { namespace filesystem {
     16 
     17     class path;
     18 
     19     void swap(path& lhs, path& rhs) noexcept;
     20     size_t hash_value(const path& p) noexcept;
     21 
     22     bool operator==(const path& lhs, const path& rhs) noexcept;
     23     bool operator!=(const path& lhs, const path& rhs) noexcept;
     24     bool operator< (const path& lhs, const path& rhs) noexcept;
     25     bool operator<=(const path& lhs, const path& rhs) noexcept;
     26     bool operator> (const path& lhs, const path& rhs) noexcept;
     27     bool operator>=(const path& lhs, const path& rhs) noexcept;
     28 
     29     path operator/ (const path& lhs, const path& rhs);
     30 
     31     // fs.path.io operators are friends of path.
     32     template <class charT, class traits>
     33     friend basic_ostream<charT, traits>&
     34     operator<<(basic_ostream<charT, traits>& os, const path& p);
     35 
     36     template <class charT, class traits>
     37     friend basic_istream<charT, traits>&
     38     operator>>(basic_istream<charT, traits>& is, path& p);
     39 
     40     template <class Source>
     41       path u8path(const Source& source);
     42     template <class InputIterator>
     43       path u8path(InputIterator first, InputIterator last);
     44 
     45     class filesystem_error;
     46     class directory_entry;
     47 
     48     class directory_iterator;
     49 
     50     // enable directory_iterator range-based for statements
     51     directory_iterator begin(directory_iterator iter) noexcept;
     52     directory_iterator end(const directory_iterator&) noexcept;
     53 
     54     class recursive_directory_iterator;
     55 
     56     // enable recursive_directory_iterator range-based for statements
     57     recursive_directory_iterator begin(recursive_directory_iterator iter) noexcept;
     58     recursive_directory_iterator end(const recursive_directory_iterator&) noexcept;
     59 
     60     class file_status;
     61 
     62     struct space_info
     63     {
     64       uintmax_t capacity;
     65       uintmax_t free;
     66       uintmax_t available;
     67     };
     68 
     69     enum class file_type;
     70     enum class perms;
     71     enum class perm_options;
     72     enum class copy_options;
     73     enum class directory_options;
     74 
     75     typedef chrono::time_point<trivial-clock>  file_time_type;
     76 
     77     // operational functions
     78 
     79     path absolute(const path& p);
     80     path absolute(const path& p, error_code &ec);
     81 
     82     path canonical(const path& p);
     83     path canonical(const path& p, error_code& ec);
     84 
     85     void copy(const path& from, const path& to);
     86     void copy(const path& from, const path& to, error_code& ec);
     87     void copy(const path& from, const path& to, copy_options options);
     88     void copy(const path& from, const path& to, copy_options options,
     89                    error_code& ec);
     90 
     91     bool copy_file(const path& from, const path& to);
     92     bool copy_file(const path& from, const path& to, error_code& ec);
     93     bool copy_file(const path& from, const path& to, copy_options option);
     94     bool copy_file(const path& from, const path& to, copy_options option,
     95                            error_code& ec);
     96 
     97     void copy_symlink(const path& existing_symlink, const path& new_symlink);
     98     void copy_symlink(const path& existing_symlink, const path& new_symlink,
     99                               error_code& ec) noexcept;
    100 
    101     bool create_directories(const path& p);
    102     bool create_directories(const path& p, error_code& ec);
    103 
    104     bool create_directory(const path& p);
    105     bool create_directory(const path& p, error_code& ec) noexcept;
    106 
    107     bool create_directory(const path& p, const path& attributes);
    108     bool create_directory(const path& p, const path& attributes,
    109                                   error_code& ec) noexcept;
    110 
    111     void create_directory_symlink(const path& to, const path& new_symlink);
    112     void create_directory_symlink(const path& to, const path& new_symlink,
    113                                           error_code& ec) noexcept;
    114 
    115     void create_hard_link(const path& to, const path& new_hard_link);
    116     void create_hard_link(const path& to, const path& new_hard_link,
    117                                   error_code& ec) noexcept;
    118 
    119     void create_symlink(const path& to, const path& new_symlink);
    120     void create_symlink(const path& to, const path& new_symlink,
    121                                 error_code& ec) noexcept;
    122 
    123     path current_path();
    124     path current_path(error_code& ec);
    125     void current_path(const path& p);
    126     void current_path(const path& p, error_code& ec) noexcept;
    127 
    128     bool exists(file_status s) noexcept;
    129     bool exists(const path& p);
    130     bool exists(const path& p, error_code& ec) noexcept;
    131 
    132     bool equivalent(const path& p1, const path& p2);
    133     bool equivalent(const path& p1, const path& p2, error_code& ec) noexcept;
    134 
    135     uintmax_t    file_size(const path& p);
    136     uintmax_t    file_size(const path& p, error_code& ec) noexcept;
    137 
    138     uintmax_t    hard_link_count(const path& p);
    139     uintmax_t    hard_link_count(const path& p, error_code& ec) noexcept;
    140 
    141     bool is_block_file(file_status s) noexcept;
    142     bool is_block_file(const path& p);
    143     bool is_block_file(const path& p, error_code& ec) noexcept;
    144 
    145     bool is_character_file(file_status s) noexcept;
    146     bool is_character_file(const path& p);
    147     bool is_character_file(const path& p, error_code& ec) noexcept;
    148 
    149     bool is_directory(file_status s) noexcept;
    150     bool is_directory(const path& p);
    151     bool is_directory(const path& p, error_code& ec) noexcept;
    152 
    153     bool is_empty(const path& p);
    154     bool is_empty(const path& p, error_code& ec) noexcept;
    155 
    156     bool is_fifo(file_status s) noexcept;
    157     bool is_fifo(const path& p);
    158     bool is_fifo(const path& p, error_code& ec) noexcept;
    159 
    160     bool is_other(file_status s) noexcept;
    161     bool is_other(const path& p);
    162     bool is_other(const path& p, error_code& ec) noexcept;
    163 
    164     bool is_regular_file(file_status s) noexcept;
    165     bool is_regular_file(const path& p);
    166     bool is_regular_file(const path& p, error_code& ec) noexcept;
    167 
    168     bool is_socket(file_status s) noexcept;
    169     bool is_socket(const path& p);
    170     bool is_socket(const path& p, error_code& ec) noexcept;
    171 
    172     bool is_symlink(file_status s) noexcept;
    173     bool is_symlink(const path& p);
    174     bool is_symlink(const path& p, error_code& ec) noexcept;
    175 
    176     file_time_type  last_write_time(const path& p);
    177     file_time_type  last_write_time(const path& p, error_code& ec) noexcept;
    178     void last_write_time(const path& p, file_time_type new_time);
    179     void last_write_time(const path& p, file_time_type new_time,
    180                                  error_code& ec) noexcept;
    181 
    182     void permissions(const path& p, perms prms,
    183                      perm_options opts=perm_options::replace);
    184     void permissions(const path& p, perms prms, error_code& ec) noexcept;
    185     void permissions(const path& p, perms prms, perm_options opts,
    186                      error_code& ec);
    187 
    188     path proximate(const path& p, error_code& ec);
    189     path proximate(const path& p, const path& base = current_path());
    190     path proximate(const path& p, const path& base, error_code &ec);
    191 
    192     path read_symlink(const path& p);
    193     path read_symlink(const path& p, error_code& ec);
    194 
    195     path relative(const path& p, error_code& ec);
    196     path relative(const path& p, const path& base=current_path());
    197     path relative(const path& p, const path& base, error_code& ec);
    198 
    199     bool remove(const path& p);
    200     bool remove(const path& p, error_code& ec) noexcept;
    201 
    202     uintmax_t    remove_all(const path& p);
    203     uintmax_t    remove_all(const path& p, error_code& ec);
    204 
    205     void rename(const path& from, const path& to);
    206     void rename(const path& from, const path& to, error_code& ec) noexcept;
    207 
    208     void resize_file(const path& p, uintmax_t size);
    209     void resize_file(const path& p, uintmax_t size, error_code& ec) noexcept;
    210 
    211     space_info   space(const path& p);
    212     space_info   space(const path& p, error_code& ec) noexcept;
    213 
    214     file_status  status(const path& p);
    215     file_status  status(const path& p, error_code& ec) noexcept;
    216 
    217     bool status_known(file_status s) noexcept;
    218 
    219     file_status  symlink_status(const path& p);
    220     file_status  symlink_status(const path& p, error_code& ec) noexcept;
    221 
    222     path temp_directory_path();
    223     path temp_directory_path(error_code& ec);
    224 
    225     path weakly_canonical(path const& p);
    226     path weakly_canonical(path const& p, error_code& ec);
    227 
    228 
    229 } }  // namespaces std::filesystem
    230 
    231 */
    232 
    233 #include <__config>
    234 #include <cstddef>
    235 #include <cstdlib>
    236 #include <chrono>
    237 #include <iterator>
    238 #include <iosfwd>
    239 #include <locale>
    240 #include <memory>
    241 #include <stack>
    242 #include <string>
    243 #include <system_error>
    244 #include <utility>
    245 #include <iomanip> // for quoted
    246 #include <string_view>
    247 #include <version>
    248 
    249 #include <__debug>
    250 
    251 #if !defined(_LIBCPP_HAS_NO_PRAGMA_SYSTEM_HEADER)
    252 #pragma GCC system_header
    253 #endif
    254 
    255 _LIBCPP_PUSH_MACROS
    256 #include <__undef_macros>
    257 
    258 #ifndef _LIBCPP_CXX03_LANG
    259 
    260 _LIBCPP_BEGIN_NAMESPACE_FILESYSTEM
    261 
    262 typedef chrono::time_point<_FilesystemClock> file_time_type;
    263 
    264 struct _LIBCPP_TYPE_VIS space_info {
    265   uintmax_t capacity;
    266   uintmax_t free;
    267   uintmax_t available;
    268 };
    269 
    270 enum class _LIBCPP_ENUM_VIS file_type : signed char {
    271   none = 0,
    272   not_found = -1,
    273   regular = 1,
    274   directory = 2,
    275   symlink = 3,
    276   block = 4,
    277   character = 5,
    278   fifo = 6,
    279   socket = 7,
    280   unknown = 8
    281 };
    282 
    283 enum class _LIBCPP_ENUM_VIS perms : unsigned {
    284   none = 0,
    285 
    286   owner_read = 0400,
    287   owner_write = 0200,
    288   owner_exec = 0100,
    289   owner_all = 0700,
    290 
    291   group_read = 040,
    292   group_write = 020,
    293   group_exec = 010,
    294   group_all = 070,
    295 
    296   others_read = 04,
    297   others_write = 02,
    298   others_exec = 01,
    299   others_all = 07,
    300 
    301   all = 0777,
    302 
    303   set_uid = 04000,
    304   set_gid = 02000,
    305   sticky_bit = 01000,
    306   mask = 07777,
    307   unknown = 0xFFFF,
    308 };
    309 
    310 _LIBCPP_INLINE_VISIBILITY
    311 inline constexpr perms operator&(perms _LHS, perms _RHS) {
    312   return static_cast<perms>(static_cast<unsigned>(_LHS) &
    313                             static_cast<unsigned>(_RHS));
    314 }
    315 
    316 _LIBCPP_INLINE_VISIBILITY
    317 inline constexpr perms operator|(perms _LHS, perms _RHS) {
    318   return static_cast<perms>(static_cast<unsigned>(_LHS) |
    319                             static_cast<unsigned>(_RHS));
    320 }
    321 
    322 _LIBCPP_INLINE_VISIBILITY
    323 inline constexpr perms operator^(perms _LHS, perms _RHS) {
    324   return static_cast<perms>(static_cast<unsigned>(_LHS) ^
    325                             static_cast<unsigned>(_RHS));
    326 }
    327 
    328 _LIBCPP_INLINE_VISIBILITY
    329 inline constexpr perms operator~(perms _LHS) {
    330   return static_cast<perms>(~static_cast<unsigned>(_LHS));
    331 }
    332 
    333 _LIBCPP_INLINE_VISIBILITY
    334 inline perms& operator&=(perms& _LHS, perms _RHS) { return _LHS = _LHS & _RHS; }
    335 
    336 _LIBCPP_INLINE_VISIBILITY
    337 inline perms& operator|=(perms& _LHS, perms _RHS) { return _LHS = _LHS | _RHS; }
    338 
    339 _LIBCPP_INLINE_VISIBILITY
    340 inline perms& operator^=(perms& _LHS, perms _RHS) { return _LHS = _LHS ^ _RHS; }
    341 
    342 enum class _LIBCPP_ENUM_VIS perm_options : unsigned char {
    343   replace = 1,
    344   add = 2,
    345   remove = 4,
    346   nofollow = 8
    347 };
    348 
    349 _LIBCPP_INLINE_VISIBILITY
    350 inline constexpr perm_options operator&(perm_options _LHS, perm_options _RHS) {
    351   return static_cast<perm_options>(static_cast<unsigned>(_LHS) &
    352                                    static_cast<unsigned>(_RHS));
    353 }
    354 
    355 _LIBCPP_INLINE_VISIBILITY
    356 inline constexpr perm_options operator|(perm_options _LHS, perm_options _RHS) {
    357   return static_cast<perm_options>(static_cast<unsigned>(_LHS) |
    358                                    static_cast<unsigned>(_RHS));
    359 }
    360 
    361 _LIBCPP_INLINE_VISIBILITY
    362 inline constexpr perm_options operator^(perm_options _LHS, perm_options _RHS) {
    363   return static_cast<perm_options>(static_cast<unsigned>(_LHS) ^
    364                                    static_cast<unsigned>(_RHS));
    365 }
    366 
    367 _LIBCPP_INLINE_VISIBILITY
    368 inline constexpr perm_options operator~(perm_options _LHS) {
    369   return static_cast<perm_options>(~static_cast<unsigned>(_LHS));
    370 }
    371 
    372 _LIBCPP_INLINE_VISIBILITY
    373 inline perm_options& operator&=(perm_options& _LHS, perm_options _RHS) {
    374   return _LHS = _LHS & _RHS;
    375 }
    376 
    377 _LIBCPP_INLINE_VISIBILITY
    378 inline perm_options& operator|=(perm_options& _LHS, perm_options _RHS) {
    379   return _LHS = _LHS | _RHS;
    380 }
    381 
    382 _LIBCPP_INLINE_VISIBILITY
    383 inline perm_options& operator^=(perm_options& _LHS, perm_options _RHS) {
    384   return _LHS = _LHS ^ _RHS;
    385 }
    386 
    387 enum class _LIBCPP_ENUM_VIS copy_options : unsigned short {
    388   none = 0,
    389   skip_existing = 1,
    390   overwrite_existing = 2,
    391   update_existing = 4,
    392   recursive = 8,
    393   copy_symlinks = 16,
    394   skip_symlinks = 32,
    395   directories_only = 64,
    396   create_symlinks = 128,
    397   create_hard_links = 256,
    398   __in_recursive_copy = 512,
    399 };
    400 
    401 _LIBCPP_INLINE_VISIBILITY
    402 inline constexpr copy_options operator&(copy_options _LHS, copy_options _RHS) {
    403   return static_cast<copy_options>(static_cast<unsigned short>(_LHS) &
    404                                    static_cast<unsigned short>(_RHS));
    405 }
    406 
    407 _LIBCPP_INLINE_VISIBILITY
    408 inline constexpr copy_options operator|(copy_options _LHS, copy_options _RHS) {
    409   return static_cast<copy_options>(static_cast<unsigned short>(_LHS) |
    410                                    static_cast<unsigned short>(_RHS));
    411 }
    412 
    413 _LIBCPP_INLINE_VISIBILITY
    414 inline constexpr copy_options operator^(copy_options _LHS, copy_options _RHS) {
    415   return static_cast<copy_options>(static_cast<unsigned short>(_LHS) ^
    416                                    static_cast<unsigned short>(_RHS));
    417 }
    418 
    419 _LIBCPP_INLINE_VISIBILITY
    420 inline constexpr copy_options operator~(copy_options _LHS) {
    421   return static_cast<copy_options>(~static_cast<unsigned short>(_LHS));
    422 }
    423 
    424 _LIBCPP_INLINE_VISIBILITY
    425 inline copy_options& operator&=(copy_options& _LHS, copy_options _RHS) {
    426   return _LHS = _LHS & _RHS;
    427 }
    428 
    429 _LIBCPP_INLINE_VISIBILITY
    430 inline copy_options& operator|=(copy_options& _LHS, copy_options _RHS) {
    431   return _LHS = _LHS | _RHS;
    432 }
    433 
    434 _LIBCPP_INLINE_VISIBILITY
    435 inline copy_options& operator^=(copy_options& _LHS, copy_options _RHS) {
    436   return _LHS = _LHS ^ _RHS;
    437 }
    438 
    439 enum class _LIBCPP_ENUM_VIS directory_options : unsigned char {
    440   none = 0,
    441   follow_directory_symlink = 1,
    442   skip_permission_denied = 2
    443 };
    444 
    445 _LIBCPP_INLINE_VISIBILITY
    446 inline constexpr directory_options operator&(directory_options _LHS,
    447                                              directory_options _RHS) {
    448   return static_cast<directory_options>(static_cast<unsigned char>(_LHS) &
    449                                         static_cast<unsigned char>(_RHS));
    450 }
    451 
    452 _LIBCPP_INLINE_VISIBILITY
    453 inline constexpr directory_options operator|(directory_options _LHS,
    454                                              directory_options _RHS) {
    455   return static_cast<directory_options>(static_cast<unsigned char>(_LHS) |
    456                                         static_cast<unsigned char>(_RHS));
    457 }
    458 
    459 _LIBCPP_INLINE_VISIBILITY
    460 inline constexpr directory_options operator^(directory_options _LHS,
    461                                              directory_options _RHS) {
    462   return static_cast<directory_options>(static_cast<unsigned char>(_LHS) ^
    463                                         static_cast<unsigned char>(_RHS));
    464 }
    465 
    466 _LIBCPP_INLINE_VISIBILITY
    467 inline constexpr directory_options operator~(directory_options _LHS) {
    468   return static_cast<directory_options>(~static_cast<unsigned char>(_LHS));
    469 }
    470 
    471 _LIBCPP_INLINE_VISIBILITY
    472 inline directory_options& operator&=(directory_options& _LHS,
    473                                      directory_options _RHS) {
    474   return _LHS = _LHS & _RHS;
    475 }
    476 
    477 _LIBCPP_INLINE_VISIBILITY
    478 inline directory_options& operator|=(directory_options& _LHS,
    479                                      directory_options _RHS) {
    480   return _LHS = _LHS | _RHS;
    481 }
    482 
    483 _LIBCPP_INLINE_VISIBILITY
    484 inline directory_options& operator^=(directory_options& _LHS,
    485                                      directory_options _RHS) {
    486   return _LHS = _LHS ^ _RHS;
    487 }
    488 
    489 class _LIBCPP_TYPE_VIS file_status {
    490 public:
    491   // constructors
    492   _LIBCPP_INLINE_VISIBILITY
    493   file_status() noexcept : file_status(file_type::none) {}
    494   _LIBCPP_INLINE_VISIBILITY
    495   explicit file_status(file_type __ft, perms __prms = perms::unknown) noexcept
    496       : __ft_(__ft),
    497         __prms_(__prms) {}
    498 
    499   file_status(const file_status&) noexcept = default;
    500   file_status(file_status&&) noexcept = default;
    501 
    502   _LIBCPP_INLINE_VISIBILITY
    503   ~file_status() {}
    504 
    505   file_status& operator=(const file_status&) noexcept = default;
    506   file_status& operator=(file_status&&) noexcept = default;
    507 
    508   // observers
    509   _LIBCPP_INLINE_VISIBILITY
    510   file_type type() const noexcept { return __ft_; }
    511 
    512   _LIBCPP_INLINE_VISIBILITY
    513   perms permissions() const noexcept { return __prms_; }
    514 
    515   // modifiers
    516   _LIBCPP_INLINE_VISIBILITY
    517   void type(file_type __ft) noexcept { __ft_ = __ft; }
    518 
    519   _LIBCPP_INLINE_VISIBILITY
    520   void permissions(perms __p) noexcept { __prms_ = __p; }
    521 
    522 private:
    523   file_type __ft_;
    524   perms __prms_;
    525 };
    526 
    527 class _LIBCPP_TYPE_VIS directory_entry;
    528 
    529 template <class _Tp>
    530 struct __can_convert_char {
    531   static const bool value = false;
    532 };
    533 template <class _Tp>
    534 struct __can_convert_char<const _Tp> : public __can_convert_char<_Tp> {};
    535 template <>
    536 struct __can_convert_char<char> {
    537   static const bool value = true;
    538   using __char_type = char;
    539 };
    540 template <>
    541 struct __can_convert_char<wchar_t> {
    542   static const bool value = true;
    543   using __char_type = wchar_t;
    544 };
    545 template <>
    546 struct __can_convert_char<char16_t> {
    547   static const bool value = true;
    548   using __char_type = char16_t;
    549 };
    550 template <>
    551 struct __can_convert_char<char32_t> {
    552   static const bool value = true;
    553   using __char_type = char32_t;
    554 };
    555 
    556 template <class _ECharT>
    557 typename enable_if<__can_convert_char<_ECharT>::value, bool>::type
    558 __is_separator(_ECharT __e) {
    559   return __e == _ECharT('/');
    560 }
    561 
    562 struct _NullSentinal {};
    563 
    564 template <class _Tp>
    565 using _Void = void;
    566 
    567 template <class _Tp, class = void>
    568 struct __is_pathable_string : public false_type {};
    569 
    570 template <class _ECharT, class _Traits, class _Alloc>
    571 struct __is_pathable_string<
    572     basic_string<_ECharT, _Traits, _Alloc>,
    573     _Void<typename __can_convert_char<_ECharT>::__char_type> >
    574     : public __can_convert_char<_ECharT> {
    575   using _Str = basic_string<_ECharT, _Traits, _Alloc>;
    576   using _Base = __can_convert_char<_ECharT>;
    577   static _ECharT const* __range_begin(_Str const& __s) { return __s.data(); }
    578   static _ECharT const* __range_end(_Str const& __s) {
    579     return __s.data() + __s.length();
    580   }
    581   static _ECharT __first_or_null(_Str const& __s) {
    582     return __s.empty() ? _ECharT{} : __s[0];
    583   }
    584 };
    585 
    586 template <class _ECharT, class _Traits>
    587 struct __is_pathable_string<
    588     basic_string_view<_ECharT, _Traits>,
    589     _Void<typename __can_convert_char<_ECharT>::__char_type> >
    590     : public __can_convert_char<_ECharT> {
    591   using _Str = basic_string_view<_ECharT, _Traits>;
    592   using _Base = __can_convert_char<_ECharT>;
    593   static _ECharT const* __range_begin(_Str const& __s) { return __s.data(); }
    594   static _ECharT const* __range_end(_Str const& __s) {
    595     return __s.data() + __s.length();
    596   }
    597   static _ECharT __first_or_null(_Str const& __s) {
    598     return __s.empty() ? _ECharT{} : __s[0];
    599   }
    600 };
    601 
    602 template <class _Source, class _DS = typename decay<_Source>::type,
    603           class _UnqualPtrType =
    604               typename remove_const<typename remove_pointer<_DS>::type>::type,
    605           bool _IsCharPtr = is_pointer<_DS>::value&&
    606               __can_convert_char<_UnqualPtrType>::value>
    607 struct __is_pathable_char_array : false_type {};
    608 
    609 template <class _Source, class _ECharT, class _UPtr>
    610 struct __is_pathable_char_array<_Source, _ECharT*, _UPtr, true>
    611     : __can_convert_char<typename remove_const<_ECharT>::type> {
    612   using _Base = __can_convert_char<typename remove_const<_ECharT>::type>;
    613 
    614   static _ECharT const* __range_begin(const _ECharT* __b) { return __b; }
    615   static _ECharT const* __range_end(const _ECharT* __b) {
    616     using _Iter = const _ECharT*;
    617     const _ECharT __sentinal = _ECharT{};
    618     _Iter __e = __b;
    619     for (; *__e != __sentinal; ++__e)
    620       ;
    621     return __e;
    622   }
    623 
    624   static _ECharT __first_or_null(const _ECharT* __b) { return *__b; }
    625 };
    626 
    627 template <class _Iter, bool _IsIt = __is_input_iterator<_Iter>::value,
    628           class = void>
    629 struct __is_pathable_iter : false_type {};
    630 
    631 template <class _Iter>
    632 struct __is_pathable_iter<
    633     _Iter, true,
    634     _Void<typename __can_convert_char<
    635         typename iterator_traits<_Iter>::value_type>::__char_type> >
    636     : __can_convert_char<typename iterator_traits<_Iter>::value_type> {
    637   using _ECharT = typename iterator_traits<_Iter>::value_type;
    638   using _Base = __can_convert_char<_ECharT>;
    639 
    640   static _Iter __range_begin(_Iter __b) { return __b; }
    641   static _NullSentinal __range_end(_Iter) { return _NullSentinal{}; }
    642 
    643   static _ECharT __first_or_null(_Iter __b) { return *__b; }
    644 };
    645 
    646 template <class _Tp, bool _IsStringT = __is_pathable_string<_Tp>::value,
    647           bool _IsCharIterT = __is_pathable_char_array<_Tp>::value,
    648           bool _IsIterT = !_IsCharIterT && __is_pathable_iter<_Tp>::value>
    649 struct __is_pathable : false_type {
    650   static_assert(!_IsStringT && !_IsCharIterT && !_IsIterT, "Must all be false");
    651 };
    652 
    653 template <class _Tp>
    654 struct __is_pathable<_Tp, true, false, false> : __is_pathable_string<_Tp> {};
    655 
    656 template <class _Tp>
    657 struct __is_pathable<_Tp, false, true, false> : __is_pathable_char_array<_Tp> {
    658 };
    659 
    660 template <class _Tp>
    661 struct __is_pathable<_Tp, false, false, true> : __is_pathable_iter<_Tp> {};
    662 
    663 template <class _ECharT>
    664 struct _PathCVT {
    665   static_assert(__can_convert_char<_ECharT>::value,
    666                 "Char type not convertible");
    667 
    668   typedef __narrow_to_utf8<sizeof(_ECharT) * __CHAR_BIT__> _Narrower;
    669 
    670   static void __append_range(string& __dest, _ECharT const* __b,
    671                              _ECharT const* __e) {
    672     _Narrower()(back_inserter(__dest), __b, __e);
    673   }
    674 
    675   template <class _Iter>
    676   static void __append_range(string& __dest, _Iter __b, _Iter __e) {
    677     static_assert(!is_same<_Iter, _ECharT*>::value, "Call const overload");
    678     if (__b == __e)
    679       return;
    680     basic_string<_ECharT> __tmp(__b, __e);
    681     _Narrower()(back_inserter(__dest), __tmp.data(),
    682                 __tmp.data() + __tmp.length());
    683   }
    684 
    685   template <class _Iter>
    686   static void __append_range(string& __dest, _Iter __b, _NullSentinal) {
    687     static_assert(!is_same<_Iter, _ECharT*>::value, "Call const overload");
    688     const _ECharT __sentinal = _ECharT{};
    689     if (*__b == __sentinal)
    690       return;
    691     basic_string<_ECharT> __tmp;
    692     for (; *__b != __sentinal; ++__b)
    693       __tmp.push_back(*__b);
    694     _Narrower()(back_inserter(__dest), __tmp.data(),
    695                 __tmp.data() + __tmp.length());
    696   }
    697 
    698   template <class _Source>
    699   static void __append_source(string& __dest, _Source const& __s) {
    700     using _Traits = __is_pathable<_Source>;
    701     __append_range(__dest, _Traits::__range_begin(__s),
    702                    _Traits::__range_end(__s));
    703   }
    704 };
    705 
    706 template <>
    707 struct _PathCVT<char> {
    708 
    709   template <class _Iter>
    710   static typename enable_if<__is_exactly_input_iterator<_Iter>::value>::type
    711   __append_range(string& __dest, _Iter __b, _Iter __e) {
    712     for (; __b != __e; ++__b)
    713       __dest.push_back(*__b);
    714   }
    715 
    716   template <class _Iter>
    717   static typename enable_if<__is_forward_iterator<_Iter>::value>::type
    718   __append_range(string& __dest, _Iter __b, _Iter __e) {
    719     __dest.__append_forward_unsafe(__b, __e);
    720   }
    721 
    722   template <class _Iter>
    723   static void __append_range(string& __dest, _Iter __b, _NullSentinal) {
    724     const char __sentinal = char{};
    725     for (; *__b != __sentinal; ++__b)
    726       __dest.push_back(*__b);
    727   }
    728 
    729   template <class _Source>
    730   static void __append_source(string& __dest, _Source const& __s) {
    731     using _Traits = __is_pathable<_Source>;
    732     __append_range(__dest, _Traits::__range_begin(__s),
    733                    _Traits::__range_end(__s));
    734   }
    735 };
    736 
    737 class _LIBCPP_TYPE_VIS path {
    738   template <class _SourceOrIter, class _Tp = path&>
    739   using _EnableIfPathable =
    740       typename enable_if<__is_pathable<_SourceOrIter>::value, _Tp>::type;
    741 
    742   template <class _Tp>
    743   using _SourceChar = typename __is_pathable<_Tp>::__char_type;
    744 
    745   template <class _Tp>
    746   using _SourceCVT = _PathCVT<_SourceChar<_Tp> >;
    747 
    748 public:
    749   typedef char value_type;
    750   typedef basic_string<value_type> string_type;
    751   typedef _VSTD::string_view __string_view;
    752   static constexpr value_type preferred_separator = '/';
    753 
    754   enum class _LIBCPP_ENUM_VIS format : unsigned char {
    755     auto_format,
    756     native_format,
    757     generic_format
    758   };
    759 
    760   // constructors and destructor
    761   _LIBCPP_INLINE_VISIBILITY path() noexcept {}
    762   _LIBCPP_INLINE_VISIBILITY path(const path& __p) : __pn_(__p.__pn_) {}
    763   _LIBCPP_INLINE_VISIBILITY path(path&& __p) noexcept
    764       : __pn_(_VSTD::move(__p.__pn_)) {}
    765 
    766   _LIBCPP_INLINE_VISIBILITY
    767   path(string_type&& __s, format = format::auto_format) noexcept
    768       : __pn_(_VSTD::move(__s)) {}
    769 
    770   template <class _Source, class = _EnableIfPathable<_Source, void> >
    771   path(const _Source& __src, format = format::auto_format) {
    772     _SourceCVT<_Source>::__append_source(__pn_, __src);
    773   }
    774 
    775   template <class _InputIt>
    776   path(_InputIt __first, _InputIt __last, format = format::auto_format) {
    777     typedef typename iterator_traits<_InputIt>::value_type _ItVal;
    778     _PathCVT<_ItVal>::__append_range(__pn_, __first, __last);
    779   }
    780 
    781   // TODO Implement locale conversions.
    782   template <class _Source, class = _EnableIfPathable<_Source, void> >
    783   path(const _Source& __src, const locale& __loc, format = format::auto_format);
    784   template <class _InputIt>
    785   path(_InputIt __first, _InputIt _last, const locale& __loc,
    786        format = format::auto_format);
    787 
    788   _LIBCPP_INLINE_VISIBILITY
    789   ~path() = default;
    790 
    791   // assignments
    792   _LIBCPP_INLINE_VISIBILITY
    793   path& operator=(const path& __p) {
    794     __pn_ = __p.__pn_;
    795     return *this;
    796   }
    797 
    798   _LIBCPP_INLINE_VISIBILITY
    799   path& operator=(path&& __p) noexcept {
    800     __pn_ = _VSTD::move(__p.__pn_);
    801     return *this;
    802   }
    803 
    804   template <class = void>
    805   _LIBCPP_INLINE_VISIBILITY path& operator=(string_type&& __s) noexcept {
    806     __pn_ = _VSTD::move(__s);
    807     return *this;
    808   }
    809 
    810   _LIBCPP_INLINE_VISIBILITY
    811   path& assign(string_type&& __s) noexcept {
    812     __pn_ = _VSTD::move(__s);
    813     return *this;
    814   }
    815 
    816   template <class _Source>
    817   _LIBCPP_INLINE_VISIBILITY _EnableIfPathable<_Source>
    818   operator=(const _Source& __src) {
    819     return this->assign(__src);
    820   }
    821 
    822   template <class _Source>
    823   _EnableIfPathable<_Source> assign(const _Source& __src) {
    824     __pn_.clear();
    825     _SourceCVT<_Source>::__append_source(__pn_, __src);
    826     return *this;
    827   }
    828 
    829   template <class _InputIt>
    830   path& assign(_InputIt __first, _InputIt __last) {
    831     typedef typename iterator_traits<_InputIt>::value_type _ItVal;
    832     __pn_.clear();
    833     _PathCVT<_ItVal>::__append_range(__pn_, __first, __last);
    834     return *this;
    835   }
    836 
    837 private:
    838   template <class _ECharT>
    839   static bool __source_is_absolute(_ECharT __first_or_null) {
    840     return __is_separator(__first_or_null);
    841   }
    842 
    843 public:
    844   // appends
    845   path& operator/=(const path& __p) {
    846     if (__p.is_absolute()) {
    847       __pn_ = __p.__pn_;
    848       return *this;
    849     }
    850     if (has_filename())
    851       __pn_ += preferred_separator;
    852     __pn_ += __p.native();
    853     return *this;
    854   }
    855 
    856   // FIXME: Use _LIBCPP_DIAGNOSE_WARNING to produce a diagnostic when __src
    857   // is known at compile time to be "/' since the user almost certainly intended
    858   // to append a separator instead of overwriting the path with "/"
    859   template <class _Source>
    860   _LIBCPP_INLINE_VISIBILITY _EnableIfPathable<_Source>
    861   operator/=(const _Source& __src) {
    862     return this->append(__src);
    863   }
    864 
    865   template <class _Source>
    866   _EnableIfPathable<_Source> append(const _Source& __src) {
    867     using _Traits = __is_pathable<_Source>;
    868     using _CVT = _PathCVT<_SourceChar<_Source> >;
    869     if (__source_is_absolute(_Traits::__first_or_null(__src)))
    870       __pn_.clear();
    871     else if (has_filename())
    872       __pn_ += preferred_separator;
    873     _CVT::__append_source(__pn_, __src);
    874     return *this;
    875   }
    876 
    877   template <class _InputIt>
    878   path& append(_InputIt __first, _InputIt __last) {
    879     typedef typename iterator_traits<_InputIt>::value_type _ItVal;
    880     static_assert(__can_convert_char<_ItVal>::value, "Must convertible");
    881     using _CVT = _PathCVT<_ItVal>;
    882     if (__first != __last && __source_is_absolute(*__first))
    883       __pn_.clear();
    884     else if (has_filename())
    885       __pn_ += preferred_separator;
    886     _CVT::__append_range(__pn_, __first, __last);
    887     return *this;
    888   }
    889 
    890   // concatenation
    891   _LIBCPP_INLINE_VISIBILITY
    892   path& operator+=(const path& __x) {
    893     __pn_ += __x.__pn_;
    894     return *this;
    895   }
    896 
    897   _LIBCPP_INLINE_VISIBILITY
    898   path& operator+=(const string_type& __x) {
    899     __pn_ += __x;
    900     return *this;
    901   }
    902 
    903   _LIBCPP_INLINE_VISIBILITY
    904   path& operator+=(__string_view __x) {
    905     __pn_ += __x;
    906     return *this;
    907   }
    908 
    909   _LIBCPP_INLINE_VISIBILITY
    910   path& operator+=(const value_type* __x) {
    911     __pn_ += __x;
    912     return *this;
    913   }
    914 
    915   _LIBCPP_INLINE_VISIBILITY
    916   path& operator+=(value_type __x) {
    917     __pn_ += __x;
    918     return *this;
    919   }
    920 
    921   template <class _ECharT>
    922   typename enable_if<__can_convert_char<_ECharT>::value, path&>::type
    923   operator+=(_ECharT __x) {
    924     basic_string<_ECharT> __tmp;
    925     __tmp += __x;
    926     _PathCVT<_ECharT>::__append_source(__pn_, __tmp);
    927     return *this;
    928   }
    929 
    930   template <class _Source>
    931   _EnableIfPathable<_Source> operator+=(const _Source& __x) {
    932     return this->concat(__x);
    933   }
    934 
    935   template <class _Source>
    936   _EnableIfPathable<_Source> concat(const _Source& __x) {
    937     _SourceCVT<_Source>::__append_source(__pn_, __x);
    938     return *this;
    939   }
    940 
    941   template <class _InputIt>
    942   path& concat(_InputIt __first, _InputIt __last) {
    943     typedef typename iterator_traits<_InputIt>::value_type _ItVal;
    944     _PathCVT<_ItVal>::__append_range(__pn_, __first, __last);
    945     return *this;
    946   }
    947 
    948   // modifiers
    949   _LIBCPP_INLINE_VISIBILITY
    950   void clear() noexcept { __pn_.clear(); }
    951 
    952   path& make_preferred() { return *this; }
    953 
    954   _LIBCPP_INLINE_VISIBILITY
    955   path& remove_filename() {
    956     auto __fname = __filename();
    957     if (!__fname.empty())
    958       __pn_.erase(__fname.data() - __pn_.data());
    959     return *this;
    960   }
    961 
    962   path& replace_filename(const path& __replacement) {
    963     remove_filename();
    964     return (*this /= __replacement);
    965   }
    966 
    967   path& replace_extension(const path& __replacement = path());
    968 
    969   _LIBCPP_INLINE_VISIBILITY
    970   void swap(path& __rhs) noexcept { __pn_.swap(__rhs.__pn_); }
    971 
    972   // private helper to allow reserving memory in the path
    973   _LIBCPP_INLINE_VISIBILITY
    974   void __reserve(size_t __s) { __pn_.reserve(__s); }
    975 
    976   // native format observers
    977   _LIBCPP_INLINE_VISIBILITY
    978   const string_type& native() const noexcept { return __pn_; }
    979 
    980   _LIBCPP_INLINE_VISIBILITY
    981   const value_type* c_str() const noexcept { return __pn_.c_str(); }
    982 
    983   _LIBCPP_INLINE_VISIBILITY operator string_type() const { return __pn_; }
    984 
    985   template <class _ECharT, class _Traits = char_traits<_ECharT>,
    986             class _Allocator = allocator<_ECharT> >
    987   basic_string<_ECharT, _Traits, _Allocator>
    988   string(const _Allocator& __a = _Allocator()) const {
    989     using _CVT = __widen_from_utf8<sizeof(_ECharT) * __CHAR_BIT__>;
    990     using _Str = basic_string<_ECharT, _Traits, _Allocator>;
    991     _Str __s(__a);
    992     __s.reserve(__pn_.size());
    993     _CVT()(back_inserter(__s), __pn_.data(), __pn_.data() + __pn_.size());
    994     return __s;
    995   }
    996 
    997   _LIBCPP_INLINE_VISIBILITY std::string string() const { return __pn_; }
    998   _LIBCPP_INLINE_VISIBILITY std::wstring wstring() const {
    999     return string<wchar_t>();
   1000   }
   1001   _LIBCPP_INLINE_VISIBILITY std::string u8string() const { return __pn_; }
   1002   _LIBCPP_INLINE_VISIBILITY std::u16string u16string() const {
   1003     return string<char16_t>();
   1004   }
   1005   _LIBCPP_INLINE_VISIBILITY std::u32string u32string() const {
   1006     return string<char32_t>();
   1007   }
   1008 
   1009   // generic format observers
   1010   template <class _ECharT, class _Traits = char_traits<_ECharT>,
   1011             class _Allocator = allocator<_ECharT> >
   1012   basic_string<_ECharT, _Traits, _Allocator>
   1013   generic_string(const _Allocator& __a = _Allocator()) const {
   1014     return string<_ECharT, _Traits, _Allocator>(__a);
   1015   }
   1016 
   1017   std::string generic_string() const { return __pn_; }
   1018   std::wstring generic_wstring() const { return string<wchar_t>(); }
   1019   std::string generic_u8string() const { return __pn_; }
   1020   std::u16string generic_u16string() const { return string<char16_t>(); }
   1021   std::u32string generic_u32string() const { return string<char32_t>(); }
   1022 
   1023 private:
   1024   int __compare(__string_view) const;
   1025   __string_view __root_name() const;
   1026   __string_view __root_directory() const;
   1027   __string_view __root_path_raw() const;
   1028   __string_view __relative_path() const;
   1029   __string_view __parent_path() const;
   1030   __string_view __filename() const;
   1031   __string_view __stem() const;
   1032   __string_view __extension() const;
   1033 
   1034 public:
   1035   // compare
   1036   _LIBCPP_INLINE_VISIBILITY int compare(const path& __p) const noexcept {
   1037     return __compare(__p.__pn_);
   1038   }
   1039   _LIBCPP_INLINE_VISIBILITY int compare(const string_type& __s) const {
   1040     return __compare(__s);
   1041   }
   1042   _LIBCPP_INLINE_VISIBILITY int compare(__string_view __s) const {
   1043     return __compare(__s);
   1044   }
   1045   _LIBCPP_INLINE_VISIBILITY int compare(const value_type* __s) const {
   1046     return __compare(__s);
   1047   }
   1048 
   1049   // decomposition
   1050   _LIBCPP_INLINE_VISIBILITY path root_name() const {
   1051     return string_type(__root_name());
   1052   }
   1053   _LIBCPP_INLINE_VISIBILITY path root_directory() const {
   1054     return string_type(__root_directory());
   1055   }
   1056   _LIBCPP_INLINE_VISIBILITY path root_path() const {
   1057     return root_name().append(string_type(__root_directory()));
   1058   }
   1059   _LIBCPP_INLINE_VISIBILITY path relative_path() const {
   1060     return string_type(__relative_path());
   1061   }
   1062   _LIBCPP_INLINE_VISIBILITY path parent_path() const {
   1063     return string_type(__parent_path());
   1064   }
   1065   _LIBCPP_INLINE_VISIBILITY path filename() const {
   1066     return string_type(__filename());
   1067   }
   1068   _LIBCPP_INLINE_VISIBILITY path stem() const { return string_type(__stem()); }
   1069   _LIBCPP_INLINE_VISIBILITY path extension() const {
   1070     return string_type(__extension());
   1071   }
   1072 
   1073   // query
   1074   _LIBCPP_NODISCARD_AFTER_CXX17 _LIBCPP_INLINE_VISIBILITY bool
   1075   empty() const noexcept {
   1076     return __pn_.empty();
   1077   }
   1078 
   1079   _LIBCPP_INLINE_VISIBILITY bool has_root_name() const {
   1080     return !__root_name().empty();
   1081   }
   1082   _LIBCPP_INLINE_VISIBILITY bool has_root_directory() const {
   1083     return !__root_directory().empty();
   1084   }
   1085   _LIBCPP_INLINE_VISIBILITY bool has_root_path() const {
   1086     return !__root_path_raw().empty();
   1087   }
   1088   _LIBCPP_INLINE_VISIBILITY bool has_relative_path() const {
   1089     return !__relative_path().empty();
   1090   }
   1091   _LIBCPP_INLINE_VISIBILITY bool has_parent_path() const {
   1092     return !__parent_path().empty();
   1093   }
   1094   _LIBCPP_INLINE_VISIBILITY bool has_filename() const {
   1095     return !__filename().empty();
   1096   }
   1097   _LIBCPP_INLINE_VISIBILITY bool has_stem() const { return !__stem().empty(); }
   1098   _LIBCPP_INLINE_VISIBILITY bool has_extension() const {
   1099     return !__extension().empty();
   1100   }
   1101 
   1102   _LIBCPP_INLINE_VISIBILITY bool is_absolute() const {
   1103     return has_root_directory();
   1104   }
   1105   _LIBCPP_INLINE_VISIBILITY bool is_relative() const { return !is_absolute(); }
   1106 
   1107   // relative paths
   1108   path lexically_normal() const;
   1109   path lexically_relative(const path& __base) const;
   1110 
   1111   _LIBCPP_INLINE_VISIBILITY path lexically_proximate(const path& __base) const {
   1112     path __result = this->lexically_relative(__base);
   1113     if (__result.native().empty())
   1114       return *this;
   1115     return __result;
   1116   }
   1117 
   1118   // iterators
   1119   class _LIBCPP_TYPE_VIS iterator;
   1120   typedef iterator const_iterator;
   1121 
   1122   iterator begin() const;
   1123   iterator end() const;
   1124 
   1125   template <class _CharT, class _Traits>
   1126   _LIBCPP_INLINE_VISIBILITY friend
   1127       typename enable_if<is_same<_CharT, char>::value &&
   1128                              is_same<_Traits, char_traits<char> >::value,
   1129                          basic_ostream<_CharT, _Traits>&>::type
   1130       operator<<(basic_ostream<_CharT, _Traits>& __os, const path& __p) {
   1131     __os << std::__quoted(__p.native());
   1132     return __os;
   1133   }
   1134 
   1135   template <class _CharT, class _Traits>
   1136   _LIBCPP_INLINE_VISIBILITY friend
   1137       typename enable_if<!is_same<_CharT, char>::value ||
   1138                              !is_same<_Traits, char_traits<char> >::value,
   1139                          basic_ostream<_CharT, _Traits>&>::type
   1140       operator<<(basic_ostream<_CharT, _Traits>& __os, const path& __p) {
   1141     __os << std::__quoted(__p.string<_CharT, _Traits>());
   1142     return __os;
   1143   }
   1144 
   1145   template <class _CharT, class _Traits>
   1146   _LIBCPP_INLINE_VISIBILITY friend basic_istream<_CharT, _Traits>&
   1147   operator>>(basic_istream<_CharT, _Traits>& __is, path& __p) {
   1148     basic_string<_CharT, _Traits> __tmp;
   1149     __is >> __quoted(__tmp);
   1150     __p = __tmp;
   1151     return __is;
   1152   }
   1153 
   1154   friend _LIBCPP_INLINE_VISIBILITY bool operator==(const path& __lhs, const path& __rhs) noexcept {
   1155     return __lhs.compare(__rhs) == 0;
   1156   }
   1157   friend _LIBCPP_INLINE_VISIBILITY bool operator!=(const path& __lhs, const path& __rhs) noexcept {
   1158     return __lhs.compare(__rhs) != 0;
   1159   }
   1160   friend _LIBCPP_INLINE_VISIBILITY bool operator<(const path& __lhs, const path& __rhs) noexcept {
   1161     return __lhs.compare(__rhs) < 0;
   1162   }
   1163   friend _LIBCPP_INLINE_VISIBILITY bool operator<=(const path& __lhs, const path& __rhs) noexcept {
   1164     return __lhs.compare(__rhs) <= 0;
   1165   }
   1166   friend _LIBCPP_INLINE_VISIBILITY bool operator>(const path& __lhs, const path& __rhs) noexcept {
   1167     return __lhs.compare(__rhs) > 0;
   1168   }
   1169   friend _LIBCPP_INLINE_VISIBILITY bool operator>=(const path& __lhs, const path& __rhs) noexcept {
   1170     return __lhs.compare(__rhs) >= 0;
   1171   }
   1172 
   1173   friend _LIBCPP_INLINE_VISIBILITY path operator/(const path& __lhs,
   1174                                                   const path& __rhs) {
   1175     path __result(__lhs);
   1176     __result /= __rhs;
   1177     return __result;
   1178   }
   1179 private:
   1180   inline _LIBCPP_INLINE_VISIBILITY path&
   1181   __assign_view(__string_view const& __s) noexcept {
   1182     __pn_ = string_type(__s);
   1183     return *this;
   1184   }
   1185   string_type __pn_;
   1186 };
   1187 
   1188 inline _LIBCPP_INLINE_VISIBILITY void swap(path& __lhs, path& __rhs) noexcept {
   1189   __lhs.swap(__rhs);
   1190 }
   1191 
   1192 _LIBCPP_FUNC_VIS
   1193 size_t hash_value(const path& __p) noexcept;
   1194 
   1195 template <class _Source>
   1196 _LIBCPP_INLINE_VISIBILITY
   1197     typename enable_if<__is_pathable<_Source>::value, path>::type
   1198     u8path(const _Source& __s) {
   1199   static_assert(
   1200       is_same<typename __is_pathable<_Source>::__char_type, char>::value,
   1201       "u8path(Source const&) requires Source have a character type of type "
   1202       "'char'");
   1203   return path(__s);
   1204 }
   1205 
   1206 template <class _InputIt>
   1207 _LIBCPP_INLINE_VISIBILITY
   1208     typename enable_if<__is_pathable<_InputIt>::value, path>::type
   1209     u8path(_InputIt __f, _InputIt __l) {
   1210   static_assert(
   1211       is_same<typename __is_pathable<_InputIt>::__char_type, char>::value,
   1212       "u8path(Iter, Iter) requires Iter have a value_type of type 'char'");
   1213   return path(__f, __l);
   1214 }
   1215 
   1216 class _LIBCPP_TYPE_VIS path::iterator {
   1217 public:
   1218   enum _ParserState : unsigned char {
   1219     _Singular,
   1220     _BeforeBegin,
   1221     _InRootName,
   1222     _InRootDir,
   1223     _InFilenames,
   1224     _InTrailingSep,
   1225     _AtEnd
   1226   };
   1227 
   1228 public:
   1229   typedef bidirectional_iterator_tag iterator_category;
   1230 
   1231   typedef path value_type;
   1232   typedef std::ptrdiff_t difference_type;
   1233   typedef const path* pointer;
   1234   typedef const path& reference;
   1235 
   1236   typedef void
   1237       __stashing_iterator_tag; // See reverse_iterator and __is_stashing_iterator
   1238 
   1239 public:
   1240   _LIBCPP_INLINE_VISIBILITY
   1241   iterator()
   1242       : __stashed_elem_(), __path_ptr_(nullptr), __entry_(),
   1243         __state_(_Singular) {}
   1244 
   1245   iterator(const iterator&) = default;
   1246   ~iterator() = default;
   1247 
   1248   iterator& operator=(const iterator&) = default;
   1249 
   1250   _LIBCPP_INLINE_VISIBILITY
   1251   reference operator*() const { return __stashed_elem_; }
   1252 
   1253   _LIBCPP_INLINE_VISIBILITY
   1254   pointer operator->() const { return &__stashed_elem_; }
   1255 
   1256   _LIBCPP_INLINE_VISIBILITY
   1257   iterator& operator++() {
   1258     _LIBCPP_ASSERT(__state_ != _Singular,
   1259                    "attempting to increment a singular iterator");
   1260     _LIBCPP_ASSERT(__state_ != _AtEnd,
   1261                    "attempting to increment the end iterator");
   1262     return __increment();
   1263   }
   1264 
   1265   _LIBCPP_INLINE_VISIBILITY
   1266   iterator operator++(int) {
   1267     iterator __it(*this);
   1268     this->operator++();
   1269     return __it;
   1270   }
   1271 
   1272   _LIBCPP_INLINE_VISIBILITY
   1273   iterator& operator--() {
   1274     _LIBCPP_ASSERT(__state_ != _Singular,
   1275                    "attempting to decrement a singular iterator");
   1276     _LIBCPP_ASSERT(__entry_.data() != __path_ptr_->native().data(),
   1277                    "attempting to decrement the begin iterator");
   1278     return __decrement();
   1279   }
   1280 
   1281   _LIBCPP_INLINE_VISIBILITY
   1282   iterator operator--(int) {
   1283     iterator __it(*this);
   1284     this->operator--();
   1285     return __it;
   1286   }
   1287 
   1288 private:
   1289   friend class path;
   1290 
   1291   inline _LIBCPP_INLINE_VISIBILITY friend bool operator==(const iterator&,
   1292                                                           const iterator&);
   1293 
   1294   iterator& __increment();
   1295   iterator& __decrement();
   1296 
   1297   path __stashed_elem_;
   1298   const path* __path_ptr_;
   1299   path::__string_view __entry_;
   1300   _ParserState __state_;
   1301 };
   1302 
   1303 inline _LIBCPP_INLINE_VISIBILITY bool operator==(const path::iterator& __lhs,
   1304                                                  const path::iterator& __rhs) {
   1305   return __lhs.__path_ptr_ == __rhs.__path_ptr_ &&
   1306          __lhs.__entry_.data() == __rhs.__entry_.data();
   1307 }
   1308 
   1309 inline _LIBCPP_INLINE_VISIBILITY bool operator!=(const path::iterator& __lhs,
   1310                                                  const path::iterator& __rhs) {
   1311   return !(__lhs == __rhs);
   1312 }
   1313 
   1314 class _LIBCPP_EXCEPTION_ABI filesystem_error : public system_error {
   1315 public:
   1316   _LIBCPP_INLINE_VISIBILITY
   1317   filesystem_error(const string& __what, error_code __ec)
   1318       : system_error(__ec, __what),
   1319         __storage_(make_shared<_Storage>(path(), path())) {
   1320     __create_what(0);
   1321   }
   1322 
   1323   _LIBCPP_INLINE_VISIBILITY
   1324   filesystem_error(const string& __what, const path& __p1, error_code __ec)
   1325       : system_error(__ec, __what),
   1326         __storage_(make_shared<_Storage>(__p1, path())) {
   1327     __create_what(1);
   1328   }
   1329 
   1330   _LIBCPP_INLINE_VISIBILITY
   1331   filesystem_error(const string& __what, const path& __p1, const path& __p2,
   1332                    error_code __ec)
   1333       : system_error(__ec, __what),
   1334         __storage_(make_shared<_Storage>(__p1, __p2)) {
   1335     __create_what(2);
   1336   }
   1337 
   1338   _LIBCPP_INLINE_VISIBILITY
   1339   const path& path1() const noexcept { return __storage_->__p1_; }
   1340 
   1341   _LIBCPP_INLINE_VISIBILITY
   1342   const path& path2() const noexcept { return __storage_->__p2_; }
   1343 
   1344   ~filesystem_error() override; // key function
   1345 
   1346   _LIBCPP_INLINE_VISIBILITY
   1347   const char* what() const noexcept override {
   1348     return __storage_->__what_.c_str();
   1349   }
   1350 
   1351   _LIBCPP_FUNC_VIS
   1352   void __create_what(int __num_paths);
   1353 
   1354 private:
   1355   struct _Storage {
   1356     _LIBCPP_INLINE_VISIBILITY
   1357     _Storage(const path& __p1, const path& __p2) : __p1_(__p1), __p2_(__p2) {}
   1358 
   1359     path __p1_;
   1360     path __p2_;
   1361     string __what_;
   1362   };
   1363   shared_ptr<_Storage> __storage_;
   1364 };
   1365 
   1366 template <class... _Args>
   1367 _LIBCPP_NORETURN inline _LIBCPP_INLINE_VISIBILITY
   1368 #ifndef _LIBCPP_NO_EXCEPTIONS
   1369     void
   1370     __throw_filesystem_error(_Args&&... __args) {
   1371   throw filesystem_error(std::forward<_Args>(__args)...);
   1372 }
   1373 #else
   1374     void
   1375     __throw_filesystem_error(_Args&&...) {
   1376   _VSTD::abort();
   1377 }
   1378 #endif
   1379 
   1380 // operational functions
   1381 
   1382 _LIBCPP_FUNC_VIS
   1383 path __absolute(const path&, error_code* __ec = nullptr);
   1384 _LIBCPP_FUNC_VIS
   1385 path __canonical(const path&, error_code* __ec = nullptr);
   1386 _LIBCPP_FUNC_VIS
   1387 void __copy(const path& __from, const path& __to, copy_options __opt,
   1388             error_code* __ec = nullptr);
   1389 _LIBCPP_FUNC_VIS
   1390 bool __copy_file(const path& __from, const path& __to, copy_options __opt,
   1391                  error_code* __ec = nullptr);
   1392 _LIBCPP_FUNC_VIS
   1393 void __copy_symlink(const path& __existing_symlink, const path& __new_symlink,
   1394                     error_code* __ec = nullptr);
   1395 _LIBCPP_FUNC_VIS
   1396 bool __create_directories(const path& p, error_code* ec = nullptr);
   1397 _LIBCPP_FUNC_VIS
   1398 bool __create_directory(const path& p, error_code* ec = nullptr);
   1399 _LIBCPP_FUNC_VIS
   1400 bool __create_directory(const path& p, const path& attributes,
   1401                         error_code* ec = nullptr);
   1402 _LIBCPP_FUNC_VIS
   1403 void __create_directory_symlink(const path& __to, const path& __new_symlink,
   1404                                 error_code* __ec = nullptr);
   1405 _LIBCPP_FUNC_VIS
   1406 void __create_hard_link(const path& __to, const path& __new_hard_link,
   1407                         error_code* __ec = nullptr);
   1408 _LIBCPP_FUNC_VIS
   1409 void __create_symlink(const path& __to, const path& __new_symlink,
   1410                       error_code* __ec = nullptr);
   1411 _LIBCPP_FUNC_VIS
   1412 path __current_path(error_code* __ec = nullptr);
   1413 _LIBCPP_FUNC_VIS
   1414 void __current_path(const path&, error_code* __ec = nullptr);
   1415 _LIBCPP_FUNC_VIS
   1416 bool __equivalent(const path&, const path&, error_code* __ec = nullptr);
   1417 _LIBCPP_FUNC_VIS
   1418 uintmax_t __file_size(const path&, error_code* __ec = nullptr);
   1419 _LIBCPP_FUNC_VIS
   1420 uintmax_t __hard_link_count(const path&, error_code* __ec = nullptr);
   1421 _LIBCPP_FUNC_VIS
   1422 bool __fs_is_empty(const path& p, error_code* ec = nullptr);
   1423 _LIBCPP_FUNC_VIS
   1424 file_time_type __last_write_time(const path& p, error_code* ec = nullptr);
   1425 _LIBCPP_FUNC_VIS
   1426 void __last_write_time(const path& p, file_time_type new_time,
   1427                        error_code* ec = nullptr);
   1428 _LIBCPP_FUNC_VIS
   1429 void __permissions(const path&, perms, perm_options, error_code* = nullptr);
   1430 _LIBCPP_FUNC_VIS
   1431 path __read_symlink(const path& p, error_code* ec = nullptr);
   1432 _LIBCPP_FUNC_VIS
   1433 bool __remove(const path& p, error_code* ec = nullptr);
   1434 _LIBCPP_FUNC_VIS
   1435 uintmax_t __remove_all(const path& p, error_code* ec = nullptr);
   1436 _LIBCPP_FUNC_VIS
   1437 void __rename(const path& from, const path& to, error_code* ec = nullptr);
   1438 _LIBCPP_FUNC_VIS
   1439 void __resize_file(const path& p, uintmax_t size, error_code* ec = nullptr);
   1440 _LIBCPP_FUNC_VIS
   1441 space_info __space(const path&, error_code* __ec = nullptr);
   1442 _LIBCPP_FUNC_VIS
   1443 file_status __status(const path&, error_code* __ec = nullptr);
   1444 _LIBCPP_FUNC_VIS
   1445 file_status __symlink_status(const path&, error_code* __ec = nullptr);
   1446 _LIBCPP_FUNC_VIS
   1447 path __system_complete(const path&, error_code* __ec = nullptr);
   1448 _LIBCPP_FUNC_VIS
   1449 path __temp_directory_path(error_code* __ec = nullptr);
   1450 _LIBCPP_FUNC_VIS
   1451 path __weakly_canonical(path const& __p, error_code* __ec = nullptr);
   1452 
   1453 inline _LIBCPP_INLINE_VISIBILITY path current_path() {
   1454   return __current_path();
   1455 }
   1456 
   1457 inline _LIBCPP_INLINE_VISIBILITY path current_path(error_code& __ec) {
   1458   return __current_path(&__ec);
   1459 }
   1460 
   1461 inline _LIBCPP_INLINE_VISIBILITY void current_path(const path& __p) {
   1462   __current_path(__p);
   1463 }
   1464 
   1465 inline _LIBCPP_INLINE_VISIBILITY void current_path(const path& __p,
   1466                                                    error_code& __ec) noexcept {
   1467   __current_path(__p, &__ec);
   1468 }
   1469 
   1470 inline _LIBCPP_INLINE_VISIBILITY path absolute(const path& __p) {
   1471   return __absolute(__p);
   1472 }
   1473 
   1474 inline _LIBCPP_INLINE_VISIBILITY path absolute(const path& __p,
   1475                                                error_code& __ec) {
   1476   return __absolute(__p, &__ec);
   1477 }
   1478 
   1479 inline _LIBCPP_INLINE_VISIBILITY path canonical(const path& __p) {
   1480   return __canonical(__p);
   1481 }
   1482 
   1483 inline _LIBCPP_INLINE_VISIBILITY path canonical(const path& __p,
   1484                                                 error_code& __ec) {
   1485   return __canonical(__p, &__ec);
   1486 }
   1487 
   1488 inline _LIBCPP_INLINE_VISIBILITY void copy(const path& __from,
   1489                                            const path& __to) {
   1490   __copy(__from, __to, copy_options::none);
   1491 }
   1492 
   1493 inline _LIBCPP_INLINE_VISIBILITY void copy(const path& __from, const path& __to,
   1494                                            error_code& __ec) {
   1495   __copy(__from, __to, copy_options::none, &__ec);
   1496 }
   1497 
   1498 inline _LIBCPP_INLINE_VISIBILITY void copy(const path& __from, const path& __to,
   1499                                            copy_options __opt) {
   1500   __copy(__from, __to, __opt);
   1501 }
   1502 
   1503 inline _LIBCPP_INLINE_VISIBILITY void copy(const path& __from, const path& __to,
   1504                                            copy_options __opt,
   1505                                            error_code& __ec) {
   1506   __copy(__from, __to, __opt, &__ec);
   1507 }
   1508 
   1509 inline _LIBCPP_INLINE_VISIBILITY bool copy_file(const path& __from,
   1510                                                 const path& __to) {
   1511   return __copy_file(__from, __to, copy_options::none);
   1512 }
   1513 
   1514 inline _LIBCPP_INLINE_VISIBILITY bool
   1515 copy_file(const path& __from, const path& __to, error_code& __ec) {
   1516   return __copy_file(__from, __to, copy_options::none, &__ec);
   1517 }
   1518 
   1519 inline _LIBCPP_INLINE_VISIBILITY bool
   1520 copy_file(const path& __from, const path& __to, copy_options __opt) {
   1521   return __copy_file(__from, __to, __opt);
   1522 }
   1523 
   1524 inline _LIBCPP_INLINE_VISIBILITY bool copy_file(const path& __from,
   1525                                                 const path& __to,
   1526                                                 copy_options __opt,
   1527                                                 error_code& __ec) {
   1528   return __copy_file(__from, __to, __opt, &__ec);
   1529 }
   1530 
   1531 inline _LIBCPP_INLINE_VISIBILITY void copy_symlink(const path& __existing,
   1532                                                    const path& __new) {
   1533   __copy_symlink(__existing, __new);
   1534 }
   1535 
   1536 inline _LIBCPP_INLINE_VISIBILITY void
   1537 copy_symlink(const path& __ext, const path& __new, error_code& __ec) noexcept {
   1538   __copy_symlink(__ext, __new, &__ec);
   1539 }
   1540 
   1541 inline _LIBCPP_INLINE_VISIBILITY bool create_directories(const path& __p) {
   1542   return __create_directories(__p);
   1543 }
   1544 
   1545 inline _LIBCPP_INLINE_VISIBILITY bool create_directories(const path& __p,
   1546                                                          error_code& __ec) {
   1547   return __create_directories(__p, &__ec);
   1548 }
   1549 
   1550 inline _LIBCPP_INLINE_VISIBILITY bool create_directory(const path& __p) {
   1551   return __create_directory(__p);
   1552 }
   1553 
   1554 inline _LIBCPP_INLINE_VISIBILITY bool
   1555 create_directory(const path& __p, error_code& __ec) noexcept {
   1556   return __create_directory(__p, &__ec);
   1557 }
   1558 
   1559 inline _LIBCPP_INLINE_VISIBILITY bool create_directory(const path& __p,
   1560                                                        const path& __attrs) {
   1561   return __create_directory(__p, __attrs);
   1562 }
   1563 
   1564 inline _LIBCPP_INLINE_VISIBILITY bool
   1565 create_directory(const path& __p, const path& __attrs,
   1566                  error_code& __ec) noexcept {
   1567   return __create_directory(__p, __attrs, &__ec);
   1568 }
   1569 
   1570 inline _LIBCPP_INLINE_VISIBILITY void
   1571 create_directory_symlink(const path& __to, const path& __new) {
   1572   __create_directory_symlink(__to, __new);
   1573 }
   1574 
   1575 inline _LIBCPP_INLINE_VISIBILITY void
   1576 create_directory_symlink(const path& __to, const path& __new,
   1577                          error_code& __ec) noexcept {
   1578   __create_directory_symlink(__to, __new, &__ec);
   1579 }
   1580 
   1581 inline _LIBCPP_INLINE_VISIBILITY void create_hard_link(const path& __to,
   1582                                                        const path& __new) {
   1583   __create_hard_link(__to, __new);
   1584 }
   1585 
   1586 inline _LIBCPP_INLINE_VISIBILITY void
   1587 create_hard_link(const path& __to, const path& __new,
   1588                  error_code& __ec) noexcept {
   1589   __create_hard_link(__to, __new, &__ec);
   1590 }
   1591 
   1592 inline _LIBCPP_INLINE_VISIBILITY void create_symlink(const path& __to,
   1593                                                      const path& __new) {
   1594   __create_symlink(__to, __new);
   1595 }
   1596 
   1597 inline _LIBCPP_INLINE_VISIBILITY void
   1598 create_symlink(const path& __to, const path& __new, error_code& __ec) noexcept {
   1599   return __create_symlink(__to, __new, &__ec);
   1600 }
   1601 
   1602 inline _LIBCPP_INLINE_VISIBILITY bool status_known(file_status __s) noexcept {
   1603   return __s.type() != file_type::none;
   1604 }
   1605 
   1606 inline _LIBCPP_INLINE_VISIBILITY bool exists(file_status __s) noexcept {
   1607   return status_known(__s) && __s.type() != file_type::not_found;
   1608 }
   1609 
   1610 inline _LIBCPP_INLINE_VISIBILITY bool exists(const path& __p) {
   1611   return exists(__status(__p));
   1612 }
   1613 
   1614 inline _LIBCPP_INLINE_VISIBILITY bool exists(const path& __p,
   1615                                              error_code& __ec) noexcept {
   1616   auto __s = __status(__p, &__ec);
   1617   if (status_known(__s))
   1618     __ec.clear();
   1619   return exists(__s);
   1620 }
   1621 
   1622 inline _LIBCPP_INLINE_VISIBILITY bool equivalent(const path& __p1,
   1623                                                  const path& __p2) {
   1624   return __equivalent(__p1, __p2);
   1625 }
   1626 
   1627 inline _LIBCPP_INLINE_VISIBILITY bool
   1628 equivalent(const path& __p1, const path& __p2, error_code& __ec) noexcept {
   1629   return __equivalent(__p1, __p2, &__ec);
   1630 }
   1631 
   1632 inline _LIBCPP_INLINE_VISIBILITY uintmax_t file_size(const path& __p) {
   1633   return __file_size(__p);
   1634 }
   1635 
   1636 inline _LIBCPP_INLINE_VISIBILITY uintmax_t
   1637 file_size(const path& __p, error_code& __ec) noexcept {
   1638   return __file_size(__p, &__ec);
   1639 }
   1640 
   1641 inline _LIBCPP_INLINE_VISIBILITY uintmax_t hard_link_count(const path& __p) {
   1642   return __hard_link_count(__p);
   1643 }
   1644 
   1645 inline _LIBCPP_INLINE_VISIBILITY uintmax_t
   1646 hard_link_count(const path& __p, error_code& __ec) noexcept {
   1647   return __hard_link_count(__p, &__ec);
   1648 }
   1649 
   1650 inline _LIBCPP_INLINE_VISIBILITY bool is_block_file(file_status __s) noexcept {
   1651   return __s.type() == file_type::block;
   1652 }
   1653 
   1654 inline _LIBCPP_INLINE_VISIBILITY bool is_block_file(const path& __p) {
   1655   return is_block_file(__status(__p));
   1656 }
   1657 
   1658 inline _LIBCPP_INLINE_VISIBILITY bool is_block_file(const path& __p,
   1659                                                     error_code& __ec) noexcept {
   1660   return is_block_file(__status(__p, &__ec));
   1661 }
   1662 
   1663 inline _LIBCPP_INLINE_VISIBILITY bool
   1664 is_character_file(file_status __s) noexcept {
   1665   return __s.type() == file_type::character;
   1666 }
   1667 
   1668 inline _LIBCPP_INLINE_VISIBILITY bool is_character_file(const path& __p) {
   1669   return is_character_file(__status(__p));
   1670 }
   1671 
   1672 inline _LIBCPP_INLINE_VISIBILITY bool
   1673 is_character_file(const path& __p, error_code& __ec) noexcept {
   1674   return is_character_file(__status(__p, &__ec));
   1675 }
   1676 
   1677 inline _LIBCPP_INLINE_VISIBILITY bool is_directory(file_status __s) noexcept {
   1678   return __s.type() == file_type::directory;
   1679 }
   1680 
   1681 inline _LIBCPP_INLINE_VISIBILITY bool is_directory(const path& __p) {
   1682   return is_directory(__status(__p));
   1683 }
   1684 
   1685 inline _LIBCPP_INLINE_VISIBILITY bool is_directory(const path& __p,
   1686                                                    error_code& __ec) noexcept {
   1687   return is_directory(__status(__p, &__ec));
   1688 }
   1689 
   1690 inline _LIBCPP_INLINE_VISIBILITY bool is_empty(const path& __p) {
   1691   return __fs_is_empty(__p);
   1692 }
   1693 
   1694 inline _LIBCPP_INLINE_VISIBILITY bool is_empty(const path& __p,
   1695                                                error_code& __ec) {
   1696   return __fs_is_empty(__p, &__ec);
   1697 }
   1698 
   1699 inline _LIBCPP_INLINE_VISIBILITY bool is_fifo(file_status __s) noexcept {
   1700   return __s.type() == file_type::fifo;
   1701 }
   1702 inline _LIBCPP_INLINE_VISIBILITY bool is_fifo(const path& __p) {
   1703   return is_fifo(__status(__p));
   1704 }
   1705 
   1706 inline _LIBCPP_INLINE_VISIBILITY bool is_fifo(const path& __p,
   1707                                               error_code& __ec) noexcept {
   1708   return is_fifo(__status(__p, &__ec));
   1709 }
   1710 
   1711 inline _LIBCPP_INLINE_VISIBILITY bool
   1712 is_regular_file(file_status __s) noexcept {
   1713   return __s.type() == file_type::regular;
   1714 }
   1715 
   1716 inline _LIBCPP_INLINE_VISIBILITY bool is_regular_file(const path& __p) {
   1717   return is_regular_file(__status(__p));
   1718 }
   1719 
   1720 inline _LIBCPP_INLINE_VISIBILITY bool
   1721 is_regular_file(const path& __p, error_code& __ec) noexcept {
   1722   return is_regular_file(__status(__p, &__ec));
   1723 }
   1724 
   1725 inline _LIBCPP_INLINE_VISIBILITY bool is_socket(file_status __s) noexcept {
   1726   return __s.type() == file_type::socket;
   1727 }
   1728 
   1729 inline _LIBCPP_INLINE_VISIBILITY bool is_socket(const path& __p) {
   1730   return is_socket(__status(__p));
   1731 }
   1732 
   1733 inline _LIBCPP_INLINE_VISIBILITY bool is_socket(const path& __p,
   1734                                                 error_code& __ec) noexcept {
   1735   return is_socket(__status(__p, &__ec));
   1736 }
   1737 
   1738 inline _LIBCPP_INLINE_VISIBILITY bool is_symlink(file_status __s) noexcept {
   1739   return __s.type() == file_type::symlink;
   1740 }
   1741 
   1742 inline _LIBCPP_INLINE_VISIBILITY bool is_symlink(const path& __p) {
   1743   return is_symlink(__symlink_status(__p));
   1744 }
   1745 
   1746 inline _LIBCPP_INLINE_VISIBILITY bool is_symlink(const path& __p,
   1747                                                  error_code& __ec) noexcept {
   1748   return is_symlink(__symlink_status(__p, &__ec));
   1749 }
   1750 
   1751 inline _LIBCPP_INLINE_VISIBILITY bool is_other(file_status __s) noexcept {
   1752   return exists(__s) && !is_regular_file(__s) && !is_directory(__s) &&
   1753          !is_symlink(__s);
   1754 }
   1755 
   1756 inline _LIBCPP_INLINE_VISIBILITY bool is_other(const path& __p) {
   1757   return is_other(__status(__p));
   1758 }
   1759 
   1760 inline _LIBCPP_INLINE_VISIBILITY bool is_other(const path& __p,
   1761                                                error_code& __ec) noexcept {
   1762   return is_other(__status(__p, &__ec));
   1763 }
   1764 
   1765 inline _LIBCPP_INLINE_VISIBILITY file_time_type
   1766 last_write_time(const path& __p) {
   1767   return __last_write_time(__p);
   1768 }
   1769 
   1770 inline _LIBCPP_INLINE_VISIBILITY file_time_type
   1771 last_write_time(const path& __p, error_code& __ec) noexcept {
   1772   return __last_write_time(__p, &__ec);
   1773 }
   1774 
   1775 inline _LIBCPP_INLINE_VISIBILITY void last_write_time(const path& __p,
   1776                                                       file_time_type __t) {
   1777   __last_write_time(__p, __t);
   1778 }
   1779 
   1780 inline _LIBCPP_INLINE_VISIBILITY void
   1781 last_write_time(const path& __p, file_time_type __t,
   1782                 error_code& __ec) noexcept {
   1783   __last_write_time(__p, __t, &__ec);
   1784 }
   1785 
   1786 inline _LIBCPP_INLINE_VISIBILITY void
   1787 permissions(const path& __p, perms __prms,
   1788             perm_options __opts = perm_options::replace) {
   1789   __permissions(__p, __prms, __opts);
   1790 }
   1791 
   1792 inline _LIBCPP_INLINE_VISIBILITY void permissions(const path& __p, perms __prms,
   1793                                                   error_code& __ec) noexcept {
   1794   __permissions(__p, __prms, perm_options::replace, &__ec);
   1795 }
   1796 
   1797 inline _LIBCPP_INLINE_VISIBILITY void permissions(const path& __p, perms __prms,
   1798                                                   perm_options __opts,
   1799                                                   error_code& __ec) {
   1800   __permissions(__p, __prms, __opts, &__ec);
   1801 }
   1802 
   1803 inline _LIBCPP_INLINE_VISIBILITY path proximate(const path& __p,
   1804                                                 const path& __base,
   1805                                                 error_code& __ec) {
   1806   path __tmp = __weakly_canonical(__p, &__ec);
   1807   if (__ec)
   1808     return {};
   1809   path __tmp_base = __weakly_canonical(__base, &__ec);
   1810   if (__ec)
   1811     return {};
   1812   return __tmp.lexically_proximate(__tmp_base);
   1813 }
   1814 
   1815 inline _LIBCPP_INLINE_VISIBILITY path proximate(const path& __p,
   1816                                                 error_code& __ec) {
   1817   return proximate(__p, current_path(), __ec);
   1818 }
   1819 
   1820 inline _LIBCPP_INLINE_VISIBILITY path
   1821 proximate(const path& __p, const path& __base = current_path()) {
   1822   return __weakly_canonical(__p).lexically_proximate(
   1823       __weakly_canonical(__base));
   1824 }
   1825 
   1826 inline _LIBCPP_INLINE_VISIBILITY path read_symlink(const path& __p) {
   1827   return __read_symlink(__p);
   1828 }
   1829 
   1830 inline _LIBCPP_INLINE_VISIBILITY path read_symlink(const path& __p,
   1831                                                    error_code& __ec) {
   1832   return __read_symlink(__p, &__ec);
   1833 }
   1834 
   1835 inline _LIBCPP_INLINE_VISIBILITY path relative(const path& __p,
   1836                                                const path& __base,
   1837                                                error_code& __ec) {
   1838   path __tmp = __weakly_canonical(__p, &__ec);
   1839   if (__ec)
   1840     return path();
   1841   path __tmpbase = __weakly_canonical(__base, &__ec);
   1842   if (__ec)
   1843     return path();
   1844   return __tmp.lexically_relative(__tmpbase);
   1845 }
   1846 
   1847 inline _LIBCPP_INLINE_VISIBILITY path relative(const path& __p,
   1848                                                error_code& __ec) {
   1849   return relative(__p, current_path(), __ec);
   1850 }
   1851 
   1852 inline _LIBCPP_INLINE_VISIBILITY path
   1853 relative(const path& __p, const path& __base = current_path()) {
   1854   return __weakly_canonical(__p).lexically_relative(__weakly_canonical(__base));
   1855 }
   1856 
   1857 inline _LIBCPP_INLINE_VISIBILITY bool remove(const path& __p) {
   1858   return __remove(__p);
   1859 }
   1860 
   1861 inline _LIBCPP_INLINE_VISIBILITY bool remove(const path& __p,
   1862                                              error_code& __ec) noexcept {
   1863   return __remove(__p, &__ec);
   1864 }
   1865 
   1866 inline _LIBCPP_INLINE_VISIBILITY uintmax_t remove_all(const path& __p) {
   1867   return __remove_all(__p);
   1868 }
   1869 
   1870 inline _LIBCPP_INLINE_VISIBILITY uintmax_t remove_all(const path& __p,
   1871                                                       error_code& __ec) {
   1872   return __remove_all(__p, &__ec);
   1873 }
   1874 
   1875 inline _LIBCPP_INLINE_VISIBILITY void rename(const path& __from,
   1876                                              const path& __to) {
   1877   return __rename(__from, __to);
   1878 }
   1879 
   1880 inline _LIBCPP_INLINE_VISIBILITY void
   1881 rename(const path& __from, const path& __to, error_code& __ec) noexcept {
   1882   return __rename(__from, __to, &__ec);
   1883 }
   1884 
   1885 inline _LIBCPP_INLINE_VISIBILITY void resize_file(const path& __p,
   1886                                                   uintmax_t __ns) {
   1887   return __resize_file(__p, __ns);
   1888 }
   1889 
   1890 inline _LIBCPP_INLINE_VISIBILITY void
   1891 resize_file(const path& __p, uintmax_t __ns, error_code& __ec) noexcept {
   1892   return __resize_file(__p, __ns, &__ec);
   1893 }
   1894 
   1895 inline _LIBCPP_INLINE_VISIBILITY space_info space(const path& __p) {
   1896   return __space(__p);
   1897 }
   1898 
   1899 inline _LIBCPP_INLINE_VISIBILITY space_info space(const path& __p,
   1900                                                   error_code& __ec) noexcept {
   1901   return __space(__p, &__ec);
   1902 }
   1903 
   1904 inline _LIBCPP_INLINE_VISIBILITY file_status status(const path& __p) {
   1905   return __status(__p);
   1906 }
   1907 
   1908 inline _LIBCPP_INLINE_VISIBILITY file_status status(const path& __p,
   1909                                                     error_code& __ec) noexcept {
   1910   return __status(__p, &__ec);
   1911 }
   1912 
   1913 inline _LIBCPP_INLINE_VISIBILITY file_status symlink_status(const path& __p) {
   1914   return __symlink_status(__p);
   1915 }
   1916 
   1917 inline _LIBCPP_INLINE_VISIBILITY file_status
   1918 symlink_status(const path& __p, error_code& __ec) noexcept {
   1919   return __symlink_status(__p, &__ec);
   1920 }
   1921 
   1922 inline _LIBCPP_INLINE_VISIBILITY path temp_directory_path() {
   1923   return __temp_directory_path();
   1924 }
   1925 
   1926 inline _LIBCPP_INLINE_VISIBILITY path temp_directory_path(error_code& __ec) {
   1927   return __temp_directory_path(&__ec);
   1928 }
   1929 
   1930 inline _LIBCPP_INLINE_VISIBILITY path weakly_canonical(path const& __p) {
   1931   return __weakly_canonical(__p);
   1932 }
   1933 
   1934 inline _LIBCPP_INLINE_VISIBILITY path weakly_canonical(path const& __p,
   1935                                                        error_code& __ec) {
   1936   return __weakly_canonical(__p, &__ec);
   1937 }
   1938 
   1939 class directory_iterator;
   1940 class recursive_directory_iterator;
   1941 class __dir_stream;
   1942 
   1943 class directory_entry {
   1944   typedef _VSTD_FS::path _Path;
   1945 
   1946 public:
   1947   // constructors and destructors
   1948   directory_entry() noexcept = default;
   1949   directory_entry(directory_entry const&) = default;
   1950   directory_entry(directory_entry&&) noexcept = default;
   1951 
   1952   _LIBCPP_INLINE_VISIBILITY
   1953   explicit directory_entry(_Path const& __p) : __p_(__p) {
   1954     error_code __ec;
   1955     __refresh(&__ec);
   1956   }
   1957 
   1958   _LIBCPP_INLINE_VISIBILITY
   1959   directory_entry(_Path const& __p, error_code& __ec) : __p_(__p) {
   1960     __refresh(&__ec);
   1961   }
   1962 
   1963   ~directory_entry() {}
   1964 
   1965   directory_entry& operator=(directory_entry const&) = default;
   1966   directory_entry& operator=(directory_entry&&) noexcept = default;
   1967 
   1968   _LIBCPP_INLINE_VISIBILITY
   1969   void assign(_Path const& __p) {
   1970     __p_ = __p;
   1971     error_code __ec;
   1972     __refresh(&__ec);
   1973   }
   1974 
   1975   _LIBCPP_INLINE_VISIBILITY
   1976   void assign(_Path const& __p, error_code& __ec) {
   1977     __p_ = __p;
   1978     __refresh(&__ec);
   1979   }
   1980 
   1981   _LIBCPP_INLINE_VISIBILITY
   1982   void replace_filename(_Path const& __p) {
   1983     __p_.replace_filename(__p);
   1984     error_code __ec;
   1985     __refresh(&__ec);
   1986   }
   1987 
   1988   _LIBCPP_INLINE_VISIBILITY
   1989   void replace_filename(_Path const& __p, error_code& __ec) {
   1990     __p_ = __p_.parent_path() / __p;
   1991     __refresh(&__ec);
   1992   }
   1993 
   1994   _LIBCPP_INLINE_VISIBILITY
   1995   void refresh() { __refresh(); }
   1996 
   1997   _LIBCPP_INLINE_VISIBILITY
   1998   void refresh(error_code& __ec) noexcept { __refresh(&__ec); }
   1999 
   2000   _LIBCPP_INLINE_VISIBILITY
   2001   _Path const& path() const noexcept { return __p_; }
   2002 
   2003   _LIBCPP_INLINE_VISIBILITY
   2004   operator const _Path&() const noexcept { return __p_; }
   2005 
   2006   _LIBCPP_INLINE_VISIBILITY
   2007   bool exists() const { return _VSTD_FS::exists(file_status{__get_ft()}); }
   2008 
   2009   _LIBCPP_INLINE_VISIBILITY
   2010   bool exists(error_code& __ec) const noexcept {
   2011     return _VSTD_FS::exists(file_status{__get_ft(&__ec)});
   2012   }
   2013 
   2014   _LIBCPP_INLINE_VISIBILITY
   2015   bool is_block_file() const { return __get_ft() == file_type::block; }
   2016 
   2017   _LIBCPP_INLINE_VISIBILITY
   2018   bool is_block_file(error_code& __ec) const noexcept {
   2019     return __get_ft(&__ec) == file_type::block;
   2020   }
   2021 
   2022   _LIBCPP_INLINE_VISIBILITY
   2023   bool is_character_file() const { return __get_ft() == file_type::character; }
   2024 
   2025   _LIBCPP_INLINE_VISIBILITY
   2026   bool is_character_file(error_code& __ec) const noexcept {
   2027     return __get_ft(&__ec) == file_type::character;
   2028   }
   2029 
   2030   _LIBCPP_INLINE_VISIBILITY
   2031   bool is_directory() const { return __get_ft() == file_type::directory; }
   2032 
   2033   _LIBCPP_INLINE_VISIBILITY
   2034   bool is_directory(error_code& __ec) const noexcept {
   2035     return __get_ft(&__ec) == file_type::directory;
   2036   }
   2037 
   2038   _LIBCPP_INLINE_VISIBILITY
   2039   bool is_fifo() const { return __get_ft() == file_type::fifo; }
   2040 
   2041   _LIBCPP_INLINE_VISIBILITY
   2042   bool is_fifo(error_code& __ec) const noexcept {
   2043     return __get_ft(&__ec) == file_type::fifo;
   2044   }
   2045 
   2046   _LIBCPP_INLINE_VISIBILITY
   2047   bool is_other() const { return _VSTD_FS::is_other(file_status{__get_ft()}); }
   2048 
   2049   _LIBCPP_INLINE_VISIBILITY
   2050   bool is_other(error_code& __ec) const noexcept {
   2051     return _VSTD_FS::is_other(file_status{__get_ft(&__ec)});
   2052   }
   2053 
   2054   _LIBCPP_INLINE_VISIBILITY
   2055   bool is_regular_file() const { return __get_ft() == file_type::regular; }
   2056 
   2057   _LIBCPP_INLINE_VISIBILITY
   2058   bool is_regular_file(error_code& __ec) const noexcept {
   2059     return __get_ft(&__ec) == file_type::regular;
   2060   }
   2061 
   2062   _LIBCPP_INLINE_VISIBILITY
   2063   bool is_socket() const { return __get_ft() == file_type::socket; }
   2064 
   2065   _LIBCPP_INLINE_VISIBILITY
   2066   bool is_socket(error_code& __ec) const noexcept {
   2067     return __get_ft(&__ec) == file_type::socket;
   2068   }
   2069 
   2070   _LIBCPP_INLINE_VISIBILITY
   2071   bool is_symlink() const { return __get_sym_ft() == file_type::symlink; }
   2072 
   2073   _LIBCPP_INLINE_VISIBILITY
   2074   bool is_symlink(error_code& __ec) const noexcept {
   2075     return __get_sym_ft(&__ec) == file_type::symlink;
   2076   }
   2077   _LIBCPP_INLINE_VISIBILITY
   2078   uintmax_t file_size() const { return __get_size(); }
   2079 
   2080   _LIBCPP_INLINE_VISIBILITY
   2081   uintmax_t file_size(error_code& __ec) const noexcept {
   2082     return __get_size(&__ec);
   2083   }
   2084 
   2085   _LIBCPP_INLINE_VISIBILITY
   2086   uintmax_t hard_link_count() const { return __get_nlink(); }
   2087 
   2088   _LIBCPP_INLINE_VISIBILITY
   2089   uintmax_t hard_link_count(error_code& __ec) const noexcept {
   2090     return __get_nlink(&__ec);
   2091   }
   2092 
   2093   _LIBCPP_INLINE_VISIBILITY
   2094   file_time_type last_write_time() const { return __get_write_time(); }
   2095 
   2096   _LIBCPP_INLINE_VISIBILITY
   2097   file_time_type last_write_time(error_code& __ec) const noexcept {
   2098     return __get_write_time(&__ec);
   2099   }
   2100 
   2101   _LIBCPP_INLINE_VISIBILITY
   2102   file_status status() const { return __get_status(); }
   2103 
   2104   _LIBCPP_INLINE_VISIBILITY
   2105   file_status status(error_code& __ec) const noexcept {
   2106     return __get_status(&__ec);
   2107   }
   2108 
   2109   _LIBCPP_INLINE_VISIBILITY
   2110   file_status symlink_status() const { return __get_symlink_status(); }
   2111 
   2112   _LIBCPP_INLINE_VISIBILITY
   2113   file_status symlink_status(error_code& __ec) const noexcept {
   2114     return __get_symlink_status(&__ec);
   2115   }
   2116 
   2117   _LIBCPP_INLINE_VISIBILITY
   2118   bool operator<(directory_entry const& __rhs) const noexcept {
   2119     return __p_ < __rhs.__p_;
   2120   }
   2121 
   2122   _LIBCPP_INLINE_VISIBILITY
   2123   bool operator==(directory_entry const& __rhs) const noexcept {
   2124     return __p_ == __rhs.__p_;
   2125   }
   2126 
   2127   _LIBCPP_INLINE_VISIBILITY
   2128   bool operator!=(directory_entry const& __rhs) const noexcept {
   2129     return __p_ != __rhs.__p_;
   2130   }
   2131 
   2132   _LIBCPP_INLINE_VISIBILITY
   2133   bool operator<=(directory_entry const& __rhs) const noexcept {
   2134     return __p_ <= __rhs.__p_;
   2135   }
   2136 
   2137   _LIBCPP_INLINE_VISIBILITY
   2138   bool operator>(directory_entry const& __rhs) const noexcept {
   2139     return __p_ > __rhs.__p_;
   2140   }
   2141 
   2142   _LIBCPP_INLINE_VISIBILITY
   2143   bool operator>=(directory_entry const& __rhs) const noexcept {
   2144     return __p_ >= __rhs.__p_;
   2145   }
   2146 
   2147 private:
   2148   friend class directory_iterator;
   2149   friend class recursive_directory_iterator;
   2150   friend class __dir_stream;
   2151 
   2152   enum _CacheType : unsigned char {
   2153     _Empty,
   2154     _IterSymlink,
   2155     _IterNonSymlink,
   2156     _RefreshSymlink,
   2157     _RefreshSymlinkUnresolved,
   2158     _RefreshNonSymlink
   2159   };
   2160 
   2161   struct __cached_data {
   2162     uintmax_t __size_;
   2163     uintmax_t __nlink_;
   2164     file_time_type __write_time_;
   2165     perms __sym_perms_;
   2166     perms __non_sym_perms_;
   2167     file_type __type_;
   2168     _CacheType __cache_type_;
   2169 
   2170     _LIBCPP_INLINE_VISIBILITY
   2171     __cached_data() noexcept { __reset(); }
   2172 
   2173     _LIBCPP_INLINE_VISIBILITY
   2174     void __reset() {
   2175       __cache_type_ = _Empty;
   2176       __type_ = file_type::none;
   2177       __sym_perms_ = __non_sym_perms_ = perms::unknown;
   2178       __size_ = __nlink_ = uintmax_t(-1);
   2179       __write_time_ = file_time_type::min();
   2180     }
   2181   };
   2182 
   2183   _LIBCPP_INLINE_VISIBILITY
   2184   static __cached_data __create_iter_result(file_type __ft) {
   2185     __cached_data __data;
   2186     __data.__type_ = __ft;
   2187     __data.__cache_type_ = [&]() {
   2188       switch (__ft) {
   2189       case file_type::none:
   2190         return _Empty;
   2191       case file_type::symlink:
   2192         return _IterSymlink;
   2193       default:
   2194         return _IterNonSymlink;
   2195       }
   2196     }();
   2197     return __data;
   2198   }
   2199 
   2200   _LIBCPP_INLINE_VISIBILITY
   2201   void __assign_iter_entry(_Path&& __p, __cached_data __dt) {
   2202     __p_ = std::move(__p);
   2203     __data_ = __dt;
   2204   }
   2205 
   2206   _LIBCPP_FUNC_VIS
   2207   error_code __do_refresh() noexcept;
   2208 
   2209   _LIBCPP_INLINE_VISIBILITY
   2210   static bool __is_dne_error(error_code const& __ec) {
   2211     if (!__ec)
   2212       return true;
   2213     switch (static_cast<errc>(__ec.value())) {
   2214     case errc::no_such_file_or_directory:
   2215     case errc::not_a_directory:
   2216       return true;
   2217     default:
   2218       return false;
   2219     }
   2220   }
   2221 
   2222   _LIBCPP_INLINE_VISIBILITY
   2223   void __handle_error(const char* __msg, error_code* __dest_ec,
   2224                       error_code const& __ec, bool __allow_dne = false) const {
   2225     if (__dest_ec) {
   2226       *__dest_ec = __ec;
   2227       return;
   2228     }
   2229     if (__ec && (!__allow_dne || !__is_dne_error(__ec)))
   2230       __throw_filesystem_error(__msg, __p_, __ec);
   2231   }
   2232 
   2233   _LIBCPP_INLINE_VISIBILITY
   2234   void __refresh(error_code* __ec = nullptr) {
   2235     __handle_error("in directory_entry::refresh", __ec, __do_refresh(),
   2236                    /*allow_dne*/ true);
   2237   }
   2238 
   2239   _LIBCPP_INLINE_VISIBILITY
   2240   file_type __get_sym_ft(error_code* __ec = nullptr) const {
   2241     switch (__data_.__cache_type_) {
   2242     case _Empty:
   2243       return __symlink_status(__p_, __ec).type();
   2244     case _IterSymlink:
   2245     case _RefreshSymlink:
   2246     case _RefreshSymlinkUnresolved:
   2247       if (__ec)
   2248         __ec->clear();
   2249       return file_type::symlink;
   2250     case _IterNonSymlink:
   2251     case _RefreshNonSymlink:
   2252       file_status __st(__data_.__type_);
   2253       if (__ec && !_VSTD_FS::exists(__st))
   2254         *__ec = make_error_code(errc::no_such_file_or_directory);
   2255       else if (__ec)
   2256         __ec->clear();
   2257       return __data_.__type_;
   2258     }
   2259     _LIBCPP_UNREACHABLE();
   2260   }
   2261 
   2262   _LIBCPP_INLINE_VISIBILITY
   2263   file_type __get_ft(error_code* __ec = nullptr) const {
   2264     switch (__data_.__cache_type_) {
   2265     case _Empty:
   2266     case _IterSymlink:
   2267     case _RefreshSymlinkUnresolved:
   2268       return __status(__p_, __ec).type();
   2269     case _IterNonSymlink:
   2270     case _RefreshNonSymlink:
   2271     case _RefreshSymlink: {
   2272       file_status __st(__data_.__type_);
   2273       if (__ec && !_VSTD_FS::exists(__st))
   2274         *__ec = make_error_code(errc::no_such_file_or_directory);
   2275       else if (__ec)
   2276         __ec->clear();
   2277       return __data_.__type_;
   2278     }
   2279     }
   2280     _LIBCPP_UNREACHABLE();
   2281   }
   2282 
   2283   _LIBCPP_INLINE_VISIBILITY
   2284   file_status __get_status(error_code* __ec = nullptr) const {
   2285     switch (__data_.__cache_type_) {
   2286     case _Empty:
   2287     case _IterNonSymlink:
   2288     case _IterSymlink:
   2289     case _RefreshSymlinkUnresolved:
   2290       return __status(__p_, __ec);
   2291     case _RefreshNonSymlink:
   2292     case _RefreshSymlink:
   2293       return file_status(__get_ft(__ec), __data_.__non_sym_perms_);
   2294     }
   2295     _LIBCPP_UNREACHABLE();
   2296   }
   2297 
   2298   _LIBCPP_INLINE_VISIBILITY
   2299   file_status __get_symlink_status(error_code* __ec = nullptr) const {
   2300     switch (__data_.__cache_type_) {
   2301     case _Empty:
   2302     case _IterNonSymlink:
   2303     case _IterSymlink:
   2304       return __symlink_status(__p_, __ec);
   2305     case _RefreshNonSymlink:
   2306       return file_status(__get_sym_ft(__ec), __data_.__non_sym_perms_);
   2307     case _RefreshSymlink:
   2308     case _RefreshSymlinkUnresolved:
   2309       return file_status(__get_sym_ft(__ec), __data_.__sym_perms_);
   2310     }
   2311     _LIBCPP_UNREACHABLE();
   2312   }
   2313 
   2314   _LIBCPP_INLINE_VISIBILITY
   2315   uintmax_t __get_size(error_code* __ec = nullptr) const {
   2316     switch (__data_.__cache_type_) {
   2317     case _Empty:
   2318     case _IterNonSymlink:
   2319     case _IterSymlink:
   2320     case _RefreshSymlinkUnresolved:
   2321       return _VSTD_FS::__file_size(__p_, __ec);
   2322     case _RefreshSymlink:
   2323     case _RefreshNonSymlink: {
   2324       error_code __m_ec;
   2325       file_status __st(__get_ft(&__m_ec));
   2326       __handle_error("in directory_entry::file_size", __ec, __m_ec);
   2327       if (_VSTD_FS::exists(__st) && !_VSTD_FS::is_regular_file(__st)) {
   2328         errc __err_kind = _VSTD_FS::is_directory(__st) ? errc::is_a_directory
   2329                                                        : errc::not_supported;
   2330         __handle_error("in directory_entry::file_size", __ec,
   2331                        make_error_code(__err_kind));
   2332       }
   2333       return __data_.__size_;
   2334     }
   2335     }
   2336     _LIBCPP_UNREACHABLE();
   2337   }
   2338 
   2339   _LIBCPP_INLINE_VISIBILITY
   2340   uintmax_t __get_nlink(error_code* __ec = nullptr) const {
   2341     switch (__data_.__cache_type_) {
   2342     case _Empty:
   2343     case _IterNonSymlink:
   2344     case _IterSymlink:
   2345     case _RefreshSymlinkUnresolved:
   2346       return _VSTD_FS::__hard_link_count(__p_, __ec);
   2347     case _RefreshSymlink:
   2348     case _RefreshNonSymlink: {
   2349       error_code __m_ec;
   2350       (void)__get_ft(&__m_ec);
   2351       __handle_error("in directory_entry::hard_link_count", __ec, __m_ec);
   2352       return __data_.__nlink_;
   2353     }
   2354     }
   2355     _LIBCPP_UNREACHABLE();
   2356   }
   2357 
   2358   _LIBCPP_INLINE_VISIBILITY
   2359   file_time_type __get_write_time(error_code* __ec = nullptr) const {
   2360     switch (__data_.__cache_type_) {
   2361     case _Empty:
   2362     case _IterNonSymlink:
   2363     case _IterSymlink:
   2364     case _RefreshSymlinkUnresolved:
   2365       return _VSTD_FS::__last_write_time(__p_, __ec);
   2366     case _RefreshSymlink:
   2367     case _RefreshNonSymlink: {
   2368       error_code __m_ec;
   2369       file_status __st(__get_ft(&__m_ec));
   2370       __handle_error("in directory_entry::last_write_time", __ec, __m_ec);
   2371       if (_VSTD_FS::exists(__st) &&
   2372           __data_.__write_time_ == file_time_type::min())
   2373         __handle_error("in directory_entry::last_write_time", __ec,
   2374                        make_error_code(errc::value_too_large));
   2375       return __data_.__write_time_;
   2376     }
   2377     }
   2378     _LIBCPP_UNREACHABLE();
   2379   }
   2380 
   2381 private:
   2382   _Path __p_;
   2383   __cached_data __data_;
   2384 };
   2385 
   2386 class __dir_element_proxy {
   2387 public:
   2388   inline _LIBCPP_INLINE_VISIBILITY directory_entry operator*() {
   2389     return _VSTD::move(__elem_);
   2390   }
   2391 
   2392 private:
   2393   friend class directory_iterator;
   2394   friend class recursive_directory_iterator;
   2395   explicit __dir_element_proxy(directory_entry const& __e) : __elem_(__e) {}
   2396   __dir_element_proxy(__dir_element_proxy&& __o)
   2397       : __elem_(_VSTD::move(__o.__elem_)) {}
   2398   directory_entry __elem_;
   2399 };
   2400 
   2401 class directory_iterator {
   2402 public:
   2403   typedef directory_entry value_type;
   2404   typedef ptrdiff_t difference_type;
   2405   typedef value_type const* pointer;
   2406   typedef value_type const& reference;
   2407   typedef input_iterator_tag iterator_category;
   2408 
   2409 public:
   2410   //ctor & dtor
   2411   directory_iterator() noexcept {}
   2412 
   2413   explicit directory_iterator(const path& __p)
   2414       : directory_iterator(__p, nullptr) {}
   2415 
   2416   directory_iterator(const path& __p, directory_options __opts)
   2417       : directory_iterator(__p, nullptr, __opts) {}
   2418 
   2419   directory_iterator(const path& __p, error_code& __ec)
   2420       : directory_iterator(__p, &__ec) {}
   2421 
   2422   directory_iterator(const path& __p, directory_options __opts,
   2423                      error_code& __ec)
   2424       : directory_iterator(__p, &__ec, __opts) {}
   2425 
   2426   directory_iterator(const directory_iterator&) = default;
   2427   directory_iterator(directory_iterator&&) = default;
   2428   directory_iterator& operator=(const directory_iterator&) = default;
   2429 
   2430   directory_iterator& operator=(directory_iterator&& __o) noexcept {
   2431     // non-default implementation provided to support self-move assign.
   2432     if (this != &__o) {
   2433       __imp_ = _VSTD::move(__o.__imp_);
   2434     }
   2435     return *this;
   2436   }
   2437 
   2438   ~directory_iterator() = default;
   2439 
   2440   const directory_entry& operator*() const {
   2441     _LIBCPP_ASSERT(__imp_, "The end iterator cannot be dereferenced");
   2442     return __dereference();
   2443   }
   2444 
   2445   const directory_entry* operator->() const { return &**this; }
   2446 
   2447   directory_iterator& operator++() { return __increment(); }
   2448 
   2449   __dir_element_proxy operator++(int) {
   2450     __dir_element_proxy __p(**this);
   2451     __increment();
   2452     return __p;
   2453   }
   2454 
   2455   directory_iterator& increment(error_code& __ec) { return __increment(&__ec); }
   2456 
   2457 private:
   2458   inline _LIBCPP_INLINE_VISIBILITY friend bool
   2459   operator==(const directory_iterator& __lhs,
   2460              const directory_iterator& __rhs) noexcept;
   2461 
   2462   // construct the dir_stream
   2463   _LIBCPP_FUNC_VIS
   2464   directory_iterator(const path&, error_code*,
   2465                      directory_options = directory_options::none);
   2466 
   2467   _LIBCPP_FUNC_VIS
   2468   directory_iterator& __increment(error_code* __ec = nullptr);
   2469 
   2470   _LIBCPP_FUNC_VIS
   2471   const directory_entry& __dereference() const;
   2472 
   2473 private:
   2474   shared_ptr<__dir_stream> __imp_;
   2475 };
   2476 
   2477 inline _LIBCPP_INLINE_VISIBILITY bool
   2478 operator==(const directory_iterator& __lhs,
   2479            const directory_iterator& __rhs) noexcept {
   2480   return __lhs.__imp_ == __rhs.__imp_;
   2481 }
   2482 
   2483 inline _LIBCPP_INLINE_VISIBILITY bool
   2484 operator!=(const directory_iterator& __lhs,
   2485            const directory_iterator& __rhs) noexcept {
   2486   return !(__lhs == __rhs);
   2487 }
   2488 
   2489 // enable directory_iterator range-based for statements
   2490 inline _LIBCPP_INLINE_VISIBILITY directory_iterator
   2491 begin(directory_iterator __iter) noexcept {
   2492   return __iter;
   2493 }
   2494 
   2495 inline _LIBCPP_INLINE_VISIBILITY directory_iterator
   2496 end(const directory_iterator&) noexcept {
   2497   return directory_iterator();
   2498 }
   2499 
   2500 class recursive_directory_iterator {
   2501 public:
   2502   using value_type = directory_entry;
   2503   using difference_type = std::ptrdiff_t;
   2504   using pointer = directory_entry const*;
   2505   using reference = directory_entry const&;
   2506   using iterator_category = std::input_iterator_tag;
   2507 
   2508 public:
   2509   // constructors and destructor
   2510   _LIBCPP_INLINE_VISIBILITY
   2511   recursive_directory_iterator() noexcept : __rec_(false) {}
   2512 
   2513   _LIBCPP_INLINE_VISIBILITY
   2514   explicit recursive_directory_iterator(
   2515       const path& __p, directory_options __xoptions = directory_options::none)
   2516       : recursive_directory_iterator(__p, __xoptions, nullptr) {}
   2517 
   2518   _LIBCPP_INLINE_VISIBILITY
   2519   recursive_directory_iterator(const path& __p, directory_options __xoptions,
   2520                                error_code& __ec)
   2521       : recursive_directory_iterator(__p, __xoptions, &__ec) {}
   2522 
   2523   _LIBCPP_INLINE_VISIBILITY
   2524   recursive_directory_iterator(const path& __p, error_code& __ec)
   2525       : recursive_directory_iterator(__p, directory_options::none, &__ec) {}
   2526 
   2527   recursive_directory_iterator(const recursive_directory_iterator&) = default;
   2528   recursive_directory_iterator(recursive_directory_iterator&&) = default;
   2529 
   2530   recursive_directory_iterator&
   2531   operator=(const recursive_directory_iterator&) = default;
   2532 
   2533   _LIBCPP_INLINE_VISIBILITY
   2534   recursive_directory_iterator&
   2535   operator=(recursive_directory_iterator&& __o) noexcept {
   2536     // non-default implementation provided to support self-move assign.
   2537     if (this != &__o) {
   2538       __imp_ = _VSTD::move(__o.__imp_);
   2539       __rec_ = __o.__rec_;
   2540     }
   2541     return *this;
   2542   }
   2543 
   2544   ~recursive_directory_iterator() = default;
   2545 
   2546   _LIBCPP_INLINE_VISIBILITY
   2547   const directory_entry& operator*() const { return __dereference(); }
   2548 
   2549   _LIBCPP_INLINE_VISIBILITY
   2550   const directory_entry* operator->() const { return &__dereference(); }
   2551 
   2552   recursive_directory_iterator& operator++() { return __increment(); }
   2553 
   2554   _LIBCPP_INLINE_VISIBILITY
   2555   __dir_element_proxy operator++(int) {
   2556     __dir_element_proxy __p(**this);
   2557     __increment();
   2558     return __p;
   2559   }
   2560 
   2561   _LIBCPP_INLINE_VISIBILITY
   2562   recursive_directory_iterator& increment(error_code& __ec) {
   2563     return __increment(&__ec);
   2564   }
   2565 
   2566   _LIBCPP_FUNC_VIS directory_options options() const;
   2567   _LIBCPP_FUNC_VIS int depth() const;
   2568 
   2569   _LIBCPP_INLINE_VISIBILITY
   2570   void pop() { __pop(); }
   2571 
   2572   _LIBCPP_INLINE_VISIBILITY
   2573   void pop(error_code& __ec) { __pop(&__ec); }
   2574 
   2575   _LIBCPP_INLINE_VISIBILITY
   2576   bool recursion_pending() const { return __rec_; }
   2577 
   2578   _LIBCPP_INLINE_VISIBILITY
   2579   void disable_recursion_pending() { __rec_ = false; }
   2580 
   2581 private:
   2582   recursive_directory_iterator(const path& __p, directory_options __opt,
   2583                                error_code* __ec);
   2584 
   2585   _LIBCPP_FUNC_VIS
   2586   const directory_entry& __dereference() const;
   2587 
   2588   _LIBCPP_FUNC_VIS
   2589   bool __try_recursion(error_code* __ec);
   2590 
   2591   _LIBCPP_FUNC_VIS
   2592   void __advance(error_code* __ec = nullptr);
   2593 
   2594   _LIBCPP_FUNC_VIS
   2595   recursive_directory_iterator& __increment(error_code* __ec = nullptr);
   2596 
   2597   _LIBCPP_FUNC_VIS
   2598   void __pop(error_code* __ec = nullptr);
   2599 
   2600   inline _LIBCPP_INLINE_VISIBILITY friend bool
   2601   operator==(const recursive_directory_iterator&,
   2602              const recursive_directory_iterator&) noexcept;
   2603 
   2604   struct __shared_imp;
   2605   shared_ptr<__shared_imp> __imp_;
   2606   bool __rec_;
   2607 }; // class recursive_directory_iterator
   2608 
   2609 inline _LIBCPP_INLINE_VISIBILITY bool
   2610 operator==(const recursive_directory_iterator& __lhs,
   2611            const recursive_directory_iterator& __rhs) noexcept {
   2612   return __lhs.__imp_ == __rhs.__imp_;
   2613 }
   2614 
   2615 _LIBCPP_INLINE_VISIBILITY
   2616 inline bool operator!=(const recursive_directory_iterator& __lhs,
   2617                        const recursive_directory_iterator& __rhs) noexcept {
   2618   return !(__lhs == __rhs);
   2619 }
   2620 // enable recursive_directory_iterator range-based for statements
   2621 inline _LIBCPP_INLINE_VISIBILITY recursive_directory_iterator
   2622 begin(recursive_directory_iterator __iter) noexcept {
   2623   return __iter;
   2624 }
   2625 
   2626 inline _LIBCPP_INLINE_VISIBILITY recursive_directory_iterator
   2627 end(const recursive_directory_iterator&) noexcept {
   2628   return recursive_directory_iterator();
   2629 }
   2630 
   2631 _LIBCPP_END_NAMESPACE_FILESYSTEM
   2632 
   2633 #endif // !_LIBCPP_CXX03_LANG
   2634 
   2635 _LIBCPP_POP_MACROS
   2636 
   2637 #endif // _LIBCPP_FILESYSTEM
   2638