path.hpp 36 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964
  1. // filesystem path.hpp ---------------------------------------------------------------//
  2. // Copyright Beman Dawes 2002-2005, 2009
  3. // Copyright Vladimir Prus 2002
  4. // Distributed under the Boost Software License, Version 1.0.
  5. // See http://www.boost.org/LICENSE_1_0.txt
  6. // Library home page: http://www.boost.org/libs/filesystem
  7. // path::stem(), extension(), and replace_extension() are based on
  8. // basename(), extension(), and change_extension() from the original
  9. // filesystem/convenience.hpp header by Vladimir Prus.
  10. #ifndef BOOST_FILESYSTEM_PATH_HPP
  11. #define BOOST_FILESYSTEM_PATH_HPP
  12. #include <boost/config.hpp>
  13. # if defined( BOOST_NO_STD_WSTRING )
  14. # error Configuration not supported: Boost.Filesystem V3 and later requires std::wstring support
  15. # endif
  16. #include <boost/filesystem/config.hpp>
  17. #include <boost/filesystem/path_traits.hpp> // includes <cwchar>
  18. #include <boost/system/error_code.hpp>
  19. #include <boost/system/system_error.hpp>
  20. #include <boost/iterator/iterator_facade.hpp>
  21. #include <boost/shared_ptr.hpp>
  22. #include <boost/io/detail/quoted_manip.hpp>
  23. #include <boost/static_assert.hpp>
  24. #include <boost/functional/hash_fwd.hpp>
  25. #include <boost/type_traits/is_integral.hpp>
  26. #include <string>
  27. #include <iterator>
  28. #include <cstring>
  29. #include <iosfwd>
  30. #include <stdexcept>
  31. #include <cassert>
  32. #include <locale>
  33. #include <algorithm>
  34. #include <boost/config/abi_prefix.hpp> // must be the last #include
  35. namespace boost
  36. {
  37. namespace filesystem
  38. {
  39. //------------------------------------------------------------------------------------//
  40. // //
  41. // class path //
  42. // //
  43. //------------------------------------------------------------------------------------//
  44. class BOOST_FILESYSTEM_DECL path
  45. {
  46. public:
  47. // value_type is the character type used by the operating system API to
  48. // represent paths.
  49. # ifdef BOOST_WINDOWS_API
  50. typedef wchar_t value_type;
  51. BOOST_STATIC_CONSTEXPR value_type preferred_separator = L'\\';
  52. # else
  53. typedef char value_type;
  54. BOOST_STATIC_CONSTEXPR value_type preferred_separator = '/';
  55. # endif
  56. typedef std::basic_string<value_type> string_type;
  57. typedef std::codecvt<wchar_t, char,
  58. std::mbstate_t> codecvt_type;
  59. // ----- character encoding conversions -----
  60. // Following the principle of least astonishment, path input arguments
  61. // passed to or obtained from the operating system via objects of
  62. // class path behave as if they were directly passed to or
  63. // obtained from the O/S API, unless conversion is explicitly requested.
  64. //
  65. // POSIX specfies that path strings are passed unchanged to and from the
  66. // API. Note that this is different from the POSIX command line utilities,
  67. // which convert according to a locale.
  68. //
  69. // Thus for POSIX, char strings do not undergo conversion. wchar_t strings
  70. // are converted to/from char using the path locale or, if a conversion
  71. // argument is given, using a conversion object modeled on
  72. // std::wstring_convert.
  73. //
  74. // The path locale, which is global to the thread, can be changed by the
  75. // imbue() function. It is initialized to an implementation defined locale.
  76. //
  77. // For Windows, wchar_t strings do not undergo conversion. char strings
  78. // are converted using the "ANSI" or "OEM" code pages, as determined by
  79. // the AreFileApisANSI() function, or, if a conversion argument is given,
  80. // using a conversion object modeled on std::wstring_convert.
  81. //
  82. // See m_pathname comments for further important rationale.
  83. // TODO: rules needed for operating systems that use / or .
  84. // differently, or format directory paths differently from file paths.
  85. //
  86. // **********************************************************************************
  87. //
  88. // More work needed: How to handle an operating system that may have
  89. // slash characters or dot characters in valid filenames, either because
  90. // it doesn't follow the POSIX standard, or because it allows MBCS
  91. // filename encodings that may contain slash or dot characters. For
  92. // example, ISO/IEC 2022 (JIS) encoding which allows switching to
  93. // JIS x0208-1983 encoding. A valid filename in this set of encodings is
  94. // 0x1B 0x24 0x42 [switch to X0208-1983] 0x24 0x2F [U+304F Kiragana letter KU]
  95. // ^^^^
  96. // Note that 0x2F is the ASCII slash character
  97. //
  98. // **********************************************************************************
  99. // Supported source arguments: half-open iterator range, container, c-array,
  100. // and single pointer to null terminated string.
  101. // All source arguments except pointers to null terminated byte strings support
  102. // multi-byte character strings which may have embedded nulls. Embedded null
  103. // support is required for some Asian languages on Windows.
  104. // "const codecvt_type& cvt=codecvt()" default arguments are not used because this
  105. // limits the impact of locale("") initialization failures on POSIX systems to programs
  106. // that actually depend on locale(""). It further ensures that exceptions thrown
  107. // as a result of such failues occur after main() has started, so can be caught.
  108. // ----- constructors -----
  109. path() BOOST_NOEXCEPT {}
  110. path(const path& p) : m_pathname(p.m_pathname) {}
  111. template <class Source>
  112. path(Source const& source,
  113. typename boost::enable_if<path_traits::is_pathable<
  114. typename boost::decay<Source>::type> >::type* =0)
  115. {
  116. path_traits::dispatch(source, m_pathname);
  117. }
  118. path(const value_type* s) : m_pathname(s) {}
  119. path(value_type* s) : m_pathname(s) {}
  120. path(const string_type& s) : m_pathname(s) {}
  121. path(string_type& s) : m_pathname(s) {}
  122. // As of October 2015 the interaction between noexcept and =default is so troublesome
  123. // for VC++, GCC, and probably other compilers, that =default is not used with noexcept
  124. // functions. GCC is not even consistent for the same release on different platforms.
  125. # if !defined(BOOST_NO_CXX11_RVALUE_REFERENCES)
  126. path(path&& p) BOOST_NOEXCEPT { m_pathname = std::move(p.m_pathname); }
  127. path& operator=(path&& p) BOOST_NOEXCEPT
  128. { m_pathname = std::move(p.m_pathname); return *this; }
  129. # endif
  130. template <class Source>
  131. path(Source const& source, const codecvt_type& cvt)
  132. {
  133. path_traits::dispatch(source, m_pathname, cvt);
  134. }
  135. template <class InputIterator>
  136. path(InputIterator begin, InputIterator end)
  137. {
  138. if (begin != end)
  139. {
  140. // convert requires contiguous string, so copy
  141. std::basic_string<typename std::iterator_traits<InputIterator>::value_type>
  142. seq(begin, end);
  143. path_traits::convert(seq.c_str(), seq.c_str()+seq.size(), m_pathname);
  144. }
  145. }
  146. template <class InputIterator>
  147. path(InputIterator begin, InputIterator end, const codecvt_type& cvt)
  148. {
  149. if (begin != end)
  150. {
  151. // convert requires contiguous string, so copy
  152. std::basic_string<typename std::iterator_traits<InputIterator>::value_type>
  153. seq(begin, end);
  154. path_traits::convert(seq.c_str(), seq.c_str()+seq.size(), m_pathname, cvt);
  155. }
  156. }
  157. // ----- assignments -----
  158. path& operator=(const path& p)
  159. {
  160. m_pathname = p.m_pathname;
  161. return *this;
  162. }
  163. template <class Source>
  164. typename boost::enable_if<path_traits::is_pathable<
  165. typename boost::decay<Source>::type>, path&>::type
  166. operator=(Source const& source)
  167. {
  168. m_pathname.clear();
  169. path_traits::dispatch(source, m_pathname);
  170. return *this;
  171. }
  172. // value_type overloads
  173. path& operator=(const value_type* ptr) // required in case ptr overlaps *this
  174. {m_pathname = ptr; return *this;}
  175. path& operator=(value_type* ptr) // required in case ptr overlaps *this
  176. {m_pathname = ptr; return *this;}
  177. path& operator=(const string_type& s) {m_pathname = s; return *this;}
  178. path& operator=(string_type& s) {m_pathname = s; return *this;}
  179. path& assign(const value_type* ptr, const codecvt_type&) // required in case ptr overlaps *this
  180. {m_pathname = ptr; return *this;}
  181. template <class Source>
  182. path& assign(Source const& source, const codecvt_type& cvt)
  183. {
  184. m_pathname.clear();
  185. path_traits::dispatch(source, m_pathname, cvt);
  186. return *this;
  187. }
  188. template <class InputIterator>
  189. path& assign(InputIterator begin, InputIterator end)
  190. {
  191. m_pathname.clear();
  192. if (begin != end)
  193. {
  194. std::basic_string<typename std::iterator_traits<InputIterator>::value_type>
  195. seq(begin, end);
  196. path_traits::convert(seq.c_str(), seq.c_str()+seq.size(), m_pathname);
  197. }
  198. return *this;
  199. }
  200. template <class InputIterator>
  201. path& assign(InputIterator begin, InputIterator end, const codecvt_type& cvt)
  202. {
  203. m_pathname.clear();
  204. if (begin != end)
  205. {
  206. std::basic_string<typename std::iterator_traits<InputIterator>::value_type>
  207. seq(begin, end);
  208. path_traits::convert(seq.c_str(), seq.c_str()+seq.size(), m_pathname, cvt);
  209. }
  210. return *this;
  211. }
  212. // ----- concatenation -----
  213. template <class Source>
  214. typename boost::enable_if<path_traits::is_pathable<
  215. typename boost::decay<Source>::type>, path&>::type
  216. operator+=(Source const& source)
  217. {
  218. return concat(source);
  219. }
  220. // value_type overloads. Same rationale as for constructors above
  221. path& operator+=(const path& p) { m_pathname += p.m_pathname; return *this; }
  222. path& operator+=(const value_type* ptr) { m_pathname += ptr; return *this; }
  223. path& operator+=(value_type* ptr) { m_pathname += ptr; return *this; }
  224. path& operator+=(const string_type& s) { m_pathname += s; return *this; }
  225. path& operator+=(string_type& s) { m_pathname += s; return *this; }
  226. path& operator+=(value_type c) { m_pathname += c; return *this; }
  227. template <class CharT>
  228. typename boost::enable_if<is_integral<CharT>, path&>::type
  229. operator+=(CharT c)
  230. {
  231. CharT tmp[2];
  232. tmp[0] = c;
  233. tmp[1] = 0;
  234. return concat(tmp);
  235. }
  236. template <class Source>
  237. path& concat(Source const& source)
  238. {
  239. path_traits::dispatch(source, m_pathname);
  240. return *this;
  241. }
  242. template <class Source>
  243. path& concat(Source const& source, const codecvt_type& cvt)
  244. {
  245. path_traits::dispatch(source, m_pathname, cvt);
  246. return *this;
  247. }
  248. template <class InputIterator>
  249. path& concat(InputIterator begin, InputIterator end)
  250. {
  251. if (begin == end)
  252. return *this;
  253. std::basic_string<typename std::iterator_traits<InputIterator>::value_type>
  254. seq(begin, end);
  255. path_traits::convert(seq.c_str(), seq.c_str()+seq.size(), m_pathname);
  256. return *this;
  257. }
  258. template <class InputIterator>
  259. path& concat(InputIterator begin, InputIterator end, const codecvt_type& cvt)
  260. {
  261. if (begin == end)
  262. return *this;
  263. std::basic_string<typename std::iterator_traits<InputIterator>::value_type>
  264. seq(begin, end);
  265. path_traits::convert(seq.c_str(), seq.c_str()+seq.size(), m_pathname, cvt);
  266. return *this;
  267. }
  268. // ----- appends -----
  269. // if a separator is added, it is the preferred separator for the platform;
  270. // slash for POSIX, backslash for Windows
  271. path& operator/=(const path& p);
  272. template <class Source>
  273. typename boost::enable_if<path_traits::is_pathable<
  274. typename boost::decay<Source>::type>, path&>::type
  275. operator/=(Source const& source)
  276. {
  277. return append(source);
  278. }
  279. path& operator/=(const value_type* ptr);
  280. path& operator/=(value_type* ptr)
  281. {
  282. return this->operator/=(const_cast<const value_type*>(ptr));
  283. }
  284. path& operator/=(const string_type& s) { return this->operator/=(path(s)); }
  285. path& operator/=(string_type& s) { return this->operator/=(path(s)); }
  286. path& append(const value_type* ptr) // required in case ptr overlaps *this
  287. {
  288. this->operator/=(ptr);
  289. return *this;
  290. }
  291. path& append(const value_type* ptr, const codecvt_type&) // required in case ptr overlaps *this
  292. {
  293. this->operator/=(ptr);
  294. return *this;
  295. }
  296. template <class Source>
  297. path& append(Source const& source);
  298. template <class Source>
  299. path& append(Source const& source, const codecvt_type& cvt);
  300. template <class InputIterator>
  301. path& append(InputIterator begin, InputIterator end);
  302. template <class InputIterator>
  303. path& append(InputIterator begin, InputIterator end, const codecvt_type& cvt);
  304. // ----- modifiers -----
  305. void clear() BOOST_NOEXCEPT { m_pathname.clear(); }
  306. path& make_preferred()
  307. # ifdef BOOST_POSIX_API
  308. { return *this; } // POSIX no effect
  309. # else // BOOST_WINDOWS_API
  310. ; // change slashes to backslashes
  311. # endif
  312. path& remove_filename();
  313. path& remove_trailing_separator();
  314. path& replace_extension(const path& new_extension = path());
  315. void swap(path& rhs) BOOST_NOEXCEPT { m_pathname.swap(rhs.m_pathname); }
  316. // ----- observers -----
  317. // For operating systems that format file paths differently than directory
  318. // paths, return values from observers are formatted as file names unless there
  319. // is a trailing separator, in which case returns are formatted as directory
  320. // paths. POSIX and Windows make no such distinction.
  321. // Implementations are permitted to return const values or const references.
  322. // The string or path returned by an observer are specified as being formatted
  323. // as "native" or "generic".
  324. //
  325. // For POSIX, these are all the same format; slashes and backslashes are as input and
  326. // are not modified.
  327. //
  328. // For Windows, native: as input; slashes and backslashes are not modified;
  329. // this is the format of the internally stored string.
  330. // generic: backslashes are converted to slashes
  331. // ----- native format observers -----
  332. const string_type& native() const BOOST_NOEXCEPT { return m_pathname; }
  333. const value_type* c_str() const BOOST_NOEXCEPT { return m_pathname.c_str(); }
  334. string_type::size_type size() const BOOST_NOEXCEPT { return m_pathname.size(); }
  335. template <class String>
  336. String string() const;
  337. template <class String>
  338. String string(const codecvt_type& cvt) const;
  339. # ifdef BOOST_WINDOWS_API
  340. const std::string string() const
  341. {
  342. std::string tmp;
  343. if (!m_pathname.empty())
  344. path_traits::convert(&*m_pathname.begin(), &*m_pathname.begin()+m_pathname.size(),
  345. tmp);
  346. return tmp;
  347. }
  348. const std::string string(const codecvt_type& cvt) const
  349. {
  350. std::string tmp;
  351. if (!m_pathname.empty())
  352. path_traits::convert(&*m_pathname.begin(), &*m_pathname.begin()+m_pathname.size(),
  353. tmp, cvt);
  354. return tmp;
  355. }
  356. // string_type is std::wstring, so there is no conversion
  357. const std::wstring& wstring() const { return m_pathname; }
  358. const std::wstring& wstring(const codecvt_type&) const { return m_pathname; }
  359. # else // BOOST_POSIX_API
  360. // string_type is std::string, so there is no conversion
  361. const std::string& string() const { return m_pathname; }
  362. const std::string& string(const codecvt_type&) const { return m_pathname; }
  363. const std::wstring wstring() const
  364. {
  365. std::wstring tmp;
  366. if (!m_pathname.empty())
  367. path_traits::convert(&*m_pathname.begin(), &*m_pathname.begin()+m_pathname.size(),
  368. tmp);
  369. return tmp;
  370. }
  371. const std::wstring wstring(const codecvt_type& cvt) const
  372. {
  373. std::wstring tmp;
  374. if (!m_pathname.empty())
  375. path_traits::convert(&*m_pathname.begin(), &*m_pathname.begin()+m_pathname.size(),
  376. tmp, cvt);
  377. return tmp;
  378. }
  379. # endif
  380. // ----- generic format observers -----
  381. // Experimental generic function returning generic formatted path (i.e. separators
  382. // are forward slashes). Motivation: simpler than a family of generic_*string
  383. // functions.
  384. path generic() const
  385. {
  386. # ifdef BOOST_WINDOWS_API
  387. path tmp;
  388. std::replace_copy(m_pathname.begin(), m_pathname.end(),
  389. std::back_inserter(tmp.m_pathname), L'\\', L'/');
  390. return tmp;
  391. # else
  392. return path(*this);
  393. # endif
  394. }
  395. template <class String>
  396. String generic_string() const;
  397. template <class String>
  398. String generic_string(const codecvt_type& cvt) const;
  399. # ifdef BOOST_WINDOWS_API
  400. const std::string generic_string() const;
  401. const std::string generic_string(const codecvt_type& cvt) const;
  402. const std::wstring generic_wstring() const;
  403. const std::wstring generic_wstring(const codecvt_type&) const { return generic_wstring(); };
  404. # else // BOOST_POSIX_API
  405. // On POSIX-like systems, the generic format is the same as the native format
  406. const std::string& generic_string() const { return m_pathname; }
  407. const std::string& generic_string(const codecvt_type&) const { return m_pathname; }
  408. const std::wstring generic_wstring() const { return wstring(); }
  409. const std::wstring generic_wstring(const codecvt_type& cvt) const { return wstring(cvt); }
  410. # endif
  411. // ----- compare -----
  412. int compare(const path& p) const BOOST_NOEXCEPT; // generic, lexicographical
  413. int compare(const std::string& s) const { return compare(path(s)); }
  414. int compare(const value_type* s) const { return compare(path(s)); }
  415. // ----- decomposition -----
  416. path root_path() const;
  417. path root_name() const; // returns 0 or 1 element path
  418. // even on POSIX, root_name() is non-empty() for network paths
  419. path root_directory() const; // returns 0 or 1 element path
  420. path relative_path() const;
  421. path parent_path() const;
  422. path filename() const; // returns 0 or 1 element path
  423. path stem() const; // returns 0 or 1 element path
  424. path extension() const; // returns 0 or 1 element path
  425. // ----- query -----
  426. bool empty() const BOOST_NOEXCEPT{ return m_pathname.empty(); }
  427. bool has_root_path() const { return has_root_directory() || has_root_name(); }
  428. bool has_root_name() const { return !root_name().empty(); }
  429. bool has_root_directory() const { return !root_directory().empty(); }
  430. bool has_relative_path() const { return !relative_path().empty(); }
  431. bool has_parent_path() const { return !parent_path().empty(); }
  432. bool has_filename() const { return !m_pathname.empty(); }
  433. bool has_stem() const { return !stem().empty(); }
  434. bool has_extension() const { return !extension().empty(); }
  435. bool is_relative() const { return !is_absolute(); }
  436. bool is_absolute() const
  437. {
  438. # ifdef BOOST_WINDOWS_API
  439. return has_root_name() && has_root_directory();
  440. # else
  441. return has_root_directory();
  442. # endif
  443. }
  444. // ----- lexical operations -----
  445. path lexically_normal() const;
  446. path lexically_relative(const path& base) const;
  447. path lexically_proximate(const path& base) const
  448. {
  449. path tmp(lexically_relative(base));
  450. return tmp.empty() ? *this : tmp;
  451. }
  452. // ----- iterators -----
  453. class iterator;
  454. typedef iterator const_iterator;
  455. class reverse_iterator;
  456. typedef reverse_iterator const_reverse_iterator;
  457. iterator begin() const;
  458. iterator end() const;
  459. reverse_iterator rbegin() const;
  460. reverse_iterator rend() const;
  461. // ----- static member functions -----
  462. static std::locale imbue(const std::locale& loc);
  463. static const codecvt_type& codecvt();
  464. // ----- deprecated functions -----
  465. # if defined(BOOST_FILESYSTEM_DEPRECATED) && defined(BOOST_FILESYSTEM_NO_DEPRECATED)
  466. # error both BOOST_FILESYSTEM_DEPRECATED and BOOST_FILESYSTEM_NO_DEPRECATED are defined
  467. # endif
  468. # if !defined(BOOST_FILESYSTEM_NO_DEPRECATED)
  469. // recently deprecated functions supplied by default
  470. path& normalize() {
  471. path tmp(lexically_normal());
  472. m_pathname.swap(tmp.m_pathname);
  473. return *this;
  474. }
  475. path& remove_leaf() { return remove_filename(); }
  476. path leaf() const { return filename(); }
  477. path branch_path() const { return parent_path(); }
  478. bool has_leaf() const { return !m_pathname.empty(); }
  479. bool has_branch_path() const { return !parent_path().empty(); }
  480. bool is_complete() const { return is_absolute(); }
  481. # endif
  482. # if defined(BOOST_FILESYSTEM_DEPRECATED)
  483. // deprecated functions with enough signature or semantic changes that they are
  484. // not supplied by default
  485. const std::string file_string() const { return string(); }
  486. const std::string directory_string() const { return string(); }
  487. const std::string native_file_string() const { return string(); }
  488. const std::string native_directory_string() const { return string(); }
  489. const string_type external_file_string() const { return native(); }
  490. const string_type external_directory_string() const { return native(); }
  491. // older functions no longer supported
  492. //typedef bool (*name_check)(const std::string & name);
  493. //basic_path(const string_type& str, name_check) { operator/=(str); }
  494. //basic_path(const typename string_type::value_type* s, name_check)
  495. // { operator/=(s);}
  496. //static bool default_name_check_writable() { return false; }
  497. //static void default_name_check(name_check) {}
  498. //static name_check default_name_check() { return 0; }
  499. //basic_path& canonize();
  500. # endif
  501. //--------------------------------------------------------------------------------------//
  502. // class path private members //
  503. //--------------------------------------------------------------------------------------//
  504. private:
  505. # if defined(_MSC_VER)
  506. # pragma warning(push) // Save warning settings
  507. # pragma warning(disable : 4251) // disable warning: class 'std::basic_string<_Elem,_Traits,_Ax>'
  508. # endif // needs to have dll-interface...
  509. /*
  510. m_pathname has the type, encoding, and format required by the native
  511. operating system. Thus for POSIX and Windows there is no conversion for
  512. passing m_pathname.c_str() to the O/S API or when obtaining a path from the
  513. O/S API. POSIX encoding is unspecified other than for dot and slash
  514. characters; POSIX just treats paths as a sequence of bytes. Windows
  515. encoding is UCS-2 or UTF-16 depending on the version.
  516. */
  517. string_type m_pathname; // Windows: as input; backslashes NOT converted to slashes,
  518. // slashes NOT converted to backslashes
  519. # if defined(_MSC_VER)
  520. # pragma warning(pop) // restore warning settings.
  521. # endif
  522. string_type::size_type m_append_separator_if_needed();
  523. // Returns: If separator is to be appended, m_pathname.size() before append. Otherwise 0.
  524. // Note: An append is never performed if size()==0, so a returned 0 is unambiguous.
  525. void m_erase_redundant_separator(string_type::size_type sep_pos);
  526. string_type::size_type m_parent_path_end() const;
  527. path& m_normalize();
  528. // Was qualified; como433beta8 reports:
  529. // warning #427-D: qualified name is not allowed in member declaration
  530. friend class iterator;
  531. friend bool operator<(const path& lhs, const path& rhs);
  532. // see path::iterator::increment/decrement comment below
  533. static void m_path_iterator_increment(path::iterator & it);
  534. static void m_path_iterator_decrement(path::iterator & it);
  535. }; // class path
  536. namespace detail
  537. {
  538. BOOST_FILESYSTEM_DECL
  539. int lex_compare(path::iterator first1, path::iterator last1,
  540. path::iterator first2, path::iterator last2);
  541. BOOST_FILESYSTEM_DECL
  542. const path& dot_path();
  543. BOOST_FILESYSTEM_DECL
  544. const path& dot_dot_path();
  545. }
  546. # ifndef BOOST_FILESYSTEM_NO_DEPRECATED
  547. typedef path wpath;
  548. # endif
  549. //------------------------------------------------------------------------------------//
  550. // class path::iterator //
  551. //------------------------------------------------------------------------------------//
  552. class path::iterator
  553. : public boost::iterator_facade<
  554. path::iterator,
  555. path const,
  556. boost::bidirectional_traversal_tag >
  557. {
  558. private:
  559. friend class boost::iterator_core_access;
  560. friend class boost::filesystem::path;
  561. friend class boost::filesystem::path::reverse_iterator;
  562. friend void m_path_iterator_increment(path::iterator & it);
  563. friend void m_path_iterator_decrement(path::iterator & it);
  564. const path& dereference() const { return m_element; }
  565. bool equal(const iterator & rhs) const
  566. {
  567. return m_path_ptr == rhs.m_path_ptr && m_pos == rhs.m_pos;
  568. }
  569. // iterator_facade derived classes don't seem to like implementations in
  570. // separate translation unit dll's, so forward to class path static members
  571. void increment() { m_path_iterator_increment(*this); }
  572. void decrement() { m_path_iterator_decrement(*this); }
  573. path m_element; // current element
  574. const path* m_path_ptr; // path being iterated over
  575. string_type::size_type m_pos; // position of m_element in
  576. // m_path_ptr->m_pathname.
  577. // if m_element is implicit dot, m_pos is the
  578. // position of the last separator in the path.
  579. // end() iterator is indicated by
  580. // m_pos == m_path_ptr->m_pathname.size()
  581. }; // path::iterator
  582. //------------------------------------------------------------------------------------//
  583. // class path::reverse_iterator //
  584. //------------------------------------------------------------------------------------//
  585. class path::reverse_iterator
  586. : public boost::iterator_facade<
  587. path::reverse_iterator,
  588. path const,
  589. boost::bidirectional_traversal_tag >
  590. {
  591. public:
  592. explicit reverse_iterator(iterator itr) : m_itr(itr)
  593. {
  594. if (itr != itr.m_path_ptr->begin())
  595. m_element = *--itr;
  596. }
  597. private:
  598. friend class boost::iterator_core_access;
  599. friend class boost::filesystem::path;
  600. const path& dereference() const { return m_element; }
  601. bool equal(const reverse_iterator& rhs) const { return m_itr == rhs.m_itr; }
  602. void increment()
  603. {
  604. --m_itr;
  605. if (m_itr != m_itr.m_path_ptr->begin())
  606. {
  607. iterator tmp = m_itr;
  608. m_element = *--tmp;
  609. }
  610. }
  611. void decrement()
  612. {
  613. m_element = *m_itr;
  614. ++m_itr;
  615. }
  616. iterator m_itr;
  617. path m_element;
  618. }; // path::reverse_iterator
  619. inline path::reverse_iterator path::rbegin() const { return reverse_iterator(end()); }
  620. inline path::reverse_iterator path::rend() const { return reverse_iterator(begin()); }
  621. //------------------------------------------------------------------------------------//
  622. // //
  623. // non-member functions //
  624. // //
  625. //------------------------------------------------------------------------------------//
  626. // std::lexicographical_compare would infinately recurse because path iterators
  627. // yield paths, so provide a path aware version
  628. inline bool lexicographical_compare(path::iterator first1, path::iterator last1,
  629. path::iterator first2, path::iterator last2)
  630. { return detail::lex_compare(first1, last1, first2, last2) < 0; }
  631. inline bool operator==(const path& lhs, const path& rhs) {return lhs.compare(rhs) == 0;}
  632. inline bool operator==(const path& lhs, const path::string_type& rhs) {return lhs.compare(rhs) == 0;}
  633. inline bool operator==(const path::string_type& lhs, const path& rhs) {return rhs.compare(lhs) == 0;}
  634. inline bool operator==(const path& lhs, const path::value_type* rhs) {return lhs.compare(rhs) == 0;}
  635. inline bool operator==(const path::value_type* lhs, const path& rhs) {return rhs.compare(lhs) == 0;}
  636. inline bool operator!=(const path& lhs, const path& rhs) {return lhs.compare(rhs) != 0;}
  637. inline bool operator!=(const path& lhs, const path::string_type& rhs) {return lhs.compare(rhs) != 0;}
  638. inline bool operator!=(const path::string_type& lhs, const path& rhs) {return rhs.compare(lhs) != 0;}
  639. inline bool operator!=(const path& lhs, const path::value_type* rhs) {return lhs.compare(rhs) != 0;}
  640. inline bool operator!=(const path::value_type* lhs, const path& rhs) {return rhs.compare(lhs) != 0;}
  641. // TODO: why do == and != have additional overloads, but the others don't?
  642. inline bool operator<(const path& lhs, const path& rhs) {return lhs.compare(rhs) < 0;}
  643. inline bool operator<=(const path& lhs, const path& rhs) {return !(rhs < lhs);}
  644. inline bool operator> (const path& lhs, const path& rhs) {return rhs < lhs;}
  645. inline bool operator>=(const path& lhs, const path& rhs) {return !(lhs < rhs);}
  646. inline std::size_t hash_value(const path& x)
  647. {
  648. # ifdef BOOST_WINDOWS_API
  649. std::size_t seed = 0;
  650. for(const path::value_type* it = x.c_str(); *it; ++it)
  651. hash_combine(seed, *it == '/' ? L'\\' : *it);
  652. return seed;
  653. # else // BOOST_POSIX_API
  654. return hash_range(x.native().begin(), x.native().end());
  655. # endif
  656. }
  657. inline void swap(path& lhs, path& rhs) { lhs.swap(rhs); }
  658. inline path operator/(const path& lhs, const path& rhs) { return path(lhs) /= rhs; }
  659. // inserters and extractors
  660. // use boost::io::quoted() to handle spaces in paths
  661. // use '&' as escape character to ease use for Windows paths
  662. template <class Char, class Traits>
  663. inline std::basic_ostream<Char, Traits>&
  664. operator<<(std::basic_ostream<Char, Traits>& os, const path& p)
  665. {
  666. return os
  667. << boost::io::quoted(p.template string<std::basic_string<Char> >(), static_cast<Char>('&'));
  668. }
  669. template <class Char, class Traits>
  670. inline std::basic_istream<Char, Traits>&
  671. operator>>(std::basic_istream<Char, Traits>& is, path& p)
  672. {
  673. std::basic_string<Char> str;
  674. is >> boost::io::quoted(str, static_cast<Char>('&'));
  675. p = str;
  676. return is;
  677. }
  678. // name_checks
  679. // These functions are holdovers from version 1. It isn't clear they have much
  680. // usefulness, or how to generalize them for later versions.
  681. BOOST_FILESYSTEM_DECL bool portable_posix_name(const std::string & name);
  682. BOOST_FILESYSTEM_DECL bool windows_name(const std::string & name);
  683. BOOST_FILESYSTEM_DECL bool portable_name(const std::string & name);
  684. BOOST_FILESYSTEM_DECL bool portable_directory_name(const std::string & name);
  685. BOOST_FILESYSTEM_DECL bool portable_file_name(const std::string & name);
  686. BOOST_FILESYSTEM_DECL bool native(const std::string & name);
  687. //--------------------------------------------------------------------------------------//
  688. // class path member template implementation //
  689. //--------------------------------------------------------------------------------------//
  690. template <class InputIterator>
  691. path& path::append(InputIterator begin, InputIterator end)
  692. {
  693. if (begin == end)
  694. return *this;
  695. string_type::size_type sep_pos(m_append_separator_if_needed());
  696. std::basic_string<typename std::iterator_traits<InputIterator>::value_type>
  697. seq(begin, end);
  698. path_traits::convert(seq.c_str(), seq.c_str()+seq.size(), m_pathname);
  699. if (sep_pos)
  700. m_erase_redundant_separator(sep_pos);
  701. return *this;
  702. }
  703. template <class InputIterator>
  704. path& path::append(InputIterator begin, InputIterator end, const codecvt_type& cvt)
  705. {
  706. if (begin == end)
  707. return *this;
  708. string_type::size_type sep_pos(m_append_separator_if_needed());
  709. std::basic_string<typename std::iterator_traits<InputIterator>::value_type>
  710. seq(begin, end);
  711. path_traits::convert(seq.c_str(), seq.c_str()+seq.size(), m_pathname, cvt);
  712. if (sep_pos)
  713. m_erase_redundant_separator(sep_pos);
  714. return *this;
  715. }
  716. template <class Source>
  717. path& path::append(Source const& source)
  718. {
  719. if (path_traits::empty(source))
  720. return *this;
  721. string_type::size_type sep_pos(m_append_separator_if_needed());
  722. path_traits::dispatch(source, m_pathname);
  723. if (sep_pos)
  724. m_erase_redundant_separator(sep_pos);
  725. return *this;
  726. }
  727. template <class Source>
  728. path& path::append(Source const& source, const codecvt_type& cvt)
  729. {
  730. if (path_traits::empty(source))
  731. return *this;
  732. string_type::size_type sep_pos(m_append_separator_if_needed());
  733. path_traits::dispatch(source, m_pathname, cvt);
  734. if (sep_pos)
  735. m_erase_redundant_separator(sep_pos);
  736. return *this;
  737. }
  738. //--------------------------------------------------------------------------------------//
  739. // class path member template specializations //
  740. //--------------------------------------------------------------------------------------//
  741. template <> inline
  742. std::string path::string<std::string>() const
  743. { return string(); }
  744. template <> inline
  745. std::wstring path::string<std::wstring>() const
  746. { return wstring(); }
  747. template <> inline
  748. std::string path::string<std::string>(const codecvt_type& cvt) const
  749. { return string(cvt); }
  750. template <> inline
  751. std::wstring path::string<std::wstring>(const codecvt_type& cvt) const
  752. { return wstring(cvt); }
  753. template <> inline
  754. std::string path::generic_string<std::string>() const
  755. { return generic_string(); }
  756. template <> inline
  757. std::wstring path::generic_string<std::wstring>() const
  758. { return generic_wstring(); }
  759. template <> inline
  760. std::string path::generic_string<std::string>(const codecvt_type& cvt) const
  761. { return generic_string(cvt); }
  762. template <> inline
  763. std::wstring path::generic_string<std::wstring>(const codecvt_type& cvt) const
  764. { return generic_wstring(cvt); }
  765. //--------------------------------------------------------------------------------------//
  766. // path_traits convert function implementations //
  767. // requiring path::codecvt() be visable //
  768. //--------------------------------------------------------------------------------------//
  769. namespace path_traits
  770. { // without codecvt
  771. inline
  772. void convert(const char* from,
  773. const char* from_end, // 0 for null terminated MBCS
  774. std::wstring & to)
  775. {
  776. convert(from, from_end, to, path::codecvt());
  777. }
  778. inline
  779. void convert(const wchar_t* from,
  780. const wchar_t* from_end, // 0 for null terminated MBCS
  781. std::string & to)
  782. {
  783. convert(from, from_end, to, path::codecvt());
  784. }
  785. inline
  786. void convert(const char* from,
  787. std::wstring & to)
  788. {
  789. BOOST_ASSERT(from);
  790. convert(from, 0, to, path::codecvt());
  791. }
  792. inline
  793. void convert(const wchar_t* from,
  794. std::string & to)
  795. {
  796. BOOST_ASSERT(from);
  797. convert(from, 0, to, path::codecvt());
  798. }
  799. } // namespace path_traits
  800. } // namespace filesystem
  801. } // namespace boost
  802. //----------------------------------------------------------------------------//
  803. #include <boost/config/abi_suffix.hpp> // pops abi_prefix.hpp pragmas
  804. #endif // BOOST_FILESYSTEM_PATH_HPP