directory.hpp 38 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121
  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. class directory_iterator;
  33. namespace detail {
  34. struct directory_iterator_params;
  35. BOOST_FILESYSTEM_DECL void directory_iterator_construct(directory_iterator& it, path const& p, unsigned int opts, directory_iterator_params* params, system::error_code* ec);
  36. BOOST_FILESYSTEM_DECL void directory_iterator_increment(directory_iterator& it, system::error_code* ec);
  37. } // namespace detail
  38. //--------------------------------------------------------------------------------------//
  39. // //
  40. // directory_entry //
  41. // //
  42. //--------------------------------------------------------------------------------------//
  43. // GCC has a problem with a member function named path within a namespace or
  44. // sub-namespace that also has a class named path. The workaround is to always
  45. // fully qualify the name path when it refers to the class name.
  46. class directory_entry
  47. {
  48. 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);
  49. friend BOOST_FILESYSTEM_DECL void detail::directory_iterator_increment(directory_iterator& it, system::error_code* ec);
  50. public:
  51. typedef boost::filesystem::path::value_type value_type; // enables class path ctor taking directory_entry
  52. directory_entry() BOOST_NOEXCEPT {}
  53. explicit directory_entry(boost::filesystem::path const& p);
  54. #if BOOST_FILESYSTEM_VERSION >= 4
  55. directory_entry(boost::filesystem::path const& p, system::error_code& ec) :
  56. m_path(p)
  57. {
  58. refresh_impl(&ec);
  59. if (ec)
  60. m_path.clear();
  61. }
  62. #else
  63. directory_entry(boost::filesystem::path const& p, file_status st, file_status symlink_st = file_status()) :
  64. m_path(p), m_status(st), m_symlink_status(symlink_st)
  65. {
  66. }
  67. #endif
  68. directory_entry(directory_entry const& rhs) :
  69. m_path(rhs.m_path), m_status(rhs.m_status), m_symlink_status(rhs.m_symlink_status)
  70. {
  71. }
  72. directory_entry& operator=(directory_entry const& rhs)
  73. {
  74. m_path = rhs.m_path;
  75. m_status = rhs.m_status;
  76. m_symlink_status = rhs.m_symlink_status;
  77. return *this;
  78. }
  79. // As of October 2015 the interaction between noexcept and =default is so troublesome
  80. // for VC++, GCC, and probably other compilers, that =default is not used with noexcept
  81. // functions. GCC is not even consistent for the same release on different platforms.
  82. #if !defined(BOOST_NO_CXX11_RVALUE_REFERENCES)
  83. directory_entry(directory_entry&& rhs) BOOST_NOEXCEPT :
  84. m_path(static_cast< boost::filesystem::path&& >(rhs.m_path)),
  85. m_status(static_cast< file_status&& >(rhs.m_status)),
  86. m_symlink_status(static_cast< file_status&& >(rhs.m_symlink_status))
  87. {
  88. }
  89. directory_entry& operator=(directory_entry&& rhs) BOOST_NOEXCEPT
  90. {
  91. m_path = static_cast< boost::filesystem::path&& >(rhs.m_path);
  92. m_status = static_cast< file_status&& >(rhs.m_status);
  93. m_symlink_status = static_cast< file_status&& >(rhs.m_symlink_status);
  94. return *this;
  95. }
  96. void assign(boost::filesystem::path&& p);
  97. #if BOOST_FILESYSTEM_VERSION >= 4
  98. void assign(boost::filesystem::path&& p, system::error_code& ec)
  99. {
  100. m_path = static_cast< boost::filesystem::path&& >(p);
  101. refresh_impl(&ec);
  102. }
  103. #else
  104. void assign(boost::filesystem::path&& p, file_status st, file_status symlink_st = file_status())
  105. {
  106. assign_with_status(static_cast< boost::filesystem::path&& >(p), st, symlink_st);
  107. }
  108. #endif
  109. #endif // !defined(BOOST_NO_CXX11_RVALUE_REFERENCES)
  110. void assign(boost::filesystem::path const& p);
  111. #if BOOST_FILESYSTEM_VERSION >= 4
  112. void assign(boost::filesystem::path const& p, system::error_code& ec)
  113. {
  114. m_path = p;
  115. refresh_impl(&ec);
  116. }
  117. #else
  118. void assign(boost::filesystem::path const& p, file_status st, file_status symlink_st = file_status())
  119. {
  120. assign_with_status(p, st, symlink_st);
  121. }
  122. #endif
  123. void replace_filename(boost::filesystem::path const& p);
  124. #if BOOST_FILESYSTEM_VERSION >= 4
  125. void replace_filename(boost::filesystem::path const& p, system::error_code& ec)
  126. {
  127. m_path.replace_filename(p);
  128. refresh_impl(&ec);
  129. }
  130. #else
  131. void replace_filename(boost::filesystem::path const& p, file_status st, file_status symlink_st = file_status())
  132. {
  133. replace_filename_with_status(p, st, symlink_st);
  134. }
  135. BOOST_FILESYSTEM_DETAIL_DEPRECATED("Use directory_entry::replace_filename() instead")
  136. void replace_leaf(boost::filesystem::path const& p, file_status st, file_status symlink_st)
  137. {
  138. replace_filename_with_status(p, st, symlink_st);
  139. }
  140. #endif
  141. boost::filesystem::path const& path() const BOOST_NOEXCEPT { return m_path; }
  142. operator boost::filesystem::path const&() const BOOST_NOEXCEPT { return m_path; }
  143. void refresh() { refresh_impl(); }
  144. void refresh(system::error_code& ec) BOOST_NOEXCEPT { refresh_impl(&ec); }
  145. file_status status() const
  146. {
  147. if (!filesystem::status_known(m_status))
  148. refresh_impl();
  149. return m_status;
  150. }
  151. file_status status(system::error_code& ec) const BOOST_NOEXCEPT
  152. {
  153. ec.clear();
  154. if (!filesystem::status_known(m_status))
  155. refresh_impl(&ec);
  156. return m_status;
  157. }
  158. file_status symlink_status() const
  159. {
  160. if (!filesystem::status_known(m_symlink_status))
  161. refresh_impl();
  162. return m_symlink_status;
  163. }
  164. file_status symlink_status(system::error_code& ec) const BOOST_NOEXCEPT
  165. {
  166. ec.clear();
  167. if (!filesystem::status_known(m_symlink_status))
  168. refresh_impl(&ec);
  169. return m_symlink_status;
  170. }
  171. filesystem::file_type file_type() const
  172. {
  173. if (!filesystem::type_present(m_status))
  174. refresh_impl();
  175. return m_status.type();
  176. }
  177. filesystem::file_type file_type(system::error_code& ec) const BOOST_NOEXCEPT
  178. {
  179. ec.clear();
  180. if (!filesystem::type_present(m_status))
  181. refresh_impl(&ec);
  182. return m_status.type();
  183. }
  184. filesystem::file_type symlink_file_type() const
  185. {
  186. if (!filesystem::type_present(m_symlink_status))
  187. refresh_impl();
  188. return m_symlink_status.type();
  189. }
  190. filesystem::file_type symlink_file_type(system::error_code& ec) const BOOST_NOEXCEPT
  191. {
  192. ec.clear();
  193. if (!filesystem::type_present(m_symlink_status))
  194. refresh_impl(&ec);
  195. return m_symlink_status.type();
  196. }
  197. bool exists() const
  198. {
  199. filesystem::file_type ft = this->file_type();
  200. return ft != filesystem::status_error && ft != filesystem::file_not_found;
  201. }
  202. bool exists(system::error_code& ec) const BOOST_NOEXCEPT
  203. {
  204. filesystem::file_type ft = this->file_type(ec);
  205. return ft != filesystem::status_error && ft != filesystem::file_not_found;
  206. }
  207. bool is_regular_file() const
  208. {
  209. return this->file_type() == filesystem::regular_file;
  210. }
  211. bool is_regular_file(system::error_code& ec) const BOOST_NOEXCEPT
  212. {
  213. return this->file_type(ec) == filesystem::regular_file;
  214. }
  215. bool is_directory() const
  216. {
  217. return this->file_type() == filesystem::directory_file;
  218. }
  219. bool is_directory(system::error_code& ec) const BOOST_NOEXCEPT
  220. {
  221. return this->file_type(ec) == filesystem::directory_file;
  222. }
  223. bool is_symlink() const
  224. {
  225. return this->symlink_file_type() == filesystem::symlink_file;
  226. }
  227. bool is_symlink(system::error_code& ec) const BOOST_NOEXCEPT
  228. {
  229. return this->symlink_file_type(ec) == filesystem::symlink_file;
  230. }
  231. bool is_block_file() const
  232. {
  233. return this->file_type() == filesystem::block_file;
  234. }
  235. bool is_block_file(system::error_code& ec) const BOOST_NOEXCEPT
  236. {
  237. return this->file_type(ec) == filesystem::block_file;
  238. }
  239. bool is_character_file() const
  240. {
  241. return this->file_type() == filesystem::character_file;
  242. }
  243. bool is_character_file(system::error_code& ec) const BOOST_NOEXCEPT
  244. {
  245. return this->file_type(ec) == filesystem::character_file;
  246. }
  247. bool is_fifo() const
  248. {
  249. return this->file_type() == filesystem::fifo_file;
  250. }
  251. bool is_fifo(system::error_code& ec) const BOOST_NOEXCEPT
  252. {
  253. return this->file_type(ec) == filesystem::fifo_file;
  254. }
  255. bool is_socket() const
  256. {
  257. return this->file_type() == filesystem::socket_file;
  258. }
  259. bool is_socket(system::error_code& ec) const BOOST_NOEXCEPT
  260. {
  261. return this->file_type(ec) == filesystem::socket_file;
  262. }
  263. bool is_reparse_file() const
  264. {
  265. return this->symlink_file_type() == filesystem::reparse_file;
  266. }
  267. bool is_reparse_file(system::error_code& ec) const BOOST_NOEXCEPT
  268. {
  269. return this->symlink_file_type(ec) == filesystem::reparse_file;
  270. }
  271. bool is_other() const
  272. {
  273. filesystem::file_type ft = this->file_type();
  274. return ft != filesystem::status_error && ft != filesystem::file_not_found &&
  275. ft != filesystem::regular_file && ft != filesystem::directory_file;
  276. }
  277. bool is_other(system::error_code& ec) const BOOST_NOEXCEPT
  278. {
  279. filesystem::file_type ft = this->file_type(ec);
  280. return ft != filesystem::status_error && ft != filesystem::file_not_found &&
  281. ft != filesystem::regular_file && ft != filesystem::directory_file;
  282. }
  283. bool operator==(directory_entry const& rhs) const { return m_path == rhs.m_path; }
  284. bool operator!=(directory_entry const& rhs) const { return m_path != rhs.m_path; }
  285. bool operator<(directory_entry const& rhs) const { return m_path < rhs.m_path; }
  286. bool operator<=(directory_entry const& rhs) const { return m_path <= rhs.m_path; }
  287. bool operator>(directory_entry const& rhs) const { return m_path > rhs.m_path; }
  288. bool operator>=(directory_entry const& rhs) const { return m_path >= rhs.m_path; }
  289. private:
  290. BOOST_FILESYSTEM_DECL void refresh_impl(system::error_code* ec = NULL) const;
  291. #if !defined(BOOST_NO_CXX11_RVALUE_REFERENCES)
  292. void assign_with_status(boost::filesystem::path&& p, file_status st, file_status symlink_st)
  293. {
  294. m_path = static_cast< boost::filesystem::path&& >(p);
  295. m_status = static_cast< file_status&& >(st);
  296. m_symlink_status = static_cast< file_status&& >(symlink_st);
  297. }
  298. #endif
  299. void assign_with_status(boost::filesystem::path const& p, file_status st, file_status symlink_st)
  300. {
  301. m_path = p;
  302. #if !defined(BOOST_NO_CXX11_RVALUE_REFERENCES)
  303. m_status = static_cast< file_status&& >(st);
  304. m_symlink_status = static_cast< file_status&& >(symlink_st);
  305. #else
  306. m_status = st;
  307. m_symlink_status = symlink_st;
  308. #endif
  309. }
  310. void replace_filename_with_status(boost::filesystem::path const& p, file_status st, file_status symlink_st)
  311. {
  312. m_path.replace_filename(p);
  313. #if !defined(BOOST_NO_CXX11_RVALUE_REFERENCES)
  314. m_status = static_cast< file_status&& >(st);
  315. m_symlink_status = static_cast< file_status&& >(symlink_st);
  316. #else
  317. m_status = st;
  318. m_symlink_status = symlink_st;
  319. #endif
  320. }
  321. private:
  322. boost::filesystem::path m_path;
  323. mutable file_status m_status; // stat()-like
  324. mutable file_status m_symlink_status; // lstat()-like
  325. };
  326. #if !defined(BOOST_FILESYSTEM_SOURCE)
  327. inline directory_entry::directory_entry(boost::filesystem::path const& p) :
  328. m_path(p)
  329. {
  330. #if BOOST_FILESYSTEM_VERSION >= 4
  331. refresh_impl();
  332. #endif
  333. }
  334. #if !defined(BOOST_NO_CXX11_RVALUE_REFERENCES)
  335. inline void directory_entry::assign(boost::filesystem::path&& p)
  336. {
  337. m_path = static_cast< boost::filesystem::path&& >(p);
  338. #if BOOST_FILESYSTEM_VERSION >= 4
  339. refresh_impl();
  340. #else
  341. m_status = file_status();
  342. m_symlink_status = file_status();
  343. #endif
  344. }
  345. #endif // !defined(BOOST_NO_CXX11_RVALUE_REFERENCES)
  346. inline void directory_entry::assign(boost::filesystem::path const& p)
  347. {
  348. m_path = p;
  349. #if BOOST_FILESYSTEM_VERSION >= 4
  350. refresh_impl();
  351. #else
  352. m_status = file_status();
  353. m_symlink_status = file_status();
  354. #endif
  355. }
  356. inline void directory_entry::replace_filename(boost::filesystem::path const& p)
  357. {
  358. m_path.replace_filename(p);
  359. #if BOOST_FILESYSTEM_VERSION >= 4
  360. refresh_impl();
  361. #else
  362. m_status = file_status();
  363. m_symlink_status = file_status();
  364. #endif
  365. }
  366. #endif // !defined(BOOST_FILESYSTEM_SOURCE)
  367. namespace detail {
  368. namespace path_traits {
  369. // Dispatch function for integration with path class
  370. template< typename Callback >
  371. BOOST_FORCEINLINE typename Callback::result_type dispatch(directory_entry const& de, Callback cb, const codecvt_type* cvt, directory_entry_tag)
  372. {
  373. boost::filesystem::path::string_type const& source = de.path().native();
  374. return cb(source.data(), source.data() + source.size(), cvt);
  375. }
  376. } // namespace path_traits
  377. } // namespace detail
  378. //--------------------------------------------------------------------------------------//
  379. // //
  380. // directory_entry overloads //
  381. // //
  382. //--------------------------------------------------------------------------------------//
  383. // Without these functions, calling (for example) 'is_directory' with a 'directory_entry' results in:
  384. // - a conversion to 'path' using 'operator boost::filesystem::path const&()',
  385. // - then a call to 'is_directory(path const& p)' which recomputes the status with 'detail::status(p)'.
  386. //
  387. // These functions avoid a costly recomputation of the status if one calls 'is_directory(e)' instead of 'is_directory(e.status())'
  388. inline file_status status(directory_entry const& e)
  389. {
  390. return e.status();
  391. }
  392. inline file_status status(directory_entry const& e, system::error_code& ec) BOOST_NOEXCEPT
  393. {
  394. return e.status(ec);
  395. }
  396. inline file_status symlink_status(directory_entry const& e)
  397. {
  398. return e.symlink_status();
  399. }
  400. inline file_status symlink_status(directory_entry const& e, system::error_code& ec) BOOST_NOEXCEPT
  401. {
  402. return e.symlink_status(ec);
  403. }
  404. inline bool type_present(directory_entry const& e)
  405. {
  406. return e.file_type() != filesystem::status_error;
  407. }
  408. inline bool type_present(directory_entry const& e, system::error_code& ec) BOOST_NOEXCEPT
  409. {
  410. return e.file_type(ec) != filesystem::status_error;
  411. }
  412. inline bool status_known(directory_entry const& e)
  413. {
  414. return filesystem::status_known(e.status());
  415. }
  416. inline bool status_known(directory_entry const& e, system::error_code& ec) BOOST_NOEXCEPT
  417. {
  418. return filesystem::status_known(e.status(ec));
  419. }
  420. inline bool exists(directory_entry const& e)
  421. {
  422. return e.exists();
  423. }
  424. inline bool exists(directory_entry const& e, system::error_code& ec) BOOST_NOEXCEPT
  425. {
  426. return e.exists(ec);
  427. }
  428. inline bool is_regular_file(directory_entry const& e)
  429. {
  430. return e.is_regular_file();
  431. }
  432. inline bool is_regular_file(directory_entry const& e, system::error_code& ec) BOOST_NOEXCEPT
  433. {
  434. return e.is_regular_file(ec);
  435. }
  436. inline bool is_directory(directory_entry const& e)
  437. {
  438. return e.is_directory();
  439. }
  440. inline bool is_directory(directory_entry const& e, system::error_code& ec) BOOST_NOEXCEPT
  441. {
  442. return e.is_directory(ec);
  443. }
  444. inline bool is_symlink(directory_entry const& e)
  445. {
  446. return e.is_symlink();
  447. }
  448. inline bool is_symlink(directory_entry const& e, system::error_code& ec) BOOST_NOEXCEPT
  449. {
  450. return e.is_symlink(ec);
  451. }
  452. inline bool is_block_file(directory_entry const& e)
  453. {
  454. return e.is_block_file();
  455. }
  456. inline bool is_block_file(directory_entry const& e, system::error_code& ec) BOOST_NOEXCEPT
  457. {
  458. return e.is_block_file(ec);
  459. }
  460. inline bool is_character_file(directory_entry const& e)
  461. {
  462. return e.is_character_file();
  463. }
  464. inline bool is_character_file(directory_entry const& e, system::error_code& ec) BOOST_NOEXCEPT
  465. {
  466. return e.is_character_file(ec);
  467. }
  468. inline bool is_fifo(directory_entry const& e)
  469. {
  470. return e.is_fifo();
  471. }
  472. inline bool is_fifo(directory_entry const& e, system::error_code& ec) BOOST_NOEXCEPT
  473. {
  474. return e.is_fifo(ec);
  475. }
  476. inline bool is_socket(directory_entry const& e)
  477. {
  478. return e.is_socket();
  479. }
  480. inline bool is_socket(directory_entry const& e, system::error_code& ec) BOOST_NOEXCEPT
  481. {
  482. return e.is_socket(ec);
  483. }
  484. inline bool is_reparse_file(directory_entry const& e)
  485. {
  486. return e.is_reparse_file();
  487. }
  488. inline bool is_reparse_file(directory_entry const& e, system::error_code& ec) BOOST_NOEXCEPT
  489. {
  490. return e.is_reparse_file(ec);
  491. }
  492. inline bool is_other(directory_entry const& e)
  493. {
  494. return e.is_other();
  495. }
  496. inline bool is_other(directory_entry const& e, system::error_code& ec) BOOST_NOEXCEPT
  497. {
  498. return e.is_other(ec);
  499. }
  500. #ifndef BOOST_FILESYSTEM_NO_DEPRECATED
  501. BOOST_FILESYSTEM_DETAIL_DEPRECATED("Use is_regular_file() instead")
  502. inline bool is_regular(directory_entry const& e)
  503. {
  504. return filesystem::is_regular_file(e);
  505. }
  506. #endif
  507. //--------------------------------------------------------------------------------------//
  508. // //
  509. // directory_iterator helpers //
  510. // //
  511. //--------------------------------------------------------------------------------------//
  512. BOOST_SCOPED_ENUM_UT_DECLARE_BEGIN(directory_options, unsigned int)
  513. {
  514. none = 0u,
  515. skip_permission_denied = 1u, // if a directory cannot be opened because of insufficient permissions, pretend that the directory is empty
  516. follow_directory_symlink = 1u << 1u, // recursive_directory_iterator: follow directory symlinks
  517. skip_dangling_symlinks = 1u << 2u, // non-standard extension for recursive_directory_iterator: don't follow dangling directory symlinks,
  518. pop_on_error = 1u << 3u, // non-standard extension for recursive_directory_iterator: instead of producing an end iterator on errors,
  519. // repeatedly invoke pop() until it succeeds or the iterator becomes equal to end iterator
  520. _detail_no_follow = 1u << 4u, // internal use only
  521. _detail_no_push = 1u << 5u // internal use only
  522. }
  523. BOOST_SCOPED_ENUM_DECLARE_END(directory_options)
  524. BOOST_BITMASK(BOOST_SCOPED_ENUM_NATIVE(directory_options))
  525. namespace detail {
  526. struct dir_itr_imp :
  527. public boost::intrusive_ref_counter< dir_itr_imp >
  528. {
  529. #ifdef BOOST_WINDOWS_API
  530. bool close_handle;
  531. unsigned char extra_data_format;
  532. std::size_t current_offset;
  533. #endif
  534. directory_entry dir_entry;
  535. void* handle;
  536. dir_itr_imp() BOOST_NOEXCEPT :
  537. #ifdef BOOST_WINDOWS_API
  538. close_handle(false),
  539. extra_data_format(0u),
  540. current_offset(0u),
  541. #endif
  542. handle(NULL)
  543. {
  544. }
  545. BOOST_FILESYSTEM_DECL ~dir_itr_imp() BOOST_NOEXCEPT;
  546. BOOST_FILESYSTEM_DECL static void* operator new(std::size_t class_size, std::size_t extra_size) BOOST_NOEXCEPT;
  547. BOOST_FILESYSTEM_DECL static void operator delete(void* p, std::size_t extra_size) BOOST_NOEXCEPT;
  548. BOOST_FILESYSTEM_DECL static void operator delete(void* p) BOOST_NOEXCEPT;
  549. };
  550. } // namespace detail
  551. //--------------------------------------------------------------------------------------//
  552. // //
  553. // directory_iterator //
  554. // //
  555. //--------------------------------------------------------------------------------------//
  556. class directory_iterator :
  557. public boost::iterator_facade<
  558. directory_iterator,
  559. directory_entry,
  560. boost::single_pass_traversal_tag
  561. >
  562. {
  563. friend class boost::iterator_core_access;
  564. 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);
  565. friend BOOST_FILESYSTEM_DECL void detail::directory_iterator_increment(directory_iterator& it, system::error_code* ec);
  566. public:
  567. directory_iterator() BOOST_NOEXCEPT {} // creates the "end" iterator
  568. // iterator_facade derived classes don't seem to like implementations in
  569. // separate translation unit dll's, so forward to detail functions
  570. explicit directory_iterator(path const& p, BOOST_SCOPED_ENUM_NATIVE(directory_options) opts = directory_options::none)
  571. {
  572. detail::directory_iterator_construct(*this, p, static_cast< unsigned int >(opts), NULL, NULL);
  573. }
  574. directory_iterator(path const& p, system::error_code& ec) BOOST_NOEXCEPT
  575. {
  576. detail::directory_iterator_construct(*this, p, static_cast< unsigned int >(directory_options::none), NULL, &ec);
  577. }
  578. directory_iterator(path const& p, BOOST_SCOPED_ENUM_NATIVE(directory_options) opts, system::error_code& ec) BOOST_NOEXCEPT
  579. {
  580. detail::directory_iterator_construct(*this, p, static_cast< unsigned int >(opts), NULL, &ec);
  581. }
  582. BOOST_DEFAULTED_FUNCTION(directory_iterator(directory_iterator const& that), : m_imp(that.m_imp) {})
  583. BOOST_DEFAULTED_FUNCTION(directory_iterator& operator=(directory_iterator const& that), { m_imp = that.m_imp; return *this; })
  584. #if !defined(BOOST_NO_CXX11_RVALUE_REFERENCES)
  585. directory_iterator(directory_iterator&& that) BOOST_NOEXCEPT :
  586. m_imp(static_cast< boost::intrusive_ptr< detail::dir_itr_imp >&& >(that.m_imp))
  587. {
  588. }
  589. directory_iterator& operator=(directory_iterator&& that) BOOST_NOEXCEPT
  590. {
  591. m_imp = static_cast< boost::intrusive_ptr< detail::dir_itr_imp >&& >(that.m_imp);
  592. return *this;
  593. }
  594. #endif // !defined(BOOST_NO_CXX11_RVALUE_REFERENCES)
  595. directory_iterator& increment(system::error_code& ec) BOOST_NOEXCEPT
  596. {
  597. detail::directory_iterator_increment(*this, &ec);
  598. return *this;
  599. }
  600. private:
  601. boost::iterator_facade<
  602. directory_iterator,
  603. directory_entry,
  604. boost::single_pass_traversal_tag
  605. >::reference dereference() const
  606. {
  607. BOOST_ASSERT_MSG(!is_end(), "attempt to dereference end directory iterator");
  608. return m_imp->dir_entry;
  609. }
  610. void increment() { detail::directory_iterator_increment(*this, NULL); }
  611. bool equal(directory_iterator const& rhs) const BOOST_NOEXCEPT
  612. {
  613. return m_imp == rhs.m_imp || (is_end() && rhs.is_end());
  614. }
  615. bool is_end() const BOOST_NOEXCEPT
  616. {
  617. // Note: The check for handle is needed because the iterator can be copied and the copy
  618. // can be incremented to end while the original iterator still refers to the same dir_itr_imp.
  619. return !m_imp || !m_imp->handle;
  620. }
  621. private:
  622. // intrusive_ptr provides the shallow-copy semantics required for single pass iterators
  623. // (i.e. InputIterators). The end iterator is indicated by is_end().
  624. boost::intrusive_ptr< detail::dir_itr_imp > m_imp;
  625. };
  626. // enable directory_iterator C++11 range-based for statement use --------------------//
  627. // begin() and end() are only used by a range-based for statement in the context of
  628. // auto - thus the top-level const is stripped - so returning const is harmless and
  629. // emphasizes begin() is just a pass through.
  630. inline directory_iterator const& begin(directory_iterator const& iter) BOOST_NOEXCEPT
  631. {
  632. return iter;
  633. }
  634. inline directory_iterator end(directory_iterator const&) BOOST_NOEXCEPT
  635. {
  636. return directory_iterator();
  637. }
  638. // enable C++14 generic accessors for range const iterators
  639. inline directory_iterator const& cbegin(directory_iterator const& iter) BOOST_NOEXCEPT
  640. {
  641. return iter;
  642. }
  643. inline directory_iterator cend(directory_iterator const&) BOOST_NOEXCEPT
  644. {
  645. return directory_iterator();
  646. }
  647. // enable directory_iterator BOOST_FOREACH -----------------------------------------//
  648. inline directory_iterator& range_begin(directory_iterator& iter) BOOST_NOEXCEPT
  649. {
  650. return iter;
  651. }
  652. inline directory_iterator range_begin(directory_iterator const& iter) BOOST_NOEXCEPT
  653. {
  654. return iter;
  655. }
  656. inline directory_iterator range_end(directory_iterator&) BOOST_NOEXCEPT
  657. {
  658. return directory_iterator();
  659. }
  660. inline directory_iterator range_end(directory_iterator const&) BOOST_NOEXCEPT
  661. {
  662. return directory_iterator();
  663. }
  664. } // namespace filesystem
  665. // namespace boost template specializations
  666. template< typename C, typename Enabler >
  667. struct range_mutable_iterator;
  668. template<>
  669. struct range_mutable_iterator< boost::filesystem::directory_iterator, void >
  670. {
  671. typedef boost::filesystem::directory_iterator type;
  672. };
  673. template< typename C, typename Enabler >
  674. struct range_const_iterator;
  675. template<>
  676. struct range_const_iterator< boost::filesystem::directory_iterator, void >
  677. {
  678. typedef boost::filesystem::directory_iterator type;
  679. };
  680. namespace filesystem {
  681. //--------------------------------------------------------------------------------------//
  682. // //
  683. // recursive_directory_iterator helpers //
  684. // //
  685. //--------------------------------------------------------------------------------------//
  686. #if !defined(BOOST_FILESYSTEM_NO_DEPRECATED)
  687. // Deprecated enum, use directory_options instead
  688. BOOST_SCOPED_ENUM_UT_DECLARE_BEGIN(symlink_option, unsigned int)
  689. {
  690. none = static_cast< unsigned int >(directory_options::none),
  691. no_recurse = none, // don't follow directory symlinks (default behavior)
  692. recurse = static_cast< unsigned int >(directory_options::follow_directory_symlink), // follow directory symlinks
  693. _detail_no_push = static_cast< unsigned int >(directory_options::_detail_no_push) // internal use only
  694. }
  695. BOOST_SCOPED_ENUM_DECLARE_END(symlink_option)
  696. BOOST_BITMASK(BOOST_SCOPED_ENUM_NATIVE(symlink_option))
  697. #endif // BOOST_FILESYSTEM_NO_DEPRECATED
  698. class recursive_directory_iterator;
  699. namespace detail {
  700. struct recur_dir_itr_imp :
  701. public boost::intrusive_ref_counter< recur_dir_itr_imp >
  702. {
  703. typedef directory_iterator element_type;
  704. std::vector< element_type > m_stack;
  705. // directory_options values, declared as unsigned int for ABI compatibility
  706. unsigned int m_options;
  707. explicit recur_dir_itr_imp(unsigned int opts) BOOST_NOEXCEPT : m_options(opts) {}
  708. };
  709. BOOST_FILESYSTEM_DECL void recursive_directory_iterator_construct(recursive_directory_iterator& it, path const& dir_path, unsigned int opts, system::error_code* ec);
  710. BOOST_FILESYSTEM_DECL void recursive_directory_iterator_increment(recursive_directory_iterator& it, system::error_code* ec);
  711. BOOST_FILESYSTEM_DECL void recursive_directory_iterator_pop(recursive_directory_iterator& it, system::error_code* ec);
  712. } // namespace detail
  713. //--------------------------------------------------------------------------------------//
  714. // //
  715. // recursive_directory_iterator //
  716. // //
  717. //--------------------------------------------------------------------------------------//
  718. class recursive_directory_iterator :
  719. public boost::iterator_facade<
  720. recursive_directory_iterator,
  721. directory_entry,
  722. boost::single_pass_traversal_tag
  723. >
  724. {
  725. friend class boost::iterator_core_access;
  726. 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);
  727. friend BOOST_FILESYSTEM_DECL void detail::recursive_directory_iterator_increment(recursive_directory_iterator& it, system::error_code* ec);
  728. friend BOOST_FILESYSTEM_DECL void detail::recursive_directory_iterator_pop(recursive_directory_iterator& it, system::error_code* ec);
  729. public:
  730. recursive_directory_iterator() BOOST_NOEXCEPT {} // creates the "end" iterator
  731. explicit recursive_directory_iterator(path const& dir_path)
  732. {
  733. detail::recursive_directory_iterator_construct(*this, dir_path, static_cast< unsigned int >(directory_options::none), NULL);
  734. }
  735. recursive_directory_iterator(path const& dir_path, system::error_code& ec)
  736. {
  737. detail::recursive_directory_iterator_construct(*this, dir_path, static_cast< unsigned int >(directory_options::none), &ec);
  738. }
  739. recursive_directory_iterator(path const& dir_path, BOOST_SCOPED_ENUM_NATIVE(directory_options) opts)
  740. {
  741. detail::recursive_directory_iterator_construct(*this, dir_path, static_cast< unsigned int >(opts), NULL);
  742. }
  743. recursive_directory_iterator(path const& dir_path, BOOST_SCOPED_ENUM_NATIVE(directory_options) opts, system::error_code& ec)
  744. {
  745. detail::recursive_directory_iterator_construct(*this, dir_path, static_cast< unsigned int >(opts), &ec);
  746. }
  747. #if !defined(BOOST_FILESYSTEM_NO_DEPRECATED)
  748. // Deprecated constructors
  749. BOOST_FILESYSTEM_DETAIL_DEPRECATED("Use directory_options instead of symlink_option")
  750. recursive_directory_iterator(path const& dir_path, BOOST_SCOPED_ENUM_NATIVE(symlink_option) opts)
  751. {
  752. detail::recursive_directory_iterator_construct(*this, dir_path, static_cast< unsigned int >(opts), NULL);
  753. }
  754. BOOST_FILESYSTEM_DETAIL_DEPRECATED("Use directory_options instead of symlink_option")
  755. recursive_directory_iterator(path const& dir_path, BOOST_SCOPED_ENUM_NATIVE(symlink_option) opts, system::error_code& ec) BOOST_NOEXCEPT
  756. {
  757. detail::recursive_directory_iterator_construct(*this, dir_path, static_cast< unsigned int >(opts), &ec);
  758. }
  759. #endif // BOOST_FILESYSTEM_NO_DEPRECATED
  760. BOOST_DEFAULTED_FUNCTION(recursive_directory_iterator(recursive_directory_iterator const& that), : m_imp(that.m_imp) {})
  761. BOOST_DEFAULTED_FUNCTION(recursive_directory_iterator& operator=(recursive_directory_iterator const& that), { m_imp = that.m_imp; return *this; })
  762. #if !defined(BOOST_NO_CXX11_RVALUE_REFERENCES)
  763. recursive_directory_iterator(recursive_directory_iterator&& that) BOOST_NOEXCEPT :
  764. m_imp(static_cast< boost::intrusive_ptr< detail::recur_dir_itr_imp >&& >(that.m_imp))
  765. {
  766. }
  767. recursive_directory_iterator& operator=(recursive_directory_iterator&& that) BOOST_NOEXCEPT
  768. {
  769. m_imp = static_cast< boost::intrusive_ptr< detail::recur_dir_itr_imp >&& >(that.m_imp);
  770. return *this;
  771. }
  772. #endif // !defined(BOOST_NO_CXX11_RVALUE_REFERENCES)
  773. recursive_directory_iterator& increment(system::error_code& ec) BOOST_NOEXCEPT
  774. {
  775. detail::recursive_directory_iterator_increment(*this, &ec);
  776. return *this;
  777. }
  778. int depth() const BOOST_NOEXCEPT
  779. {
  780. BOOST_ASSERT_MSG(!is_end(), "depth() on end recursive_directory_iterator");
  781. return static_cast< int >(m_imp->m_stack.size() - 1u);
  782. }
  783. bool recursion_pending() const BOOST_NOEXCEPT
  784. {
  785. BOOST_ASSERT_MSG(!is_end(), "recursion_pending() on end recursive_directory_iterator");
  786. return (m_imp->m_options & static_cast< unsigned int >(directory_options::_detail_no_push)) == 0u;
  787. }
  788. #ifndef BOOST_FILESYSTEM_NO_DEPRECATED
  789. BOOST_FILESYSTEM_DETAIL_DEPRECATED("Use recursive_directory_iterator::depth() instead")
  790. int level() const BOOST_NOEXCEPT
  791. {
  792. return depth();
  793. }
  794. bool no_push_pending() const BOOST_NOEXCEPT { return !recursion_pending(); }
  795. bool no_push_request() const BOOST_NOEXCEPT { return !recursion_pending(); }
  796. #endif
  797. void pop()
  798. {
  799. detail::recursive_directory_iterator_pop(*this, NULL);
  800. }
  801. void pop(system::error_code& ec) BOOST_NOEXCEPT
  802. {
  803. detail::recursive_directory_iterator_pop(*this, &ec);
  804. }
  805. void disable_recursion_pending(bool value = true) BOOST_NOEXCEPT
  806. {
  807. BOOST_ASSERT_MSG(!is_end(), "disable_recursion_pending() on end recursive_directory_iterator");
  808. if (value)
  809. m_imp->m_options |= static_cast< unsigned int >(directory_options::_detail_no_push);
  810. else
  811. m_imp->m_options &= ~static_cast< unsigned int >(directory_options::_detail_no_push);
  812. }
  813. #ifndef BOOST_FILESYSTEM_NO_DEPRECATED
  814. BOOST_FILESYSTEM_DETAIL_DEPRECATED("Use recursive_directory_iterator::disable_recursion_pending() instead")
  815. void no_push(bool value = true) BOOST_NOEXCEPT
  816. {
  817. disable_recursion_pending(value);
  818. }
  819. #endif
  820. file_status status() const
  821. {
  822. BOOST_ASSERT_MSG(!is_end(), "status() on end recursive_directory_iterator");
  823. return m_imp->m_stack.back()->status();
  824. }
  825. file_status symlink_status() const
  826. {
  827. BOOST_ASSERT_MSG(!is_end(), "symlink_status() on end recursive_directory_iterator");
  828. return m_imp->m_stack.back()->symlink_status();
  829. }
  830. private:
  831. boost::iterator_facade<
  832. recursive_directory_iterator,
  833. directory_entry,
  834. boost::single_pass_traversal_tag
  835. >::reference dereference() const
  836. {
  837. BOOST_ASSERT_MSG(!is_end(), "dereference of end recursive_directory_iterator");
  838. return *m_imp->m_stack.back();
  839. }
  840. void increment() { detail::recursive_directory_iterator_increment(*this, NULL); }
  841. bool equal(recursive_directory_iterator const& rhs) const BOOST_NOEXCEPT
  842. {
  843. return m_imp == rhs.m_imp || (is_end() && rhs.is_end());
  844. }
  845. bool is_end() const BOOST_NOEXCEPT
  846. {
  847. // Note: The check for m_stack.empty() is needed because the iterator can be copied and the copy
  848. // can be incremented to end while the original iterator still refers to the same recur_dir_itr_imp.
  849. return !m_imp || m_imp->m_stack.empty();
  850. }
  851. private:
  852. // intrusive_ptr provides the shallow-copy semantics required for single pass iterators
  853. // (i.e. InputIterators). The end iterator is indicated by is_end().
  854. boost::intrusive_ptr< detail::recur_dir_itr_imp > m_imp;
  855. };
  856. #if !defined(BOOST_FILESYSTEM_NO_DEPRECATED)
  857. BOOST_FILESYSTEM_DETAIL_DEPRECATED("Use recursive_directory_iterator instead")
  858. typedef recursive_directory_iterator wrecursive_directory_iterator;
  859. #endif
  860. // enable recursive directory iterator C++11 range-base for statement use ----------//
  861. // begin() and end() are only used by a range-based for statement in the context of
  862. // auto - thus the top-level const is stripped - so returning const is harmless and
  863. // emphasizes begin() is just a pass through.
  864. inline recursive_directory_iterator const& begin(recursive_directory_iterator const& iter) BOOST_NOEXCEPT
  865. {
  866. return iter;
  867. }
  868. inline recursive_directory_iterator end(recursive_directory_iterator const&) BOOST_NOEXCEPT
  869. {
  870. return recursive_directory_iterator();
  871. }
  872. // enable C++14 generic accessors for range const iterators
  873. inline recursive_directory_iterator const& cbegin(recursive_directory_iterator const& iter) BOOST_NOEXCEPT
  874. {
  875. return iter;
  876. }
  877. inline recursive_directory_iterator cend(recursive_directory_iterator const&) BOOST_NOEXCEPT
  878. {
  879. return recursive_directory_iterator();
  880. }
  881. // enable recursive directory iterator BOOST_FOREACH -------------------------------//
  882. inline recursive_directory_iterator& range_begin(recursive_directory_iterator& iter) BOOST_NOEXCEPT
  883. {
  884. return iter;
  885. }
  886. inline recursive_directory_iterator range_begin(recursive_directory_iterator const& iter) BOOST_NOEXCEPT
  887. {
  888. return iter;
  889. }
  890. inline recursive_directory_iterator range_end(recursive_directory_iterator&) BOOST_NOEXCEPT
  891. {
  892. return recursive_directory_iterator();
  893. }
  894. inline recursive_directory_iterator range_end(recursive_directory_iterator const&) BOOST_NOEXCEPT
  895. {
  896. return recursive_directory_iterator();
  897. }
  898. } // namespace filesystem
  899. // namespace boost template specializations
  900. template<>
  901. struct range_mutable_iterator< boost::filesystem::recursive_directory_iterator, void >
  902. {
  903. typedef boost::filesystem::recursive_directory_iterator type;
  904. };
  905. template<>
  906. struct range_const_iterator< boost::filesystem::recursive_directory_iterator, void >
  907. {
  908. typedef boost::filesystem::recursive_directory_iterator type;
  909. };
  910. } // namespace boost
  911. #include <boost/filesystem/detail/footer.hpp>
  912. #endif // BOOST_FILESYSTEM_DIRECTORY_HPP