directory.hpp 29 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774
  1. // boost/filesystem/directory.hpp ---------------------------------------------------//
  2. // Copyright Beman Dawes 2002-2009
  3. // Copyright Jan Langer 2002
  4. // Copyright Dietmar Kuehl 2001
  5. // Copyright Vladimir Prus 2002
  6. // Copyright Andrey Semashev 2019, 2022
  7. // Distributed under the Boost Software License, Version 1.0.
  8. // See http://www.boost.org/LICENSE_1_0.txt
  9. // Library home page: http://www.boost.org/libs/filesystem
  10. //--------------------------------------------------------------------------------------//
  11. #ifndef BOOST_FILESYSTEM_DIRECTORY_HPP
  12. #define BOOST_FILESYSTEM_DIRECTORY_HPP
  13. #include <boost/filesystem/config.hpp>
  14. #include <boost/filesystem/path.hpp>
  15. #include <boost/filesystem/file_status.hpp>
  16. #include <boost/filesystem/detail/path_traits.hpp>
  17. #include <cstddef>
  18. #include <string>
  19. #include <vector>
  20. #include <boost/assert.hpp>
  21. #include <boost/core/scoped_enum.hpp>
  22. #include <boost/detail/bitmask.hpp>
  23. #include <boost/system/error_code.hpp>
  24. #include <boost/smart_ptr/intrusive_ptr.hpp>
  25. #include <boost/smart_ptr/intrusive_ref_counter.hpp>
  26. #include <boost/iterator/iterator_facade.hpp>
  27. #include <boost/iterator/iterator_categories.hpp>
  28. #include <boost/filesystem/detail/header.hpp> // must be the last #include
  29. //--------------------------------------------------------------------------------------//
  30. namespace boost {
  31. namespace filesystem {
  32. //--------------------------------------------------------------------------------------//
  33. // //
  34. // directory_entry //
  35. // //
  36. //--------------------------------------------------------------------------------------//
  37. // GCC has a problem with a member function named path within a namespace or
  38. // sub-namespace that also has a class named path. The workaround is to always
  39. // fully qualify the name path when it refers to the class name.
  40. class directory_entry
  41. {
  42. public:
  43. typedef boost::filesystem::path::value_type value_type; // enables class path ctor taking directory_entry
  44. directory_entry() BOOST_NOEXCEPT {}
  45. explicit directory_entry(boost::filesystem::path const& p) :
  46. m_path(p), m_status(file_status()), m_symlink_status(file_status())
  47. {
  48. }
  49. directory_entry(boost::filesystem::path const& p, file_status st, file_status symlink_st = file_status()) :
  50. m_path(p), m_status(st), m_symlink_status(symlink_st)
  51. {
  52. }
  53. directory_entry(directory_entry const& rhs) :
  54. m_path(rhs.m_path), m_status(rhs.m_status), m_symlink_status(rhs.m_symlink_status)
  55. {
  56. }
  57. directory_entry& operator=(directory_entry const& rhs)
  58. {
  59. m_path = rhs.m_path;
  60. m_status = rhs.m_status;
  61. m_symlink_status = rhs.m_symlink_status;
  62. return *this;
  63. }
  64. // As of October 2015 the interaction between noexcept and =default is so troublesome
  65. // for VC++, GCC, and probably other compilers, that =default is not used with noexcept
  66. // functions. GCC is not even consistent for the same release on different platforms.
  67. #if !defined(BOOST_NO_CXX11_RVALUE_REFERENCES)
  68. directory_entry(directory_entry&& rhs) BOOST_NOEXCEPT :
  69. m_path(static_cast< boost::filesystem::path&& >(rhs.m_path)),
  70. m_status(static_cast< file_status&& >(rhs.m_status)),
  71. m_symlink_status(static_cast< file_status&& >(rhs.m_symlink_status))
  72. {
  73. }
  74. directory_entry& operator=(directory_entry&& rhs) BOOST_NOEXCEPT
  75. {
  76. m_path = static_cast< boost::filesystem::path&& >(rhs.m_path);
  77. m_status = static_cast< file_status&& >(rhs.m_status);
  78. m_symlink_status = static_cast< file_status&& >(rhs.m_symlink_status);
  79. return *this;
  80. }
  81. void assign(boost::filesystem::path&& p, file_status st = file_status(), file_status symlink_st = file_status())
  82. {
  83. m_path = static_cast< boost::filesystem::path&& >(p);
  84. m_status = static_cast< file_status&& >(st);
  85. m_symlink_status = static_cast< file_status&& >(symlink_st);
  86. }
  87. #endif
  88. void assign(boost::filesystem::path const& p, file_status st = file_status(), file_status symlink_st = file_status())
  89. {
  90. m_path = p;
  91. #if !defined(BOOST_NO_CXX11_RVALUE_REFERENCES)
  92. m_status = static_cast< file_status&& >(st);
  93. m_symlink_status = static_cast< file_status&& >(symlink_st);
  94. #else
  95. m_status = st;
  96. m_symlink_status = symlink_st;
  97. #endif
  98. }
  99. void replace_filename(boost::filesystem::path const& p, file_status st = file_status(), file_status symlink_st = file_status())
  100. {
  101. m_path.replace_filename(p);
  102. #if !defined(BOOST_NO_CXX11_RVALUE_REFERENCES)
  103. m_status = static_cast< file_status&& >(st);
  104. m_symlink_status = static_cast< file_status&& >(symlink_st);
  105. #else
  106. m_status = st;
  107. m_symlink_status = symlink_st;
  108. #endif
  109. }
  110. #ifndef BOOST_FILESYSTEM_NO_DEPRECATED
  111. BOOST_FILESYSTEM_DETAIL_DEPRECATED("Use directory_entry::replace_filename() instead")
  112. void replace_leaf(boost::filesystem::path const& p, file_status st, file_status symlink_st)
  113. {
  114. replace_filename(p, st, symlink_st);
  115. }
  116. #endif
  117. boost::filesystem::path const& path() const BOOST_NOEXCEPT
  118. {
  119. return m_path;
  120. }
  121. operator boost::filesystem::path const&() const BOOST_NOEXCEPT { return m_path; }
  122. file_status status() const { return get_status(); }
  123. file_status status(system::error_code& ec) const BOOST_NOEXCEPT { return get_status(&ec); }
  124. file_status symlink_status() const { return get_symlink_status(); }
  125. file_status symlink_status(system::error_code& ec) const BOOST_NOEXCEPT { return get_symlink_status(&ec); }
  126. bool operator==(directory_entry const& rhs) const BOOST_NOEXCEPT { return m_path == rhs.m_path; }
  127. bool operator!=(directory_entry const& rhs) const BOOST_NOEXCEPT { return m_path != rhs.m_path; }
  128. bool operator<(directory_entry const& rhs) const BOOST_NOEXCEPT { return m_path < rhs.m_path; }
  129. bool operator<=(directory_entry const& rhs) const BOOST_NOEXCEPT { return m_path <= rhs.m_path; }
  130. bool operator>(directory_entry const& rhs) const BOOST_NOEXCEPT { return m_path > rhs.m_path; }
  131. bool operator>=(directory_entry const& rhs) const BOOST_NOEXCEPT { return m_path >= rhs.m_path; }
  132. private:
  133. BOOST_FILESYSTEM_DECL file_status get_status(system::error_code* ec = NULL) const;
  134. BOOST_FILESYSTEM_DECL file_status get_symlink_status(system::error_code* ec = NULL) const;
  135. private:
  136. boost::filesystem::path m_path;
  137. mutable file_status m_status; // stat()-like
  138. mutable file_status m_symlink_status; // lstat()-like
  139. }; // directory_entry
  140. namespace detail {
  141. namespace path_traits {
  142. // Dispatch function for integration with path class
  143. template< typename Callback >
  144. BOOST_FORCEINLINE typename Callback::result_type dispatch(directory_entry const& de, Callback cb, const codecvt_type* cvt, directory_entry_tag)
  145. {
  146. boost::filesystem::path::string_type const& source = de.path().native();
  147. return cb(source.data(), source.data() + source.size(), cvt);
  148. }
  149. } // namespace path_traits
  150. } // namespace detail
  151. //--------------------------------------------------------------------------------------//
  152. // //
  153. // directory_entry overloads //
  154. // //
  155. //--------------------------------------------------------------------------------------//
  156. // Without these functions, calling (for example) 'is_directory' with a 'directory_entry' results in:
  157. // - a conversion to 'path' using 'operator boost::filesystem::path const&()',
  158. // - then a call to 'is_directory(path const& p)' which recomputes the status with 'detail::status(p)'.
  159. //
  160. // These functions avoid a costly recomputation of the status if one calls 'is_directory(e)' instead of 'is_directory(e.status())'
  161. inline file_status status(directory_entry const& e)
  162. {
  163. return e.status();
  164. }
  165. inline file_status status(directory_entry const& e, system::error_code& ec) BOOST_NOEXCEPT
  166. {
  167. return e.status(ec);
  168. }
  169. inline bool type_present(directory_entry const& e)
  170. {
  171. return filesystem::type_present(e.status());
  172. }
  173. inline bool type_present(directory_entry const& e, system::error_code& ec) BOOST_NOEXCEPT
  174. {
  175. return filesystem::type_present(e.status(ec));
  176. }
  177. inline bool status_known(directory_entry const& e)
  178. {
  179. return filesystem::status_known(e.status());
  180. }
  181. inline bool status_known(directory_entry const& e, system::error_code& ec) BOOST_NOEXCEPT
  182. {
  183. return filesystem::status_known(e.status(ec));
  184. }
  185. inline bool exists(directory_entry const& e)
  186. {
  187. return filesystem::exists(e.status());
  188. }
  189. inline bool exists(directory_entry const& e, system::error_code& ec) BOOST_NOEXCEPT
  190. {
  191. return filesystem::exists(e.status(ec));
  192. }
  193. inline bool is_regular_file(directory_entry const& e)
  194. {
  195. return filesystem::is_regular_file(e.status());
  196. }
  197. inline bool is_regular_file(directory_entry const& e, system::error_code& ec) BOOST_NOEXCEPT
  198. {
  199. return filesystem::is_regular_file(e.status(ec));
  200. }
  201. inline bool is_directory(directory_entry const& e)
  202. {
  203. return filesystem::is_directory(e.status());
  204. }
  205. inline bool is_directory(directory_entry const& e, system::error_code& ec) BOOST_NOEXCEPT
  206. {
  207. return filesystem::is_directory(e.status(ec));
  208. }
  209. inline bool is_symlink(directory_entry const& e)
  210. {
  211. return filesystem::is_symlink(e.symlink_status());
  212. }
  213. inline bool is_symlink(directory_entry const& e, system::error_code& ec) BOOST_NOEXCEPT
  214. {
  215. return filesystem::is_symlink(e.symlink_status(ec));
  216. }
  217. inline bool is_other(directory_entry const& e)
  218. {
  219. return filesystem::is_other(e.status());
  220. }
  221. inline bool is_other(directory_entry const& e, system::error_code& ec) BOOST_NOEXCEPT
  222. {
  223. return filesystem::is_other(e.status(ec));
  224. }
  225. #ifndef BOOST_FILESYSTEM_NO_DEPRECATED
  226. BOOST_FILESYSTEM_DETAIL_DEPRECATED("Use is_regular_file() instead")
  227. inline bool is_regular(directory_entry const& e)
  228. {
  229. return filesystem::is_regular_file(e);
  230. }
  231. #endif
  232. //--------------------------------------------------------------------------------------//
  233. // //
  234. // directory_iterator helpers //
  235. // //
  236. //--------------------------------------------------------------------------------------//
  237. BOOST_SCOPED_ENUM_UT_DECLARE_BEGIN(directory_options, unsigned int)
  238. {
  239. none = 0u,
  240. skip_permission_denied = 1u, // if a directory cannot be opened because of insufficient permissions, pretend that the directory is empty
  241. follow_directory_symlink = 1u << 1u, // recursive_directory_iterator: follow directory symlinks
  242. skip_dangling_symlinks = 1u << 2u, // non-standard extension for recursive_directory_iterator: don't follow dangling directory symlinks,
  243. pop_on_error = 1u << 3u, // non-standard extension for recursive_directory_iterator: instead of producing an end iterator on errors,
  244. // repeatedly invoke pop() until it succeeds or the iterator becomes equal to end iterator
  245. _detail_no_follow = 1u << 4u, // internal use only
  246. _detail_no_push = 1u << 5u // internal use only
  247. }
  248. BOOST_SCOPED_ENUM_DECLARE_END(directory_options)
  249. BOOST_BITMASK(BOOST_SCOPED_ENUM_NATIVE(directory_options))
  250. class directory_iterator;
  251. namespace detail {
  252. struct dir_itr_imp :
  253. public boost::intrusive_ref_counter< dir_itr_imp >
  254. {
  255. #ifdef BOOST_WINDOWS_API
  256. bool close_handle;
  257. unsigned char extra_data_format;
  258. std::size_t current_offset;
  259. #endif
  260. directory_entry dir_entry;
  261. void* handle;
  262. dir_itr_imp() BOOST_NOEXCEPT :
  263. #ifdef BOOST_WINDOWS_API
  264. close_handle(false),
  265. extra_data_format(0u),
  266. current_offset(0u),
  267. #endif
  268. handle(NULL)
  269. {
  270. }
  271. BOOST_FILESYSTEM_DECL ~dir_itr_imp() BOOST_NOEXCEPT;
  272. BOOST_FILESYSTEM_DECL static void* operator new(std::size_t class_size, std::size_t extra_size) BOOST_NOEXCEPT;
  273. BOOST_FILESYSTEM_DECL static void operator delete(void* p, std::size_t extra_size) BOOST_NOEXCEPT;
  274. BOOST_FILESYSTEM_DECL static void operator delete(void* p) BOOST_NOEXCEPT;
  275. };
  276. struct directory_iterator_params;
  277. BOOST_FILESYSTEM_DECL void directory_iterator_construct(directory_iterator& it, path const& p, unsigned int opts, directory_iterator_params* params, system::error_code* ec);
  278. BOOST_FILESYSTEM_DECL void directory_iterator_increment(directory_iterator& it, system::error_code* ec);
  279. } // namespace detail
  280. //--------------------------------------------------------------------------------------//
  281. // //
  282. // directory_iterator //
  283. // //
  284. //--------------------------------------------------------------------------------------//
  285. class directory_iterator :
  286. public boost::iterator_facade<
  287. directory_iterator,
  288. directory_entry,
  289. boost::single_pass_traversal_tag
  290. >
  291. {
  292. friend class boost::iterator_core_access;
  293. friend BOOST_FILESYSTEM_DECL void detail::directory_iterator_construct(directory_iterator& it, path const& p, unsigned int opts, detail::directory_iterator_params* params, system::error_code* ec);
  294. friend BOOST_FILESYSTEM_DECL void detail::directory_iterator_increment(directory_iterator& it, system::error_code* ec);
  295. public:
  296. directory_iterator() BOOST_NOEXCEPT {} // creates the "end" iterator
  297. // iterator_facade derived classes don't seem to like implementations in
  298. // separate translation unit dll's, so forward to detail functions
  299. explicit directory_iterator(path const& p, BOOST_SCOPED_ENUM_NATIVE(directory_options) opts = directory_options::none)
  300. {
  301. detail::directory_iterator_construct(*this, p, static_cast< unsigned int >(opts), NULL, NULL);
  302. }
  303. directory_iterator(path const& p, system::error_code& ec) BOOST_NOEXCEPT
  304. {
  305. detail::directory_iterator_construct(*this, p, static_cast< unsigned int >(directory_options::none), NULL, &ec);
  306. }
  307. directory_iterator(path const& p, BOOST_SCOPED_ENUM_NATIVE(directory_options) opts, system::error_code& ec) BOOST_NOEXCEPT
  308. {
  309. detail::directory_iterator_construct(*this, p, static_cast< unsigned int >(opts), NULL, &ec);
  310. }
  311. BOOST_DEFAULTED_FUNCTION(directory_iterator(directory_iterator const& that), : m_imp(that.m_imp) {})
  312. BOOST_DEFAULTED_FUNCTION(directory_iterator& operator=(directory_iterator const& that), { m_imp = that.m_imp; return *this; })
  313. #if !defined(BOOST_NO_CXX11_RVALUE_REFERENCES)
  314. directory_iterator(directory_iterator&& that) BOOST_NOEXCEPT :
  315. m_imp(static_cast< boost::intrusive_ptr< detail::dir_itr_imp >&& >(that.m_imp))
  316. {
  317. }
  318. directory_iterator& operator=(directory_iterator&& that) BOOST_NOEXCEPT
  319. {
  320. m_imp = static_cast< boost::intrusive_ptr< detail::dir_itr_imp >&& >(that.m_imp);
  321. return *this;
  322. }
  323. #endif // !defined(BOOST_NO_CXX11_RVALUE_REFERENCES)
  324. directory_iterator& increment(system::error_code& ec) BOOST_NOEXCEPT
  325. {
  326. detail::directory_iterator_increment(*this, &ec);
  327. return *this;
  328. }
  329. private:
  330. boost::iterator_facade<
  331. directory_iterator,
  332. directory_entry,
  333. boost::single_pass_traversal_tag
  334. >::reference dereference() const
  335. {
  336. BOOST_ASSERT_MSG(!is_end(), "attempt to dereference end directory iterator");
  337. return m_imp->dir_entry;
  338. }
  339. void increment() { detail::directory_iterator_increment(*this, NULL); }
  340. bool equal(directory_iterator const& rhs) const BOOST_NOEXCEPT
  341. {
  342. return m_imp == rhs.m_imp || (is_end() && rhs.is_end());
  343. }
  344. bool is_end() const BOOST_NOEXCEPT
  345. {
  346. // Note: The check for handle is needed because the iterator can be copied and the copy
  347. // can be incremented to end while the original iterator still refers to the same dir_itr_imp.
  348. return !m_imp || !m_imp->handle;
  349. }
  350. private:
  351. // intrusive_ptr provides the shallow-copy semantics required for single pass iterators
  352. // (i.e. InputIterators). The end iterator is indicated by is_end().
  353. boost::intrusive_ptr< detail::dir_itr_imp > m_imp;
  354. };
  355. // enable directory_iterator C++11 range-based for statement use --------------------//
  356. // begin() and end() are only used by a range-based for statement in the context of
  357. // auto - thus the top-level const is stripped - so returning const is harmless and
  358. // emphasizes begin() is just a pass through.
  359. inline directory_iterator const& begin(directory_iterator const& iter) BOOST_NOEXCEPT
  360. {
  361. return iter;
  362. }
  363. inline directory_iterator end(directory_iterator const&) BOOST_NOEXCEPT
  364. {
  365. return directory_iterator();
  366. }
  367. // enable C++14 generic accessors for range const iterators
  368. inline directory_iterator const& cbegin(directory_iterator const& iter) BOOST_NOEXCEPT
  369. {
  370. return iter;
  371. }
  372. inline directory_iterator cend(directory_iterator const&) BOOST_NOEXCEPT
  373. {
  374. return directory_iterator();
  375. }
  376. // enable directory_iterator BOOST_FOREACH -----------------------------------------//
  377. inline directory_iterator& range_begin(directory_iterator& iter) BOOST_NOEXCEPT
  378. {
  379. return iter;
  380. }
  381. inline directory_iterator range_begin(directory_iterator const& iter) BOOST_NOEXCEPT
  382. {
  383. return iter;
  384. }
  385. inline directory_iterator range_end(directory_iterator&) BOOST_NOEXCEPT
  386. {
  387. return directory_iterator();
  388. }
  389. inline directory_iterator range_end(directory_iterator const&) BOOST_NOEXCEPT
  390. {
  391. return directory_iterator();
  392. }
  393. } // namespace filesystem
  394. // namespace boost template specializations
  395. template< typename C, typename Enabler >
  396. struct range_mutable_iterator;
  397. template<>
  398. struct range_mutable_iterator< boost::filesystem::directory_iterator, void >
  399. {
  400. typedef boost::filesystem::directory_iterator type;
  401. };
  402. template< typename C, typename Enabler >
  403. struct range_const_iterator;
  404. template<>
  405. struct range_const_iterator< boost::filesystem::directory_iterator, void >
  406. {
  407. typedef boost::filesystem::directory_iterator type;
  408. };
  409. namespace filesystem {
  410. //--------------------------------------------------------------------------------------//
  411. // //
  412. // recursive_directory_iterator helpers //
  413. // //
  414. //--------------------------------------------------------------------------------------//
  415. #if !defined(BOOST_FILESYSTEM_NO_DEPRECATED)
  416. // Deprecated enum, use directory_options instead
  417. BOOST_SCOPED_ENUM_UT_DECLARE_BEGIN(symlink_option, unsigned int)
  418. {
  419. none = static_cast< unsigned int >(directory_options::none),
  420. no_recurse = none, // don't follow directory symlinks (default behavior)
  421. recurse = static_cast< unsigned int >(directory_options::follow_directory_symlink), // follow directory symlinks
  422. _detail_no_push = static_cast< unsigned int >(directory_options::_detail_no_push) // internal use only
  423. }
  424. BOOST_SCOPED_ENUM_DECLARE_END(symlink_option)
  425. BOOST_BITMASK(BOOST_SCOPED_ENUM_NATIVE(symlink_option))
  426. #endif // BOOST_FILESYSTEM_NO_DEPRECATED
  427. class recursive_directory_iterator;
  428. namespace detail {
  429. struct recur_dir_itr_imp :
  430. public boost::intrusive_ref_counter< recur_dir_itr_imp >
  431. {
  432. typedef directory_iterator element_type;
  433. std::vector< element_type > m_stack;
  434. // directory_options values, declared as unsigned int for ABI compatibility
  435. unsigned int m_options;
  436. explicit recur_dir_itr_imp(unsigned int opts) BOOST_NOEXCEPT : m_options(opts) {}
  437. };
  438. BOOST_FILESYSTEM_DECL void recursive_directory_iterator_construct(recursive_directory_iterator& it, path const& dir_path, unsigned int opts, system::error_code* ec);
  439. BOOST_FILESYSTEM_DECL void recursive_directory_iterator_increment(recursive_directory_iterator& it, system::error_code* ec);
  440. BOOST_FILESYSTEM_DECL void recursive_directory_iterator_pop(recursive_directory_iterator& it, system::error_code* ec);
  441. } // namespace detail
  442. //--------------------------------------------------------------------------------------//
  443. // //
  444. // recursive_directory_iterator //
  445. // //
  446. //--------------------------------------------------------------------------------------//
  447. class recursive_directory_iterator :
  448. public boost::iterator_facade<
  449. recursive_directory_iterator,
  450. directory_entry,
  451. boost::single_pass_traversal_tag
  452. >
  453. {
  454. friend class boost::iterator_core_access;
  455. friend BOOST_FILESYSTEM_DECL void detail::recursive_directory_iterator_construct(recursive_directory_iterator& it, path const& dir_path, unsigned int opts, system::error_code* ec);
  456. friend BOOST_FILESYSTEM_DECL void detail::recursive_directory_iterator_increment(recursive_directory_iterator& it, system::error_code* ec);
  457. friend BOOST_FILESYSTEM_DECL void detail::recursive_directory_iterator_pop(recursive_directory_iterator& it, system::error_code* ec);
  458. public:
  459. recursive_directory_iterator() BOOST_NOEXCEPT {} // creates the "end" iterator
  460. explicit recursive_directory_iterator(path const& dir_path)
  461. {
  462. detail::recursive_directory_iterator_construct(*this, dir_path, static_cast< unsigned int >(directory_options::none), NULL);
  463. }
  464. recursive_directory_iterator(path const& dir_path, system::error_code& ec)
  465. {
  466. detail::recursive_directory_iterator_construct(*this, dir_path, static_cast< unsigned int >(directory_options::none), &ec);
  467. }
  468. recursive_directory_iterator(path const& dir_path, BOOST_SCOPED_ENUM_NATIVE(directory_options) opts)
  469. {
  470. detail::recursive_directory_iterator_construct(*this, dir_path, static_cast< unsigned int >(opts), NULL);
  471. }
  472. recursive_directory_iterator(path const& dir_path, BOOST_SCOPED_ENUM_NATIVE(directory_options) opts, system::error_code& ec)
  473. {
  474. detail::recursive_directory_iterator_construct(*this, dir_path, static_cast< unsigned int >(opts), &ec);
  475. }
  476. #if !defined(BOOST_FILESYSTEM_NO_DEPRECATED)
  477. // Deprecated constructors
  478. BOOST_FILESYSTEM_DETAIL_DEPRECATED("Use directory_options instead of symlink_option")
  479. recursive_directory_iterator(path const& dir_path, BOOST_SCOPED_ENUM_NATIVE(symlink_option) opts)
  480. {
  481. detail::recursive_directory_iterator_construct(*this, dir_path, static_cast< unsigned int >(opts), NULL);
  482. }
  483. BOOST_FILESYSTEM_DETAIL_DEPRECATED("Use directory_options instead of symlink_option")
  484. recursive_directory_iterator(path const& dir_path, BOOST_SCOPED_ENUM_NATIVE(symlink_option) opts, system::error_code& ec) BOOST_NOEXCEPT
  485. {
  486. detail::recursive_directory_iterator_construct(*this, dir_path, static_cast< unsigned int >(opts), &ec);
  487. }
  488. #endif // BOOST_FILESYSTEM_NO_DEPRECATED
  489. BOOST_DEFAULTED_FUNCTION(recursive_directory_iterator(recursive_directory_iterator const& that), : m_imp(that.m_imp) {})
  490. BOOST_DEFAULTED_FUNCTION(recursive_directory_iterator& operator=(recursive_directory_iterator const& that), { m_imp = that.m_imp; return *this; })
  491. #if !defined(BOOST_NO_CXX11_RVALUE_REFERENCES)
  492. recursive_directory_iterator(recursive_directory_iterator&& that) BOOST_NOEXCEPT :
  493. m_imp(static_cast< boost::intrusive_ptr< detail::recur_dir_itr_imp >&& >(that.m_imp))
  494. {
  495. }
  496. recursive_directory_iterator& operator=(recursive_directory_iterator&& that) BOOST_NOEXCEPT
  497. {
  498. m_imp = static_cast< boost::intrusive_ptr< detail::recur_dir_itr_imp >&& >(that.m_imp);
  499. return *this;
  500. }
  501. #endif // !defined(BOOST_NO_CXX11_RVALUE_REFERENCES)
  502. recursive_directory_iterator& increment(system::error_code& ec) BOOST_NOEXCEPT
  503. {
  504. detail::recursive_directory_iterator_increment(*this, &ec);
  505. return *this;
  506. }
  507. int depth() const BOOST_NOEXCEPT
  508. {
  509. BOOST_ASSERT_MSG(!is_end(), "depth() on end recursive_directory_iterator");
  510. return static_cast< int >(m_imp->m_stack.size() - 1u);
  511. }
  512. bool recursion_pending() const BOOST_NOEXCEPT
  513. {
  514. BOOST_ASSERT_MSG(!is_end(), "recursion_pending() on end recursive_directory_iterator");
  515. return (m_imp->m_options & static_cast< unsigned int >(directory_options::_detail_no_push)) == 0u;
  516. }
  517. #ifndef BOOST_FILESYSTEM_NO_DEPRECATED
  518. BOOST_FILESYSTEM_DETAIL_DEPRECATED("Use recursive_directory_iterator::depth() instead")
  519. int level() const BOOST_NOEXCEPT
  520. {
  521. return depth();
  522. }
  523. bool no_push_pending() const BOOST_NOEXCEPT { return !recursion_pending(); }
  524. bool no_push_request() const BOOST_NOEXCEPT { return !recursion_pending(); }
  525. #endif
  526. void pop()
  527. {
  528. detail::recursive_directory_iterator_pop(*this, NULL);
  529. }
  530. void pop(system::error_code& ec) BOOST_NOEXCEPT
  531. {
  532. detail::recursive_directory_iterator_pop(*this, &ec);
  533. }
  534. void disable_recursion_pending(bool value = true) BOOST_NOEXCEPT
  535. {
  536. BOOST_ASSERT_MSG(!is_end(), "disable_recursion_pending() on end recursive_directory_iterator");
  537. if (value)
  538. m_imp->m_options |= static_cast< unsigned int >(directory_options::_detail_no_push);
  539. else
  540. m_imp->m_options &= ~static_cast< unsigned int >(directory_options::_detail_no_push);
  541. }
  542. #ifndef BOOST_FILESYSTEM_NO_DEPRECATED
  543. BOOST_FILESYSTEM_DETAIL_DEPRECATED("Use recursive_directory_iterator::disable_recursion_pending() instead")
  544. void no_push(bool value = true) BOOST_NOEXCEPT
  545. {
  546. disable_recursion_pending(value);
  547. }
  548. #endif
  549. file_status status() const
  550. {
  551. BOOST_ASSERT_MSG(!is_end(), "status() on end recursive_directory_iterator");
  552. return m_imp->m_stack.back()->status();
  553. }
  554. file_status symlink_status() const
  555. {
  556. BOOST_ASSERT_MSG(!is_end(), "symlink_status() on end recursive_directory_iterator");
  557. return m_imp->m_stack.back()->symlink_status();
  558. }
  559. private:
  560. boost::iterator_facade<
  561. recursive_directory_iterator,
  562. directory_entry,
  563. boost::single_pass_traversal_tag
  564. >::reference dereference() const
  565. {
  566. BOOST_ASSERT_MSG(!is_end(), "dereference of end recursive_directory_iterator");
  567. return *m_imp->m_stack.back();
  568. }
  569. void increment() { detail::recursive_directory_iterator_increment(*this, NULL); }
  570. bool equal(recursive_directory_iterator const& rhs) const BOOST_NOEXCEPT
  571. {
  572. return m_imp == rhs.m_imp || (is_end() && rhs.is_end());
  573. }
  574. bool is_end() const BOOST_NOEXCEPT
  575. {
  576. // Note: The check for m_stack.empty() is needed because the iterator can be copied and the copy
  577. // can be incremented to end while the original iterator still refers to the same recur_dir_itr_imp.
  578. return !m_imp || m_imp->m_stack.empty();
  579. }
  580. private:
  581. // intrusive_ptr provides the shallow-copy semantics required for single pass iterators
  582. // (i.e. InputIterators). The end iterator is indicated by is_end().
  583. boost::intrusive_ptr< detail::recur_dir_itr_imp > m_imp;
  584. };
  585. #if !defined(BOOST_FILESYSTEM_NO_DEPRECATED)
  586. BOOST_FILESYSTEM_DETAIL_DEPRECATED("Use recursive_directory_iterator instead")
  587. typedef recursive_directory_iterator wrecursive_directory_iterator;
  588. #endif
  589. // enable recursive directory iterator C++11 range-base for statement use ----------//
  590. // begin() and end() are only used by a range-based for statement in the context of
  591. // auto - thus the top-level const is stripped - so returning const is harmless and
  592. // emphasizes begin() is just a pass through.
  593. inline recursive_directory_iterator const& begin(recursive_directory_iterator const& iter) BOOST_NOEXCEPT
  594. {
  595. return iter;
  596. }
  597. inline recursive_directory_iterator end(recursive_directory_iterator const&) BOOST_NOEXCEPT
  598. {
  599. return recursive_directory_iterator();
  600. }
  601. // enable C++14 generic accessors for range const iterators
  602. inline recursive_directory_iterator const& cbegin(recursive_directory_iterator const& iter) BOOST_NOEXCEPT
  603. {
  604. return iter;
  605. }
  606. inline recursive_directory_iterator cend(recursive_directory_iterator const&) BOOST_NOEXCEPT
  607. {
  608. return recursive_directory_iterator();
  609. }
  610. // enable recursive directory iterator BOOST_FOREACH -------------------------------//
  611. inline recursive_directory_iterator& range_begin(recursive_directory_iterator& iter) BOOST_NOEXCEPT
  612. {
  613. return iter;
  614. }
  615. inline recursive_directory_iterator range_begin(recursive_directory_iterator const& iter) BOOST_NOEXCEPT
  616. {
  617. return iter;
  618. }
  619. inline recursive_directory_iterator range_end(recursive_directory_iterator&) BOOST_NOEXCEPT
  620. {
  621. return recursive_directory_iterator();
  622. }
  623. inline recursive_directory_iterator range_end(recursive_directory_iterator const&) BOOST_NOEXCEPT
  624. {
  625. return recursive_directory_iterator();
  626. }
  627. } // namespace filesystem
  628. // namespace boost template specializations
  629. template<>
  630. struct range_mutable_iterator< boost::filesystem::recursive_directory_iterator, void >
  631. {
  632. typedef boost::filesystem::recursive_directory_iterator type;
  633. };
  634. template<>
  635. struct range_const_iterator< boost::filesystem::recursive_directory_iterator, void >
  636. {
  637. typedef boost::filesystem::recursive_directory_iterator type;
  638. };
  639. } // namespace boost
  640. #include <boost/filesystem/detail/footer.hpp>
  641. #endif // BOOST_FILESYSTEM_DIRECTORY_HPP