[29] | 1 | // boost/filesystem/path.hpp -----------------------------------------------// |
---|
| 2 | |
---|
| 3 | // Copyright Beman Dawes 2002-2005 |
---|
| 4 | // Use, modification, and distribution is subject to the Boost Software |
---|
| 5 | // License, Version 1.0. (See accompanying file LICENSE_1_0.txt or copy at |
---|
| 6 | // http://www.boost.org/LICENSE_1_0.txt) |
---|
| 7 | |
---|
| 8 | // See library home page at http://www.boost.org/libs/filesystem |
---|
| 9 | |
---|
| 10 | //----------------------------------------------------------------------------// |
---|
| 11 | |
---|
| 12 | #ifndef BOOST_FILESYSTEM_PATH_HPP |
---|
| 13 | #define BOOST_FILESYSTEM_PATH_HPP |
---|
| 14 | |
---|
| 15 | #include <boost/filesystem/config.hpp> |
---|
| 16 | #include <boost/iterator/iterator_facade.hpp> |
---|
| 17 | #include <boost/throw_exception.hpp> |
---|
| 18 | #include <boost/shared_ptr.hpp> |
---|
| 19 | #include <boost/type_traits/is_same.hpp> |
---|
| 20 | #include <boost/static_assert.hpp> |
---|
| 21 | |
---|
| 22 | #include <string> |
---|
| 23 | #include <algorithm> // for lexicographical_compare |
---|
| 24 | #include <iosfwd> // needed by basic_path inserter and extractor |
---|
| 25 | #include <stdexcept> |
---|
| 26 | #include <cassert> |
---|
| 27 | |
---|
| 28 | # ifndef BOOST_FILESYSTEM_NARROW_ONLY |
---|
| 29 | # include <locale> |
---|
| 30 | # endif |
---|
| 31 | |
---|
| 32 | #include <boost/config/abi_prefix.hpp> // must be the last #include |
---|
| 33 | |
---|
| 34 | //----------------------------------------------------------------------------// |
---|
| 35 | |
---|
| 36 | namespace boost |
---|
| 37 | { |
---|
| 38 | namespace BOOST_FILESYSTEM_NAMESPACE |
---|
| 39 | { |
---|
| 40 | template<class String, class Traits> class basic_path; |
---|
| 41 | |
---|
| 42 | struct path_traits; |
---|
| 43 | typedef basic_path< std::string, path_traits > path; |
---|
| 44 | |
---|
| 45 | struct path_traits |
---|
| 46 | { |
---|
| 47 | typedef std::string internal_string_type; |
---|
| 48 | typedef std::string external_string_type; |
---|
| 49 | static external_string_type to_external( const path &, |
---|
| 50 | const internal_string_type & src ) { return src; } |
---|
| 51 | static internal_string_type to_internal( |
---|
| 52 | const external_string_type & src ) { return src; } |
---|
| 53 | }; |
---|
| 54 | |
---|
| 55 | # ifndef BOOST_FILESYSTEM_NARROW_ONLY |
---|
| 56 | |
---|
| 57 | struct wpath_traits; |
---|
| 58 | |
---|
| 59 | typedef basic_path< std::wstring, wpath_traits > wpath; |
---|
| 60 | |
---|
| 61 | struct wpath_traits |
---|
| 62 | { |
---|
| 63 | typedef std::wstring internal_string_type; |
---|
| 64 | # ifdef BOOST_WINDOWS_API |
---|
| 65 | typedef std::wstring external_string_type; |
---|
| 66 | static external_string_type to_external( const wpath &, |
---|
| 67 | const internal_string_type & src ) { return src; } |
---|
| 68 | static internal_string_type to_internal( |
---|
| 69 | const external_string_type & src ) { return src; } |
---|
| 70 | # else |
---|
| 71 | typedef std::string external_string_type; |
---|
| 72 | static external_string_type to_external( const wpath & ph, |
---|
| 73 | const internal_string_type & src ); |
---|
| 74 | static internal_string_type to_internal( |
---|
| 75 | const external_string_type & src ); |
---|
| 76 | # endif |
---|
| 77 | static void imbue( const std::locale & loc ); |
---|
| 78 | static bool imbue( const std::locale & loc, const std::nothrow_t & ); |
---|
| 79 | }; |
---|
| 80 | |
---|
| 81 | # endif // ifndef BOOST_FILESYSTEM_NARROW_ONLY |
---|
| 82 | |
---|
| 83 | // error reporting support -------------------------------------------------// |
---|
| 84 | |
---|
| 85 | typedef int errno_type; // determined by C standard |
---|
| 86 | |
---|
| 87 | # ifdef BOOST_WINDOWS_API |
---|
| 88 | typedef unsigned system_error_type; |
---|
| 89 | |
---|
| 90 | BOOST_FILESYSTEM_DECL |
---|
| 91 | errno_type lookup_errno( system_error_type sys_err_code ); |
---|
| 92 | # else |
---|
| 93 | typedef int system_error_type; |
---|
| 94 | |
---|
| 95 | inline errno_type lookup_errno( system_error_type sys_err_code ) |
---|
| 96 | { return sys_err_code; } |
---|
| 97 | # endif |
---|
| 98 | |
---|
| 99 | // deprecated support for legacy function name |
---|
| 100 | inline errno_type lookup_error_code( system_error_type sys_err_code ) |
---|
| 101 | { return lookup_errno( sys_err_code ); } |
---|
| 102 | |
---|
| 103 | BOOST_FILESYSTEM_DECL |
---|
| 104 | void system_message( system_error_type sys_err_code, std::string & target ); |
---|
| 105 | // Effects: appends error message to target |
---|
| 106 | |
---|
| 107 | # if defined(BOOST_WINDOWS_API) && !defined(BOOST_FILESYSTEM_NARROW_ONLY) |
---|
| 108 | BOOST_FILESYSTEM_DECL void |
---|
| 109 | system_message( system_error_type sys_err_code, std::wstring & target ); |
---|
| 110 | # endif |
---|
| 111 | |
---|
| 112 | // filesystem_error ----------------------------------------------------// |
---|
| 113 | |
---|
| 114 | class filesystem_error : public std::runtime_error |
---|
| 115 | // see http://www.boost.org/more/error_handling.html for design rationale |
---|
| 116 | { |
---|
| 117 | public: |
---|
| 118 | filesystem_error() |
---|
| 119 | : std::runtime_error("filesystem error"), m_sys_err(0) {} |
---|
| 120 | explicit filesystem_error( |
---|
| 121 | const std::string & what_arg, system_error_type sys_ec = 0 ) |
---|
| 122 | : std::runtime_error(what_arg), m_sys_err(sys_ec) {} |
---|
| 123 | |
---|
| 124 | system_error_type system_error() const { return m_sys_err; } |
---|
| 125 | // Note: system_error() == 0 implies a library (rather than system) error |
---|
| 126 | |
---|
| 127 | private: |
---|
| 128 | system_error_type m_sys_err; |
---|
| 129 | }; |
---|
| 130 | |
---|
| 131 | // basic_filesystem_error ----------------------------------------------// |
---|
| 132 | |
---|
| 133 | template<class Path> |
---|
| 134 | class basic_filesystem_error : public filesystem_error |
---|
| 135 | { |
---|
| 136 | // see http://www.boost.org/more/error_handling.html for design rationale |
---|
| 137 | public: |
---|
| 138 | // compiler generates copy constructor and copy assignment |
---|
| 139 | |
---|
| 140 | typedef Path path_type; |
---|
| 141 | |
---|
| 142 | basic_filesystem_error( const std::string & what, |
---|
| 143 | system_error_type sys_err_code ); |
---|
| 144 | |
---|
| 145 | basic_filesystem_error( const std::string & what, |
---|
| 146 | const path_type & path1, system_error_type sys_err_code ); |
---|
| 147 | |
---|
| 148 | basic_filesystem_error( const std::string & what, const path_type & path1, |
---|
| 149 | const path_type & path2, system_error_type sys_err_code ); |
---|
| 150 | |
---|
| 151 | ~basic_filesystem_error() throw() {} |
---|
| 152 | |
---|
| 153 | const path_type & path1() const |
---|
| 154 | { |
---|
| 155 | static const path_type empty_path; |
---|
| 156 | return m_imp_ptr.get() ? m_imp_ptr->m_path1 : empty_path ; |
---|
| 157 | } |
---|
| 158 | const path_type & path2() const |
---|
| 159 | { |
---|
| 160 | static const path_type empty_path; |
---|
| 161 | return m_imp_ptr.get() ? m_imp_ptr->m_path2 : empty_path ; |
---|
| 162 | } |
---|
| 163 | |
---|
| 164 | private: |
---|
| 165 | struct m_imp |
---|
| 166 | { |
---|
| 167 | path_type m_path1; // may be empty() |
---|
| 168 | path_type m_path2; // may be empty() |
---|
| 169 | }; |
---|
| 170 | boost::shared_ptr<m_imp> m_imp_ptr; |
---|
| 171 | }; |
---|
| 172 | |
---|
| 173 | typedef basic_filesystem_error<path> filesystem_path_error; |
---|
| 174 | |
---|
| 175 | # ifndef BOOST_FILESYSTEM_NARROW_ONLY |
---|
| 176 | typedef basic_filesystem_error<wpath> filesystem_wpath_error; |
---|
| 177 | # endif |
---|
| 178 | |
---|
| 179 | // path traits ---------------------------------------------------------// |
---|
| 180 | |
---|
| 181 | template<class Path> struct is_basic_path |
---|
| 182 | { BOOST_STATIC_CONSTANT( bool, value = false ); }; |
---|
| 183 | template<> struct is_basic_path<path> |
---|
| 184 | { BOOST_STATIC_CONSTANT( bool, value = true ); }; |
---|
| 185 | # ifndef BOOST_FILESYSTEM_NARROW_ONLY |
---|
| 186 | template<> struct is_basic_path<wpath> |
---|
| 187 | { BOOST_STATIC_CONSTANT( bool, value = true ); }; |
---|
| 188 | # endif |
---|
| 189 | |
---|
| 190 | // these only have to be specialized if Path::string_type::value_type |
---|
| 191 | // is not convertible from char |
---|
| 192 | template<class Path> struct slash |
---|
| 193 | { BOOST_STATIC_CONSTANT( char, value = '/' ); }; |
---|
| 194 | |
---|
| 195 | template<class Path> struct dot |
---|
| 196 | { BOOST_STATIC_CONSTANT( char, value = '.' ); }; |
---|
| 197 | |
---|
| 198 | template<class Path> struct colon |
---|
| 199 | { BOOST_STATIC_CONSTANT( char, value = ':' ); }; |
---|
| 200 | |
---|
| 201 | # ifdef BOOST_WINDOWS_PATH |
---|
| 202 | template<class Path> struct path_alt_separator |
---|
| 203 | { BOOST_STATIC_CONSTANT( char, value = '\\' ); }; |
---|
| 204 | # endif |
---|
| 205 | |
---|
| 206 | // workaround for VC++ 7.0 and earlier issues with nested classes |
---|
| 207 | namespace detail |
---|
| 208 | { |
---|
| 209 | template<class Path> |
---|
| 210 | class iterator_helper |
---|
| 211 | { |
---|
| 212 | public: |
---|
| 213 | typedef typename Path::iterator iterator; |
---|
| 214 | static void do_increment( iterator & ph ); |
---|
| 215 | static void do_decrement( iterator & ph ); |
---|
| 216 | }; |
---|
| 217 | } |
---|
| 218 | |
---|
| 219 | // basic_path ----------------------------------------------------------// |
---|
| 220 | |
---|
| 221 | template<class String, class Traits> |
---|
| 222 | class basic_path |
---|
| 223 | { |
---|
| 224 | // invariant: m_path valid according to the portable generic path grammar |
---|
| 225 | |
---|
| 226 | // validate template arguments |
---|
| 227 | // TODO: get these working |
---|
| 228 | // BOOST_STATIC_ASSERT( ::boost::is_same<String,typename Traits::internal_string_type>::value ); |
---|
| 229 | // BOOST_STATIC_ASSERT( ::boost::is_same<typename Traits::external_string_type,std::string>::value || ::boost::is_same<typename Traits::external_string_type,std::wstring>::value ); |
---|
| 230 | |
---|
| 231 | public: |
---|
| 232 | // compiler generates copy constructor and copy assignment |
---|
| 233 | |
---|
| 234 | typedef basic_path<String, Traits> path_type; |
---|
| 235 | typedef String string_type; |
---|
| 236 | typedef typename String::value_type value_type; |
---|
| 237 | typedef Traits traits_type; |
---|
| 238 | typedef typename Traits::external_string_type external_string_type; |
---|
| 239 | |
---|
| 240 | // constructors/destructor |
---|
| 241 | basic_path() {} |
---|
| 242 | basic_path( const string_type & s ) { operator/=( s ); } |
---|
| 243 | basic_path( const value_type * s ) { operator/=( s ); } |
---|
| 244 | # ifndef BOOST_NO_MEMBER_TEMPLATES |
---|
| 245 | template <class InputIterator> |
---|
| 246 | basic_path( InputIterator first, InputIterator last ) |
---|
| 247 | { append( first, last ); } |
---|
| 248 | # endif |
---|
| 249 | ~basic_path() {} |
---|
| 250 | |
---|
| 251 | // assignments |
---|
| 252 | basic_path & operator=( const string_type & s ) |
---|
| 253 | { |
---|
| 254 | # if BOOST_WORKAROUND(BOOST_DINKUMWARE_STDLIB, >= 310) |
---|
| 255 | m_path.clear(); |
---|
| 256 | # else |
---|
| 257 | m_path.erase( m_path.begin(), m_path.end() ); |
---|
| 258 | # endif |
---|
| 259 | operator/=( s ); |
---|
| 260 | return *this; |
---|
| 261 | } |
---|
| 262 | basic_path & operator=( const value_type * s ) |
---|
| 263 | { |
---|
| 264 | # if BOOST_WORKAROUND(BOOST_DINKUMWARE_STDLIB, >= 310) |
---|
| 265 | m_path.clear(); |
---|
| 266 | # else |
---|
| 267 | m_path.erase( m_path.begin(), m_path.end() ); |
---|
| 268 | # endif |
---|
| 269 | operator/=( s ); |
---|
| 270 | return *this; |
---|
| 271 | } |
---|
| 272 | # ifndef BOOST_NO_MEMBER_TEMPLATES |
---|
| 273 | template <class InputIterator> |
---|
| 274 | basic_path & assign( InputIterator first, InputIterator last ) |
---|
| 275 | { m_path.clear(); append( first, last ); return *this; } |
---|
| 276 | # endif |
---|
| 277 | |
---|
| 278 | // modifiers |
---|
| 279 | basic_path & operator/=( const basic_path & rhs ) { return operator /=( rhs.string().c_str() ); } |
---|
| 280 | basic_path & operator/=( const string_type & rhs ) { return operator /=( rhs.c_str() ); } |
---|
| 281 | basic_path & operator/=( const value_type * s ); |
---|
| 282 | # ifndef BOOST_NO_MEMBER_TEMPLATES |
---|
| 283 | template <class InputIterator> |
---|
| 284 | basic_path & append( InputIterator first, InputIterator last ); |
---|
| 285 | # endif |
---|
| 286 | |
---|
| 287 | void swap( basic_path & rhs ) |
---|
| 288 | { |
---|
| 289 | m_path.swap( rhs.m_path ); |
---|
| 290 | # ifdef BOOST_CYGWIN_PATH |
---|
| 291 | std::swap( m_cygwin_root, rhs.m_cygwin_root ); |
---|
| 292 | # endif |
---|
| 293 | } |
---|
| 294 | |
---|
| 295 | basic_path & remove_leaf(); |
---|
| 296 | |
---|
| 297 | // observers |
---|
| 298 | const string_type & string() const { return m_path; } |
---|
| 299 | const string_type file_string() const; |
---|
| 300 | const string_type directory_string() const { return file_string(); } |
---|
| 301 | |
---|
| 302 | const external_string_type external_file_string() const { return Traits::to_external( *this, file_string() ); } |
---|
| 303 | const external_string_type external_directory_string() const { return Traits::to_external( *this, directory_string() ); } |
---|
| 304 | |
---|
| 305 | basic_path root_path() const; |
---|
| 306 | string_type root_name() const; |
---|
| 307 | string_type root_directory() const; |
---|
| 308 | basic_path relative_path() const; |
---|
| 309 | string_type leaf() const; |
---|
| 310 | basic_path branch_path() const; |
---|
| 311 | |
---|
| 312 | bool empty() const { return m_path.empty(); } // name consistent with std containers |
---|
| 313 | bool is_complete() const; |
---|
| 314 | bool has_root_path() const; |
---|
| 315 | bool has_root_name() const; |
---|
| 316 | bool has_root_directory() const; |
---|
| 317 | bool has_relative_path() const { return !relative_path().empty(); } |
---|
| 318 | bool has_leaf() const { return !m_path.empty(); } |
---|
| 319 | bool has_branch_path() const { return !branch_path().empty(); } |
---|
| 320 | |
---|
| 321 | // iterators |
---|
| 322 | class iterator : public boost::iterator_facade< |
---|
| 323 | iterator, |
---|
| 324 | string_type const, |
---|
| 325 | boost::bidirectional_traversal_tag > |
---|
| 326 | { |
---|
| 327 | private: |
---|
| 328 | friend class boost::iterator_core_access; |
---|
| 329 | friend class boost::BOOST_FILESYSTEM_NAMESPACE::basic_path<String, Traits>; |
---|
| 330 | |
---|
| 331 | const string_type & dereference() const |
---|
| 332 | { return m_name; } |
---|
| 333 | bool equal( const iterator & rhs ) const |
---|
| 334 | { return m_path_ptr == rhs.m_path_ptr && m_pos == rhs.m_pos; } |
---|
| 335 | |
---|
| 336 | friend class boost::BOOST_FILESYSTEM_NAMESPACE::detail::iterator_helper<path_type>; |
---|
| 337 | |
---|
| 338 | void increment() |
---|
| 339 | { |
---|
| 340 | boost::BOOST_FILESYSTEM_NAMESPACE::detail::iterator_helper<path_type>::do_increment( |
---|
| 341 | *this ); |
---|
| 342 | } |
---|
| 343 | void decrement() |
---|
| 344 | { |
---|
| 345 | boost::BOOST_FILESYSTEM_NAMESPACE::detail::iterator_helper<path_type>::do_decrement( |
---|
| 346 | *this ); |
---|
| 347 | } |
---|
| 348 | |
---|
| 349 | string_type m_name; // current element |
---|
| 350 | const basic_path * m_path_ptr; // path being iterated over |
---|
| 351 | typename string_type::size_type m_pos; // position of name in |
---|
| 352 | // path_ptr->string(). The |
---|
| 353 | // end() iterator is indicated by |
---|
| 354 | // pos == path_ptr->m_path.size() |
---|
| 355 | }; // iterator |
---|
| 356 | |
---|
| 357 | typedef iterator const_iterator; |
---|
| 358 | |
---|
| 359 | iterator begin() const; |
---|
| 360 | iterator end() const; |
---|
| 361 | |
---|
| 362 | private: |
---|
| 363 | // Note: This is an implementation for POSIX and Windows, where there |
---|
| 364 | // are only minor differences between generic and native path grammars. |
---|
| 365 | // Private members might be quite different in other implementations, |
---|
| 366 | // particularly where there were wide differences between portable and |
---|
| 367 | // native path formats, or between file_string() and |
---|
| 368 | // directory_string() formats, or simply that the implementation |
---|
| 369 | // was willing expend additional memory to achieve greater speed for |
---|
| 370 | // some operations at the expense of other operations. |
---|
| 371 | |
---|
| 372 | string_type m_path; // invariant: portable path grammar |
---|
| 373 | // on Windows, backslashes converted to slashes |
---|
| 374 | |
---|
| 375 | # ifdef BOOST_CYGWIN_PATH |
---|
| 376 | bool m_cygwin_root; // if present, m_path[0] was slash. note: initialization |
---|
| 377 | // done by append |
---|
| 378 | # endif |
---|
| 379 | |
---|
| 380 | void m_append_separator_if_needed(); |
---|
| 381 | void m_append( value_type value ); // converts Windows alt_separator |
---|
| 382 | |
---|
| 383 | // Was qualified; como433beta8 reports: |
---|
| 384 | // warning #427-D: qualified name is not allowed in member declaration |
---|
| 385 | friend class iterator; |
---|
| 386 | friend class boost::BOOST_FILESYSTEM_NAMESPACE::detail::iterator_helper<path_type>; |
---|
| 387 | |
---|
| 388 | // Deprecated features ease transition for existing code. Don't use these |
---|
| 389 | // in new code. |
---|
| 390 | # ifndef BOOST_FILESYSTEM_NO_DEPRECATED |
---|
| 391 | public: |
---|
| 392 | typedef bool (*name_check)( const std::string & name ); |
---|
| 393 | basic_path( const string_type & str, name_check ) { operator/=( str ); } |
---|
| 394 | basic_path( const typename string_type::value_type * s, name_check ) |
---|
| 395 | { operator/=( s );} |
---|
| 396 | string_type native_file_string() const { return file_string(); } |
---|
| 397 | string_type native_directory_string() const { return directory_string(); } |
---|
| 398 | static bool default_name_check_writable() { return false; } |
---|
| 399 | static void default_name_check( name_check ) {} |
---|
| 400 | static name_check default_name_check() { return 0; } |
---|
| 401 | basic_path & canonize(); |
---|
| 402 | basic_path & normalize(); |
---|
| 403 | # endif |
---|
| 404 | }; |
---|
| 405 | |
---|
| 406 | // basic_path non-member functions ---------------------------------------// |
---|
| 407 | |
---|
| 408 | template< class String, class Traits > |
---|
| 409 | inline void swap( basic_path<String, Traits> & lhs, |
---|
| 410 | basic_path<String, Traits> & rhs ) { lhs.swap( rhs ); } |
---|
| 411 | |
---|
| 412 | template< class String, class Traits > |
---|
| 413 | bool operator<( const basic_path<String, Traits> & lhs, const basic_path<String, Traits> & rhs ) |
---|
| 414 | { |
---|
| 415 | return std::lexicographical_compare( |
---|
| 416 | lhs.begin(), lhs.end(), rhs.begin(), rhs.end() ); |
---|
| 417 | } |
---|
| 418 | |
---|
| 419 | template< class String, class Traits > |
---|
| 420 | bool operator<( const typename basic_path<String, Traits>::string_type::value_type * lhs, |
---|
| 421 | const basic_path<String, Traits> & rhs ) |
---|
| 422 | { |
---|
| 423 | basic_path<String, Traits> tmp( lhs ); |
---|
| 424 | return std::lexicographical_compare( |
---|
| 425 | tmp.begin(), tmp.end(), rhs.begin(), rhs.end() ); |
---|
| 426 | } |
---|
| 427 | |
---|
| 428 | template< class String, class Traits > |
---|
| 429 | bool operator<( const typename basic_path<String, Traits>::string_type & lhs, |
---|
| 430 | const basic_path<String, Traits> & rhs ) |
---|
| 431 | { |
---|
| 432 | basic_path<String, Traits> tmp( lhs ); |
---|
| 433 | return std::lexicographical_compare( |
---|
| 434 | tmp.begin(), tmp.end(), rhs.begin(), rhs.end() ); |
---|
| 435 | } |
---|
| 436 | |
---|
| 437 | template< class String, class Traits > |
---|
| 438 | bool operator<( const basic_path<String, Traits> & lhs, |
---|
| 439 | const typename basic_path<String, Traits>::string_type::value_type * rhs ) |
---|
| 440 | { |
---|
| 441 | basic_path<String, Traits> tmp( rhs ); |
---|
| 442 | return std::lexicographical_compare( |
---|
| 443 | lhs.begin(), lhs.end(), tmp.begin(), tmp.end() ); |
---|
| 444 | } |
---|
| 445 | |
---|
| 446 | template< class String, class Traits > |
---|
| 447 | bool operator<( const basic_path<String, Traits> & lhs, |
---|
| 448 | const typename basic_path<String, Traits>::string_type & rhs ) |
---|
| 449 | { |
---|
| 450 | basic_path<String, Traits> tmp( rhs ); |
---|
| 451 | return std::lexicographical_compare( |
---|
| 452 | lhs.begin(), lhs.end(), tmp.begin(), tmp.end() ); |
---|
| 453 | } |
---|
| 454 | |
---|
| 455 | template< class String, class Traits > |
---|
| 456 | inline bool operator==( const basic_path<String, Traits> & lhs, const basic_path<String, Traits> & rhs ) |
---|
| 457 | { |
---|
| 458 | return !(lhs < rhs) && !(rhs < lhs); |
---|
| 459 | } |
---|
| 460 | |
---|
| 461 | template< class String, class Traits > |
---|
| 462 | inline bool operator==( const typename basic_path<String, Traits>::string_type::value_type * lhs, |
---|
| 463 | const basic_path<String, Traits> & rhs ) |
---|
| 464 | { |
---|
| 465 | basic_path<String, Traits> tmp( lhs ); |
---|
| 466 | return !(tmp < rhs) && !(rhs < tmp); |
---|
| 467 | } |
---|
| 468 | |
---|
| 469 | template< class String, class Traits > |
---|
| 470 | inline bool operator==( const typename basic_path<String, Traits>::string_type & lhs, |
---|
| 471 | const basic_path<String, Traits> & rhs ) |
---|
| 472 | { |
---|
| 473 | basic_path<String, Traits> tmp( lhs ); |
---|
| 474 | return !(tmp < rhs) && !(rhs < tmp); |
---|
| 475 | } |
---|
| 476 | |
---|
| 477 | template< class String, class Traits > |
---|
| 478 | inline bool operator==( const basic_path<String, Traits> & lhs, |
---|
| 479 | const typename basic_path<String, Traits>::string_type::value_type * rhs ) |
---|
| 480 | { |
---|
| 481 | basic_path<String, Traits> tmp( rhs ); |
---|
| 482 | return !(lhs < tmp) && !(tmp < lhs); |
---|
| 483 | } |
---|
| 484 | |
---|
| 485 | template< class String, class Traits > |
---|
| 486 | inline bool operator==( const basic_path<String, Traits> & lhs, |
---|
| 487 | const typename basic_path<String, Traits>::string_type & rhs ) |
---|
| 488 | { |
---|
| 489 | basic_path<String, Traits> tmp( rhs ); |
---|
| 490 | return !(lhs < tmp) && !(tmp < lhs); |
---|
| 491 | } |
---|
| 492 | |
---|
| 493 | template< class String, class Traits > |
---|
| 494 | inline bool operator!=( const basic_path<String, Traits> & lhs, const basic_path<String, Traits> & rhs ) { return !(lhs == rhs); } |
---|
| 495 | |
---|
| 496 | template< class String, class Traits > |
---|
| 497 | inline bool operator!=( const typename basic_path<String, Traits>::string_type::value_type * lhs, |
---|
| 498 | const basic_path<String, Traits> & rhs ) { return !(basic_path<String, Traits>(lhs) == rhs); } |
---|
| 499 | |
---|
| 500 | template< class String, class Traits > |
---|
| 501 | inline bool operator!=( const typename basic_path<String, Traits>::string_type & lhs, |
---|
| 502 | const basic_path<String, Traits> & rhs ) { return !(basic_path<String, Traits>(lhs) == rhs); } |
---|
| 503 | |
---|
| 504 | template< class String, class Traits > |
---|
| 505 | inline bool operator!=( const basic_path<String, Traits> & lhs, |
---|
| 506 | const typename basic_path<String, Traits>::string_type::value_type * rhs ) |
---|
| 507 | { return !(lhs == basic_path<String, Traits>(rhs)); } |
---|
| 508 | |
---|
| 509 | template< class String, class Traits > |
---|
| 510 | inline bool operator!=( const basic_path<String, Traits> & lhs, |
---|
| 511 | const typename basic_path<String, Traits>::string_type & rhs ) |
---|
| 512 | { return !(lhs == basic_path<String, Traits>(rhs)); } |
---|
| 513 | |
---|
| 514 | template< class String, class Traits > |
---|
| 515 | inline bool operator>( const basic_path<String, Traits> & lhs, const basic_path<String, Traits> & rhs ) { return rhs < lhs; } |
---|
| 516 | |
---|
| 517 | template< class String, class Traits > |
---|
| 518 | inline bool operator>( const typename basic_path<String, Traits>::string_type::value_type * lhs, |
---|
| 519 | const basic_path<String, Traits> & rhs ) { return rhs < basic_path<String, Traits>(lhs); } |
---|
| 520 | |
---|
| 521 | template< class String, class Traits > |
---|
| 522 | inline bool operator>( const typename basic_path<String, Traits>::string_type & lhs, |
---|
| 523 | const basic_path<String, Traits> & rhs ) { return rhs < basic_path<String, Traits>(lhs); } |
---|
| 524 | |
---|
| 525 | template< class String, class Traits > |
---|
| 526 | inline bool operator>( const basic_path<String, Traits> & lhs, |
---|
| 527 | const typename basic_path<String, Traits>::string_type::value_type * rhs ) |
---|
| 528 | { return basic_path<String, Traits>(rhs) < lhs; } |
---|
| 529 | |
---|
| 530 | template< class String, class Traits > |
---|
| 531 | inline bool operator>( const basic_path<String, Traits> & lhs, |
---|
| 532 | const typename basic_path<String, Traits>::string_type & rhs ) |
---|
| 533 | { return basic_path<String, Traits>(rhs) < lhs; } |
---|
| 534 | |
---|
| 535 | template< class String, class Traits > |
---|
| 536 | inline bool operator<=( const basic_path<String, Traits> & lhs, const basic_path<String, Traits> & rhs ) { return !(rhs < lhs); } |
---|
| 537 | |
---|
| 538 | template< class String, class Traits > |
---|
| 539 | inline bool operator<=( const typename basic_path<String, Traits>::string_type::value_type * lhs, |
---|
| 540 | const basic_path<String, Traits> & rhs ) { return !(rhs < basic_path<String, Traits>(lhs)); } |
---|
| 541 | |
---|
| 542 | template< class String, class Traits > |
---|
| 543 | inline bool operator<=( const typename basic_path<String, Traits>::string_type & lhs, |
---|
| 544 | const basic_path<String, Traits> & rhs ) { return !(rhs < basic_path<String, Traits>(lhs)); } |
---|
| 545 | |
---|
| 546 | template< class String, class Traits > |
---|
| 547 | inline bool operator<=( const basic_path<String, Traits> & lhs, |
---|
| 548 | const typename basic_path<String, Traits>::string_type::value_type * rhs ) |
---|
| 549 | { return !(basic_path<String, Traits>(rhs) < lhs); } |
---|
| 550 | |
---|
| 551 | template< class String, class Traits > |
---|
| 552 | inline bool operator<=( const basic_path<String, Traits> & lhs, |
---|
| 553 | const typename basic_path<String, Traits>::string_type & rhs ) |
---|
| 554 | { return !(basic_path<String, Traits>(rhs) < lhs); } |
---|
| 555 | |
---|
| 556 | template< class String, class Traits > |
---|
| 557 | inline bool operator>=( const basic_path<String, Traits> & lhs, const basic_path<String, Traits> & rhs ) { return !(lhs < rhs); } |
---|
| 558 | |
---|
| 559 | template< class String, class Traits > |
---|
| 560 | inline bool operator>=( const typename basic_path<String, Traits>::string_type::value_type * lhs, |
---|
| 561 | const basic_path<String, Traits> & rhs ) { return !(lhs < basic_path<String, Traits>(rhs)); } |
---|
| 562 | |
---|
| 563 | template< class String, class Traits > |
---|
| 564 | inline bool operator>=( const typename basic_path<String, Traits>::string_type & lhs, |
---|
| 565 | const basic_path<String, Traits> & rhs ) { return !(lhs < basic_path<String, Traits>(rhs)); } |
---|
| 566 | |
---|
| 567 | template< class String, class Traits > |
---|
| 568 | inline bool operator>=( const basic_path<String, Traits> & lhs, |
---|
| 569 | const typename basic_path<String, Traits>::string_type::value_type * rhs ) |
---|
| 570 | { return !(basic_path<String, Traits>(lhs) < rhs); } |
---|
| 571 | |
---|
| 572 | template< class String, class Traits > |
---|
| 573 | inline bool operator>=( const basic_path<String, Traits> & lhs, |
---|
| 574 | const typename basic_path<String, Traits>::string_type & rhs ) |
---|
| 575 | { return !(basic_path<String, Traits>(lhs) < rhs); } |
---|
| 576 | |
---|
| 577 | // operator / |
---|
| 578 | |
---|
| 579 | template< class String, class Traits > |
---|
| 580 | inline basic_path<String, Traits> operator/( |
---|
| 581 | const basic_path<String, Traits> & lhs, |
---|
| 582 | const basic_path<String, Traits> & rhs ) |
---|
| 583 | { return basic_path<String, Traits>( lhs ) /= rhs; } |
---|
| 584 | |
---|
| 585 | template< class String, class Traits > |
---|
| 586 | inline basic_path<String, Traits> operator/( |
---|
| 587 | const basic_path<String, Traits> & lhs, |
---|
| 588 | const typename String::value_type * rhs ) |
---|
| 589 | { return basic_path<String, Traits>( lhs ) /= |
---|
| 590 | basic_path<String, Traits>( rhs ); } |
---|
| 591 | |
---|
| 592 | template< class String, class Traits > |
---|
| 593 | inline basic_path<String, Traits> operator/( |
---|
| 594 | const basic_path<String, Traits> & lhs, const String & rhs ) |
---|
| 595 | { return basic_path<String, Traits>( lhs ) /= |
---|
| 596 | basic_path<String, Traits>( rhs ); } |
---|
| 597 | |
---|
| 598 | template< class String, class Traits > |
---|
| 599 | inline basic_path<String, Traits> operator/( |
---|
| 600 | const typename String::value_type * lhs, |
---|
| 601 | const basic_path<String, Traits> & rhs ) |
---|
| 602 | { return basic_path<String, Traits>( lhs ) /= rhs; } |
---|
| 603 | |
---|
| 604 | template< class String, class Traits > |
---|
| 605 | inline basic_path<String, Traits> operator/( |
---|
| 606 | const String & lhs, const basic_path<String, Traits> & rhs ) |
---|
| 607 | { return basic_path<String, Traits>( lhs ) /= rhs; } |
---|
| 608 | |
---|
| 609 | // inserters and extractors --------------------------------------------// |
---|
| 610 | |
---|
| 611 | // bypass VC++ 7.0 and earlier, and broken Borland compilers |
---|
| 612 | # if !BOOST_WORKAROUND(BOOST_MSVC, <= 1300) && !BOOST_WORKAROUND(__BORLANDC__, BOOST_TESTED_AT(0x564)) |
---|
| 613 | template< class Path > |
---|
| 614 | std::basic_ostream< typename Path::string_type::value_type, |
---|
| 615 | typename Path::string_type::traits_type > & |
---|
| 616 | operator<< |
---|
| 617 | ( std::basic_ostream< typename Path::string_type::value_type, |
---|
| 618 | typename Path::string_type::traits_type >& os, const Path & ph ) |
---|
| 619 | { |
---|
| 620 | os << ph.string(); |
---|
| 621 | return os; |
---|
| 622 | } |
---|
| 623 | |
---|
| 624 | template< class Path > |
---|
| 625 | std::basic_istream< typename Path::string_type::value_type, |
---|
| 626 | typename Path::string_type::traits_type > & |
---|
| 627 | operator>> |
---|
| 628 | ( std::basic_istream< typename Path::string_type::value_type, |
---|
| 629 | typename Path::string_type::traits_type >& is, Path & ph ) |
---|
| 630 | { |
---|
| 631 | typename Path::string_type str; |
---|
| 632 | is >> str; |
---|
| 633 | ph = str; |
---|
| 634 | return is; |
---|
| 635 | } |
---|
| 636 | # elif BOOST_WORKAROUND(__BORLANDC__, BOOST_TESTED_AT(0x564)) |
---|
| 637 | template< class String, class Traits > |
---|
| 638 | std::basic_ostream< BOOST_DEDUCED_TYPENAME String::value_type, |
---|
| 639 | BOOST_DEDUCED_TYPENAME String::traits_type > & |
---|
| 640 | operator<< |
---|
| 641 | ( std::basic_ostream< BOOST_DEDUCED_TYPENAME String::value_type, |
---|
| 642 | BOOST_DEDUCED_TYPENAME String::traits_type >& os, |
---|
| 643 | const basic_path< String, Traits > & ph ) |
---|
| 644 | { |
---|
| 645 | os << ph.string(); |
---|
| 646 | return os; |
---|
| 647 | } |
---|
| 648 | |
---|
| 649 | template< class String, class Traits > |
---|
| 650 | std::basic_istream< BOOST_DEDUCED_TYPENAME String::value_type, |
---|
| 651 | BOOST_DEDUCED_TYPENAME String::traits_type > & |
---|
| 652 | operator>> |
---|
| 653 | ( std::basic_istream< BOOST_DEDUCED_TYPENAME String::value_type, |
---|
| 654 | BOOST_DEDUCED_TYPENAME String::traits_type> & is, |
---|
| 655 | basic_path< String, Traits > & ph ) |
---|
| 656 | { |
---|
| 657 | String str; |
---|
| 658 | is >> str; |
---|
| 659 | ph = str; |
---|
| 660 | return is; |
---|
| 661 | } |
---|
| 662 | # endif |
---|
| 663 | |
---|
| 664 | // path::name_checks -----------------------------------------------------// |
---|
| 665 | |
---|
| 666 | BOOST_FILESYSTEM_DECL bool portable_posix_name( const std::string & name ); |
---|
| 667 | BOOST_FILESYSTEM_DECL bool windows_name( const std::string & name ); |
---|
| 668 | BOOST_FILESYSTEM_DECL bool portable_name( const std::string & name ); |
---|
| 669 | BOOST_FILESYSTEM_DECL bool portable_directory_name( const std::string & name ); |
---|
| 670 | BOOST_FILESYSTEM_DECL bool portable_file_name( const std::string & name ); |
---|
| 671 | BOOST_FILESYSTEM_DECL bool native( const std::string & name ); |
---|
| 672 | inline bool no_check( const std::string & ) |
---|
| 673 | { return true; } |
---|
| 674 | |
---|
| 675 | // implementation -----------------------------------------------------------// |
---|
| 676 | |
---|
| 677 | namespace detail |
---|
| 678 | { |
---|
| 679 | |
---|
| 680 | // is_separator helper ------------------------------------------------// |
---|
| 681 | |
---|
| 682 | template<class Path> |
---|
| 683 | inline bool is_separator( typename Path::string_type::value_type c ) |
---|
| 684 | { |
---|
| 685 | return c == slash<Path>::value |
---|
| 686 | # ifdef BOOST_WINDOWS_PATH |
---|
| 687 | || c == path_alt_separator<Path>::value |
---|
| 688 | # endif |
---|
| 689 | ; |
---|
| 690 | } |
---|
| 691 | |
---|
| 692 | // leaf_pos helper ----------------------------------------------------// |
---|
| 693 | |
---|
| 694 | template<class String, class Traits> |
---|
| 695 | typename String::size_type leaf_pos( |
---|
| 696 | const String & str, // precondition: portable generic path grammar |
---|
| 697 | typename String::size_type end_pos ) // end_pos is past-the-end position |
---|
| 698 | // return 0 if str itself is leaf (or empty) |
---|
| 699 | { |
---|
| 700 | typedef typename |
---|
| 701 | boost::BOOST_FILESYSTEM_NAMESPACE::basic_path<String, Traits> path_type; |
---|
| 702 | |
---|
| 703 | // case: "//" |
---|
| 704 | if ( end_pos == 2 |
---|
| 705 | && str[0] == slash<path_type>::value |
---|
| 706 | && str[1] == slash<path_type>::value ) return 0; |
---|
| 707 | |
---|
| 708 | // case: ends in "/" |
---|
| 709 | if ( end_pos && str[end_pos-1] == slash<path_type>::value ) |
---|
| 710 | return end_pos-1; |
---|
| 711 | |
---|
| 712 | // set pos to start of last element |
---|
| 713 | typename String::size_type pos( |
---|
| 714 | str.find_last_of( slash<path_type>::value, end_pos-1 ) ); |
---|
| 715 | # ifdef BOOST_WINDOWS_PATH |
---|
| 716 | if ( pos == String::npos ) |
---|
| 717 | pos = str.find_last_of( path_alt_separator<path_type>::value, end_pos-1 ); |
---|
| 718 | if ( pos == String::npos ) |
---|
| 719 | pos = str.find_last_of( colon<path_type>::value, end_pos-2 ); |
---|
| 720 | # endif |
---|
| 721 | |
---|
| 722 | return ( pos == String::npos // path itself must be a leaf (or empty) |
---|
| 723 | || (pos == 1 && str[0] == slash<path_type>::value) ) // or net |
---|
| 724 | ? 0 // so leaf is entire string |
---|
| 725 | : pos + 1; // or starts after delimiter |
---|
| 726 | } |
---|
| 727 | |
---|
| 728 | // first_element helper -----------------------------------------------// |
---|
| 729 | // sets pos and len of first element, excluding extra separators |
---|
| 730 | // if src.empty(), sets pos,len, to 0,0. |
---|
| 731 | |
---|
| 732 | template<class String, class Traits> |
---|
| 733 | void first_element( |
---|
| 734 | const String & src, // precondition: portable generic path grammar |
---|
| 735 | typename String::size_type & element_pos, |
---|
| 736 | typename String::size_type & element_size, |
---|
| 737 | # if !BOOST_WORKAROUND( BOOST_MSVC, <= 1310 ) // VC++ 7.1 |
---|
| 738 | typename String::size_type size = String::npos |
---|
| 739 | # else |
---|
| 740 | typename String::size_type size = -1 |
---|
| 741 | # endif |
---|
| 742 | ) |
---|
| 743 | { |
---|
| 744 | if ( size == String::npos ) size = src.size(); |
---|
| 745 | element_pos = 0; |
---|
| 746 | element_size = 0; |
---|
| 747 | if ( src.empty() ) return; |
---|
| 748 | |
---|
| 749 | typedef typename boost::BOOST_FILESYSTEM_NAMESPACE::basic_path<String, Traits> path_type; |
---|
| 750 | |
---|
| 751 | typename String::size_type cur(0); |
---|
| 752 | |
---|
| 753 | // deal with // [network] |
---|
| 754 | if ( size >= 2 && src[0] == slash<path_type>::value |
---|
| 755 | && src[1] == slash<path_type>::value |
---|
| 756 | && (size == 2 |
---|
| 757 | || src[2] != slash<path_type>::value) ) |
---|
| 758 | { |
---|
| 759 | cur += 2; |
---|
| 760 | element_size += 2; |
---|
| 761 | } |
---|
| 762 | |
---|
| 763 | // leading (not non-network) separator |
---|
| 764 | else if ( src[0] == slash<path_type>::value ) |
---|
| 765 | { |
---|
| 766 | ++element_size; |
---|
| 767 | // bypass extra leading separators |
---|
| 768 | while ( cur+1 < size |
---|
| 769 | && src[cur+1] == slash<path_type>::value ) |
---|
| 770 | { |
---|
| 771 | ++cur; |
---|
| 772 | ++element_pos; |
---|
| 773 | } |
---|
| 774 | return; |
---|
| 775 | } |
---|
| 776 | |
---|
| 777 | // at this point, we have either a plain name, a network name, |
---|
| 778 | // or (on Windows only) a device name |
---|
| 779 | |
---|
| 780 | // find the end |
---|
| 781 | while ( cur < size |
---|
| 782 | # ifdef BOOST_WINDOWS_PATH |
---|
| 783 | && src[cur] != colon<path_type>::value |
---|
| 784 | # endif |
---|
| 785 | && src[cur] != slash<path_type>::value ) |
---|
| 786 | { |
---|
| 787 | ++cur; |
---|
| 788 | ++element_size; |
---|
| 789 | } |
---|
| 790 | |
---|
| 791 | # ifdef BOOST_WINDOWS_PATH |
---|
| 792 | if ( cur == size ) return; |
---|
| 793 | // include device delimiter |
---|
| 794 | if ( src[cur] == colon<path_type>::value ) |
---|
| 795 | { ++element_size; } |
---|
| 796 | # endif |
---|
| 797 | |
---|
| 798 | return; |
---|
| 799 | } |
---|
| 800 | |
---|
| 801 | // root_directory_start helper ----------------------------------------// |
---|
| 802 | |
---|
| 803 | template<class String, class Traits> |
---|
| 804 | typename String::size_type root_directory_start( |
---|
| 805 | const String & s, // precondition: portable generic path grammar |
---|
| 806 | typename String::size_type size ) |
---|
| 807 | // return npos if no root_directory found |
---|
| 808 | { |
---|
| 809 | typedef typename boost::BOOST_FILESYSTEM_NAMESPACE::basic_path<String, Traits> path_type; |
---|
| 810 | |
---|
| 811 | # ifdef BOOST_WINDOWS_PATH |
---|
| 812 | // case "c:/" |
---|
| 813 | if ( size > 2 |
---|
| 814 | && s[1] == colon<path_type>::value |
---|
| 815 | && s[2] == slash<path_type>::value ) return 2; |
---|
| 816 | # endif |
---|
| 817 | |
---|
| 818 | // case "//" |
---|
| 819 | if ( size == 2 |
---|
| 820 | && s[0] == slash<path_type>::value |
---|
| 821 | && s[1] == slash<path_type>::value ) return String::npos; |
---|
| 822 | |
---|
| 823 | // case "//net {/}" |
---|
| 824 | if ( size > 3 |
---|
| 825 | && s[0] == slash<path_type>::value |
---|
| 826 | && s[1] == slash<path_type>::value |
---|
| 827 | && s[2] != slash<path_type>::value ) |
---|
| 828 | { |
---|
| 829 | typename String::size_type pos( |
---|
| 830 | s.find( slash<path_type>::value, 2 ) ); |
---|
| 831 | return pos < size ? pos : String::npos; |
---|
| 832 | } |
---|
| 833 | |
---|
| 834 | // case "/" |
---|
| 835 | if ( size > 0 && s[0] == slash<path_type>::value ) return 0; |
---|
| 836 | |
---|
| 837 | return String::npos; |
---|
| 838 | } |
---|
| 839 | |
---|
| 840 | // is_non_root_slash helper -------------------------------------------// |
---|
| 841 | |
---|
| 842 | template<class String, class Traits> |
---|
| 843 | bool is_non_root_slash( const String & str, |
---|
| 844 | typename String::size_type pos ) // pos is position of the slash |
---|
| 845 | { |
---|
| 846 | typedef typename |
---|
| 847 | boost::BOOST_FILESYSTEM_NAMESPACE::basic_path<String, Traits> |
---|
| 848 | path_type; |
---|
| 849 | |
---|
| 850 | assert( !str.empty() && str[pos] == slash<path_type>::value |
---|
| 851 | && "precondition violation" ); |
---|
| 852 | |
---|
| 853 | // subsequent logic expects pos to be for leftmost slash of a set |
---|
| 854 | while ( pos > 0 && str[pos-1] == slash<path_type>::value ) |
---|
| 855 | --pos; |
---|
| 856 | |
---|
| 857 | return pos != 0 |
---|
| 858 | && (pos <= 2 || str[1] != slash<path_type>::value |
---|
| 859 | || str.find( slash<path_type>::value, 2 ) != pos) |
---|
| 860 | # ifdef BOOST_WINDOWS_PATH |
---|
| 861 | && (pos !=2 || str[1] != colon<path_type>::value) |
---|
| 862 | # endif |
---|
| 863 | ; |
---|
| 864 | } |
---|
| 865 | } // namespace detail |
---|
| 866 | |
---|
| 867 | // decomposition functions ----------------------------------------------// |
---|
| 868 | |
---|
| 869 | template<class String, class Traits> |
---|
| 870 | String basic_path<String, Traits>::leaf() const |
---|
| 871 | { |
---|
| 872 | typename String::size_type end_pos( |
---|
| 873 | detail::leaf_pos<String, Traits>( m_path, m_path.size() ) ); |
---|
| 874 | return (m_path.size() |
---|
| 875 | && end_pos |
---|
| 876 | && m_path[end_pos] == slash<path_type>::value |
---|
| 877 | && detail::is_non_root_slash< String, Traits >(m_path, end_pos)) |
---|
| 878 | ? String( 1, dot<path_type>::value ) |
---|
| 879 | : m_path.substr( end_pos ); |
---|
| 880 | } |
---|
| 881 | |
---|
| 882 | template<class String, class Traits> |
---|
| 883 | basic_path<String, Traits> basic_path<String, Traits>::branch_path() const |
---|
| 884 | { |
---|
| 885 | typename String::size_type end_pos( |
---|
| 886 | detail::leaf_pos<String, Traits>( m_path, m_path.size() ) ); |
---|
| 887 | |
---|
| 888 | bool leaf_was_separator( m_path.size() |
---|
| 889 | && m_path[end_pos] == slash<path_type>::value ); |
---|
| 890 | |
---|
| 891 | // skip separators unless root directory |
---|
| 892 | typename string_type::size_type root_dir_pos( detail::root_directory_start |
---|
| 893 | <string_type, traits_type>( m_path, end_pos ) ); |
---|
| 894 | for ( ; |
---|
| 895 | end_pos > 0 |
---|
| 896 | && (end_pos-1) != root_dir_pos |
---|
| 897 | && m_path[end_pos-1] == slash<path_type>::value |
---|
| 898 | ; |
---|
| 899 | --end_pos ) {} |
---|
| 900 | |
---|
| 901 | return (end_pos == 1 && root_dir_pos == 0 && leaf_was_separator) |
---|
| 902 | ? path_type() |
---|
| 903 | : path_type( m_path.substr( 0, end_pos ) ); |
---|
| 904 | } |
---|
| 905 | |
---|
| 906 | template<class String, class Traits> |
---|
| 907 | basic_path<String, Traits> basic_path<String, Traits>::relative_path() const |
---|
| 908 | { |
---|
| 909 | iterator itr( begin() ); |
---|
| 910 | for ( ; itr.m_pos != m_path.size() |
---|
| 911 | && (itr.m_name[0] == slash<path_type>::value |
---|
| 912 | # ifdef BOOST_WINDOWS_PATH |
---|
| 913 | || itr.m_name[itr.m_name.size()-1] |
---|
| 914 | == colon<path_type>::value |
---|
| 915 | # endif |
---|
| 916 | ); ++itr ) {} |
---|
| 917 | |
---|
| 918 | return basic_path<String, Traits>( m_path.substr( itr.m_pos ) ); |
---|
| 919 | } |
---|
| 920 | |
---|
| 921 | template<class String, class Traits> |
---|
| 922 | String basic_path<String, Traits>::root_name() const |
---|
| 923 | { |
---|
| 924 | iterator itr( begin() ); |
---|
| 925 | |
---|
| 926 | return ( itr.m_pos != m_path.size() |
---|
| 927 | && ( |
---|
| 928 | ( itr.m_name.size() > 1 |
---|
| 929 | && itr.m_name[0] == slash<path_type>::value |
---|
| 930 | && itr.m_name[1] == slash<path_type>::value |
---|
| 931 | ) |
---|
| 932 | # ifdef BOOST_WINDOWS_PATH |
---|
| 933 | || itr.m_name[itr.m_name.size()-1] |
---|
| 934 | == colon<path_type>::value |
---|
| 935 | # endif |
---|
| 936 | ) ) |
---|
| 937 | ? *itr |
---|
| 938 | : String(); |
---|
| 939 | } |
---|
| 940 | |
---|
| 941 | template<class String, class Traits> |
---|
| 942 | String basic_path<String, Traits>::root_directory() const |
---|
| 943 | { |
---|
| 944 | typename string_type::size_type start( |
---|
| 945 | detail::root_directory_start<String, Traits>( m_path, m_path.size() ) ); |
---|
| 946 | |
---|
| 947 | return start == string_type::npos |
---|
| 948 | ? string_type() |
---|
| 949 | : m_path.substr( start, 1 ); |
---|
| 950 | } |
---|
| 951 | |
---|
| 952 | template<class String, class Traits> |
---|
| 953 | basic_path<String, Traits> basic_path<String, Traits>::root_path() const |
---|
| 954 | { |
---|
| 955 | // even on POSIX, root_name() is non-empty() on network paths |
---|
| 956 | return basic_path<String, Traits>( root_name() ) /= root_directory(); |
---|
| 957 | } |
---|
| 958 | |
---|
| 959 | // path query functions -------------------------------------------------// |
---|
| 960 | |
---|
| 961 | template<class String, class Traits> |
---|
| 962 | inline bool basic_path<String, Traits>::is_complete() const |
---|
| 963 | { |
---|
| 964 | # ifdef BOOST_WINDOWS_PATH |
---|
| 965 | return has_root_name() && has_root_directory(); |
---|
| 966 | # else |
---|
| 967 | return has_root_directory(); |
---|
| 968 | # endif |
---|
| 969 | } |
---|
| 970 | |
---|
| 971 | template<class String, class Traits> |
---|
| 972 | inline bool basic_path<String, Traits>::has_root_path() const |
---|
| 973 | { |
---|
| 974 | return !root_path().empty(); |
---|
| 975 | } |
---|
| 976 | |
---|
| 977 | template<class String, class Traits> |
---|
| 978 | inline bool basic_path<String, Traits>::has_root_name() const |
---|
| 979 | { |
---|
| 980 | return !root_name().empty(); |
---|
| 981 | } |
---|
| 982 | |
---|
| 983 | template<class String, class Traits> |
---|
| 984 | inline bool basic_path<String, Traits>::has_root_directory() const |
---|
| 985 | { |
---|
| 986 | return !root_directory().empty(); |
---|
| 987 | } |
---|
| 988 | |
---|
| 989 | // append ---------------------------------------------------------------// |
---|
| 990 | |
---|
| 991 | template<class String, class Traits> |
---|
| 992 | void basic_path<String, Traits>::m_append_separator_if_needed() |
---|
| 993 | // requires: !empty() |
---|
| 994 | { |
---|
| 995 | if ( |
---|
| 996 | # ifdef BOOST_WINDOWS_PATH |
---|
| 997 | *(m_path.end()-1) != colon<path_type>::value && |
---|
| 998 | # endif |
---|
| 999 | *(m_path.end()-1) != slash<path_type>::value ) |
---|
| 1000 | { |
---|
| 1001 | m_path += slash<path_type>::value; |
---|
| 1002 | } |
---|
| 1003 | } |
---|
| 1004 | |
---|
| 1005 | template<class String, class Traits> |
---|
| 1006 | void basic_path<String, Traits>::m_append( value_type value ) |
---|
| 1007 | { |
---|
| 1008 | # ifdef BOOST_CYGWIN_PATH |
---|
| 1009 | if ( m_path.empty() ) m_cygwin_root = (value == slash<path_type>::value); |
---|
| 1010 | # endif |
---|
| 1011 | |
---|
| 1012 | # ifdef BOOST_WINDOWS_PATH |
---|
| 1013 | // for BOOST_WINDOWS_PATH, convert alt_separator ('\') to separator ('/') |
---|
| 1014 | m_path += ( value == path_alt_separator<path_type>::value |
---|
| 1015 | ? slash<path_type>::value |
---|
| 1016 | : value ); |
---|
| 1017 | # else |
---|
| 1018 | m_path += value; |
---|
| 1019 | # endif |
---|
| 1020 | } |
---|
| 1021 | |
---|
| 1022 | // except that it wouldn't work for BOOST_NO_MEMBER_TEMPLATES compilers, |
---|
| 1023 | // the append() member template could replace this code. |
---|
| 1024 | template<class String, class Traits> |
---|
| 1025 | basic_path<String, Traits> & basic_path<String, Traits>::operator /= |
---|
| 1026 | ( const value_type * next_p ) |
---|
| 1027 | { |
---|
| 1028 | // ignore escape sequence on POSIX or Windows |
---|
| 1029 | if ( *next_p == slash<path_type>::value |
---|
| 1030 | && *(next_p+1) == slash<path_type>::value |
---|
| 1031 | && *(next_p+2) == colon<path_type>::value ) next_p += 3; |
---|
| 1032 | |
---|
| 1033 | // append slash<path_type>::value if needed |
---|
| 1034 | if ( !empty() && *next_p != 0 |
---|
| 1035 | && !detail::is_separator<path_type>( *next_p ) ) |
---|
| 1036 | { m_append_separator_if_needed(); } |
---|
| 1037 | |
---|
| 1038 | for ( ; *next_p != 0; ++next_p ) m_append( *next_p ); |
---|
| 1039 | return *this; |
---|
| 1040 | } |
---|
| 1041 | |
---|
| 1042 | # ifndef BOOST_NO_MEMBER_TEMPLATES |
---|
| 1043 | template<class String, class Traits> template <class InputIterator> |
---|
| 1044 | basic_path<String, Traits> & basic_path<String, Traits>::append( |
---|
| 1045 | InputIterator first, InputIterator last ) |
---|
| 1046 | { |
---|
| 1047 | // append slash<path_type>::value if needed |
---|
| 1048 | if ( !empty() && first != last |
---|
| 1049 | && !detail::is_separator<path_type>( *first ) ) |
---|
| 1050 | { m_append_separator_if_needed(); } |
---|
| 1051 | |
---|
| 1052 | // song-and-dance to avoid violating InputIterator requirements |
---|
| 1053 | // (which prohibit lookahead) in detecting a possible escape sequence |
---|
| 1054 | // (escape sequences are simply ignored on POSIX and Windows) |
---|
| 1055 | bool was_escape_sequence(true); |
---|
| 1056 | std::size_t append_count(0); |
---|
| 1057 | typename String::size_type initial_pos( m_path.size() ); |
---|
| 1058 | |
---|
| 1059 | for ( ; first != last && *first; ++first ) |
---|
| 1060 | { |
---|
| 1061 | if ( append_count == 0 && *first != slash<path_type>::value ) |
---|
| 1062 | was_escape_sequence = false; |
---|
| 1063 | if ( append_count == 1 && *first != slash<path_type>::value ) |
---|
| 1064 | was_escape_sequence = false; |
---|
| 1065 | if ( append_count == 2 && *first != colon<path_type>::value ) |
---|
| 1066 | was_escape_sequence = false; |
---|
| 1067 | m_append( *first ); |
---|
| 1068 | ++append_count; |
---|
| 1069 | } |
---|
| 1070 | |
---|
| 1071 | // erase escape sequence if any |
---|
| 1072 | if ( was_escape_sequence && append_count >= 3 ) |
---|
| 1073 | m_path.erase( initial_pos, 3 ); |
---|
| 1074 | |
---|
| 1075 | return *this; |
---|
| 1076 | } |
---|
| 1077 | # endif |
---|
| 1078 | |
---|
| 1079 | # ifndef BOOST_FILESYSTEM_NO_DEPRECATED |
---|
| 1080 | |
---|
| 1081 | // canonize ------------------------------------------------------------// |
---|
| 1082 | |
---|
| 1083 | template<class String, class Traits> |
---|
| 1084 | basic_path<String, Traits> & basic_path<String, Traits>::canonize() |
---|
| 1085 | { |
---|
| 1086 | static const typename string_type::value_type dot_str[] |
---|
| 1087 | = { dot<path_type>::value, 0 }; |
---|
| 1088 | |
---|
| 1089 | if ( m_path.empty() ) return *this; |
---|
| 1090 | |
---|
| 1091 | path_type temp; |
---|
| 1092 | |
---|
| 1093 | for ( iterator itr( begin() ); itr != end(); ++itr ) |
---|
| 1094 | { |
---|
| 1095 | temp /= *itr; |
---|
| 1096 | }; |
---|
| 1097 | |
---|
| 1098 | if ( temp.empty() ) temp /= dot_str; |
---|
| 1099 | m_path = temp.m_path; |
---|
| 1100 | return *this; |
---|
| 1101 | } |
---|
| 1102 | |
---|
| 1103 | // normalize ------------------------------------------------------------// |
---|
| 1104 | |
---|
| 1105 | template<class String, class Traits> |
---|
| 1106 | basic_path<String, Traits> & basic_path<String, Traits>::normalize() |
---|
| 1107 | { |
---|
| 1108 | static const typename string_type::value_type dot_str[] |
---|
| 1109 | = { dot<path_type>::value, 0 }; |
---|
| 1110 | |
---|
| 1111 | if ( m_path.empty() ) return *this; |
---|
| 1112 | |
---|
| 1113 | path_type temp; |
---|
| 1114 | iterator start( begin() ); |
---|
| 1115 | iterator last( end() ); |
---|
| 1116 | iterator stop( last-- ); |
---|
| 1117 | for ( iterator itr( start ); itr != stop; ++itr ) |
---|
| 1118 | { |
---|
| 1119 | // ignore "." except at start and last |
---|
| 1120 | if ( itr->size() == 1 |
---|
| 1121 | && (*itr)[0] == dot<path_type>::value |
---|
| 1122 | && itr != start |
---|
| 1123 | && itr != last ) continue; |
---|
| 1124 | |
---|
| 1125 | // ignore a name and following ".." |
---|
| 1126 | if ( !temp.empty() |
---|
| 1127 | && itr->size() == 2 |
---|
| 1128 | && (*itr)[0] == dot<path_type>::value |
---|
| 1129 | && (*itr)[1] == dot<path_type>::value ) // dot dot |
---|
| 1130 | { |
---|
| 1131 | string_type lf( temp.leaf() ); |
---|
| 1132 | if ( lf.size() > 0 |
---|
| 1133 | && (lf.size() != 1 |
---|
| 1134 | || (lf[0] != dot<path_type>::value |
---|
| 1135 | && lf[0] != slash<path_type>::value)) |
---|
| 1136 | && (lf.size() != 2 |
---|
| 1137 | || (lf[0] != dot<path_type>::value |
---|
| 1138 | && lf[1] != dot<path_type>::value |
---|
| 1139 | # ifdef BOOST_WINDOWS_PATH |
---|
| 1140 | && lf[1] != colon<path_type>::value |
---|
| 1141 | # endif |
---|
| 1142 | ) |
---|
| 1143 | ) |
---|
| 1144 | ) |
---|
| 1145 | { |
---|
| 1146 | temp.remove_leaf(); |
---|
| 1147 | // if not root directory, must also remove "/" if any |
---|
| 1148 | if ( temp.m_path.size() > 0 |
---|
| 1149 | && temp.m_path[temp.m_path.size()-1] |
---|
| 1150 | == slash<path_type>::value ) |
---|
| 1151 | { |
---|
| 1152 | typename string_type::size_type rds( |
---|
| 1153 | detail::root_directory_start<String,Traits>( temp.m_path, |
---|
| 1154 | temp.m_path.size() ) ); |
---|
| 1155 | if ( rds == string_type::npos |
---|
| 1156 | || rds != temp.m_path.size()-1 ) |
---|
| 1157 | { temp.m_path.erase( temp.m_path.size()-1 ); } |
---|
| 1158 | } |
---|
| 1159 | |
---|
| 1160 | iterator next( itr ); |
---|
| 1161 | if ( temp.empty() && ++next != stop |
---|
| 1162 | && next == last && *last == dot_str ) temp /= dot_str; |
---|
| 1163 | continue; |
---|
| 1164 | } |
---|
| 1165 | } |
---|
| 1166 | |
---|
| 1167 | temp /= *itr; |
---|
| 1168 | }; |
---|
| 1169 | |
---|
| 1170 | if ( temp.empty() ) temp /= dot_str; |
---|
| 1171 | m_path = temp.m_path; |
---|
| 1172 | return *this; |
---|
| 1173 | } |
---|
| 1174 | |
---|
| 1175 | # endif |
---|
| 1176 | |
---|
| 1177 | // remove_leaf ----------------------------------------------------------// |
---|
| 1178 | |
---|
| 1179 | template<class String, class Traits> |
---|
| 1180 | basic_path<String, Traits> & basic_path<String, Traits>::remove_leaf() |
---|
| 1181 | { |
---|
| 1182 | m_path.erase( |
---|
| 1183 | detail::leaf_pos<String, Traits>( m_path, m_path.size() ) ); |
---|
| 1184 | return *this; |
---|
| 1185 | } |
---|
| 1186 | |
---|
| 1187 | // path conversion functions --------------------------------------------// |
---|
| 1188 | |
---|
| 1189 | template<class String, class Traits> |
---|
| 1190 | const String |
---|
| 1191 | basic_path<String, Traits>::file_string() const |
---|
| 1192 | { |
---|
| 1193 | # ifdef BOOST_WINDOWS_PATH |
---|
| 1194 | // for Windows, use the alternate separator, and bypass extra |
---|
| 1195 | // root separators |
---|
| 1196 | |
---|
| 1197 | typename string_type::size_type root_dir_start( |
---|
| 1198 | detail::root_directory_start<String, Traits>( m_path, m_path.size() ) ); |
---|
| 1199 | bool in_root( root_dir_start != string_type::npos ); |
---|
| 1200 | String s; |
---|
| 1201 | for ( typename string_type::size_type pos( 0 ); |
---|
| 1202 | pos != m_path.size(); ++pos ) |
---|
| 1203 | { |
---|
| 1204 | // special case // [net] |
---|
| 1205 | if ( pos == 0 && m_path.size() > 1 |
---|
| 1206 | && m_path[0] == slash<path_type>::value |
---|
| 1207 | && m_path[1] == slash<path_type>::value |
---|
| 1208 | && ( m_path.size() == 2 |
---|
| 1209 | || !detail::is_separator<path_type>( m_path[2] ) |
---|
| 1210 | ) ) |
---|
| 1211 | { |
---|
| 1212 | ++pos; |
---|
| 1213 | s += path_alt_separator<path_type>::value; |
---|
| 1214 | s += path_alt_separator<path_type>::value; |
---|
| 1215 | continue; |
---|
| 1216 | } |
---|
| 1217 | |
---|
| 1218 | // bypass extra root separators |
---|
| 1219 | if ( in_root ) |
---|
| 1220 | { |
---|
| 1221 | if ( s.size() > 0 |
---|
| 1222 | && s[s.size()-1] == path_alt_separator<path_type>::value |
---|
| 1223 | && m_path[pos] == slash<path_type>::value |
---|
| 1224 | ) continue; |
---|
| 1225 | } |
---|
| 1226 | |
---|
| 1227 | if ( m_path[pos] == slash<path_type>::value ) |
---|
| 1228 | s += path_alt_separator<path_type>::value; |
---|
| 1229 | else |
---|
| 1230 | s += m_path[pos]; |
---|
| 1231 | |
---|
| 1232 | if ( pos > root_dir_start |
---|
| 1233 | && m_path[pos] == slash<path_type>::value ) |
---|
| 1234 | { in_root = false; } |
---|
| 1235 | } |
---|
| 1236 | # ifdef BOOST_CYGWIN_PATH |
---|
| 1237 | if ( m_cygwin_root ) s[0] = slash<path_type>::value; |
---|
| 1238 | # endif |
---|
| 1239 | return s; |
---|
| 1240 | # else |
---|
| 1241 | return m_path; |
---|
| 1242 | # endif |
---|
| 1243 | } |
---|
| 1244 | |
---|
| 1245 | // iterator functions ---------------------------------------------------// |
---|
| 1246 | |
---|
| 1247 | template<class String, class Traits> |
---|
| 1248 | typename basic_path<String, Traits>::iterator basic_path<String, Traits>::begin() const |
---|
| 1249 | { |
---|
| 1250 | iterator itr; |
---|
| 1251 | itr.m_path_ptr = this; |
---|
| 1252 | typename string_type::size_type element_size; |
---|
| 1253 | detail::first_element<String, Traits>( m_path, itr.m_pos, element_size ); |
---|
| 1254 | itr.m_name = m_path.substr( itr.m_pos, element_size ); |
---|
| 1255 | return itr; |
---|
| 1256 | } |
---|
| 1257 | |
---|
| 1258 | template<class String, class Traits> |
---|
| 1259 | typename basic_path<String, Traits>::iterator basic_path<String, Traits>::end() const |
---|
| 1260 | { |
---|
| 1261 | iterator itr; |
---|
| 1262 | itr.m_path_ptr = this; |
---|
| 1263 | itr.m_pos = m_path.size(); |
---|
| 1264 | return itr; |
---|
| 1265 | } |
---|
| 1266 | |
---|
| 1267 | namespace detail |
---|
| 1268 | { |
---|
| 1269 | // do_increment ------------------------------------------------------// |
---|
| 1270 | |
---|
| 1271 | template<class Path> |
---|
| 1272 | void iterator_helper<Path>::do_increment( iterator & itr ) |
---|
| 1273 | { |
---|
| 1274 | typedef typename Path::string_type string_type; |
---|
| 1275 | typedef typename Path::traits_type traits_type; |
---|
| 1276 | |
---|
| 1277 | assert( itr.m_pos < itr.m_path_ptr->m_path.size() && "basic_path::iterator increment past end()" ); |
---|
| 1278 | |
---|
| 1279 | bool was_net( itr.m_name.size() > 2 |
---|
| 1280 | && itr.m_name[0] == slash<Path>::value |
---|
| 1281 | && itr.m_name[1] == slash<Path>::value |
---|
| 1282 | && itr.m_name[2] != slash<Path>::value ); |
---|
| 1283 | |
---|
| 1284 | // increment to position past current element |
---|
| 1285 | itr.m_pos += itr.m_name.size(); |
---|
| 1286 | |
---|
| 1287 | // if end reached, create end iterator |
---|
| 1288 | if ( itr.m_pos == itr.m_path_ptr->m_path.size() ) |
---|
| 1289 | { |
---|
| 1290 | itr.m_name.erase( itr.m_name.begin(), itr.m_name.end() ); // VC++ 6.0 lib didn't supply clear() |
---|
| 1291 | return; |
---|
| 1292 | } |
---|
| 1293 | |
---|
| 1294 | // process separator (Windows drive spec is only case not a separator) |
---|
| 1295 | if ( itr.m_path_ptr->m_path[itr.m_pos] == slash<Path>::value ) |
---|
| 1296 | { |
---|
| 1297 | // detect root directory |
---|
| 1298 | if ( was_net |
---|
| 1299 | # ifdef BOOST_WINDOWS_PATH |
---|
| 1300 | // case "c:/" |
---|
| 1301 | || itr.m_name[itr.m_name.size()-1] == colon<Path>::value |
---|
| 1302 | # endif |
---|
| 1303 | ) |
---|
| 1304 | { |
---|
| 1305 | itr.m_name = slash<Path>::value; |
---|
| 1306 | return; |
---|
| 1307 | } |
---|
| 1308 | |
---|
| 1309 | // bypass separators |
---|
| 1310 | while ( itr.m_pos != itr.m_path_ptr->m_path.size() |
---|
| 1311 | && itr.m_path_ptr->m_path[itr.m_pos] == slash<Path>::value ) |
---|
| 1312 | { ++itr.m_pos; } |
---|
| 1313 | |
---|
| 1314 | // detect trailing separator, and treat it as ".", per POSIX spec |
---|
| 1315 | if ( itr.m_pos == itr.m_path_ptr->m_path.size() |
---|
| 1316 | && detail::is_non_root_slash< string_type, traits_type >( |
---|
| 1317 | itr.m_path_ptr->m_path, itr.m_pos-1 ) ) |
---|
| 1318 | { |
---|
| 1319 | --itr.m_pos; |
---|
| 1320 | itr.m_name = dot<Path>::value; |
---|
| 1321 | return; |
---|
| 1322 | } |
---|
| 1323 | } |
---|
| 1324 | |
---|
| 1325 | // get next element |
---|
| 1326 | typename string_type::size_type end_pos( |
---|
| 1327 | itr.m_path_ptr->m_path.find( slash<Path>::value, itr.m_pos ) ); |
---|
| 1328 | itr.m_name = itr.m_path_ptr->m_path.substr( itr.m_pos, end_pos - itr.m_pos ); |
---|
| 1329 | } |
---|
| 1330 | |
---|
| 1331 | // do_decrement ------------------------------------------------------// |
---|
| 1332 | |
---|
| 1333 | template<class Path> |
---|
| 1334 | void iterator_helper<Path>::do_decrement( iterator & itr ) |
---|
| 1335 | { |
---|
| 1336 | assert( itr.m_pos && "basic_path::iterator decrement past begin()" ); |
---|
| 1337 | |
---|
| 1338 | typedef typename Path::string_type string_type; |
---|
| 1339 | typedef typename Path::traits_type traits_type; |
---|
| 1340 | |
---|
| 1341 | typename string_type::size_type end_pos( itr.m_pos ); |
---|
| 1342 | |
---|
| 1343 | typename string_type::size_type root_dir_pos( |
---|
| 1344 | detail::root_directory_start<string_type, traits_type>( |
---|
| 1345 | itr.m_path_ptr->m_path, end_pos ) ); |
---|
| 1346 | |
---|
| 1347 | // if at end and there was a trailing non-root '/', return "." |
---|
| 1348 | if ( itr.m_pos == itr.m_path_ptr->m_path.size() |
---|
| 1349 | && itr.m_path_ptr->m_path.size() > 1 |
---|
| 1350 | && itr.m_path_ptr->m_path[itr.m_pos-1] == slash<Path>::value |
---|
| 1351 | && detail::is_non_root_slash< string_type, traits_type >( |
---|
| 1352 | itr.m_path_ptr->m_path, itr.m_pos-1 ) |
---|
| 1353 | ) |
---|
| 1354 | { |
---|
| 1355 | --itr.m_pos; |
---|
| 1356 | itr.m_name = dot<Path>::value; |
---|
| 1357 | return; |
---|
| 1358 | } |
---|
| 1359 | |
---|
| 1360 | // skip separators unless root directory |
---|
| 1361 | for ( |
---|
| 1362 | ; |
---|
| 1363 | end_pos > 0 |
---|
| 1364 | && (end_pos-1) != root_dir_pos |
---|
| 1365 | && itr.m_path_ptr->m_path[end_pos-1] == slash<Path>::value |
---|
| 1366 | ; |
---|
| 1367 | --end_pos ) {} |
---|
| 1368 | |
---|
| 1369 | itr.m_pos = detail::leaf_pos<string_type, traits_type> |
---|
| 1370 | ( itr.m_path_ptr->m_path, end_pos ); |
---|
| 1371 | itr.m_name = itr.m_path_ptr->m_path.substr( itr.m_pos, end_pos - itr.m_pos ); |
---|
| 1372 | } |
---|
| 1373 | } // namespace detail |
---|
| 1374 | |
---|
| 1375 | // basic_filesystem_error implementation --------------------------------// |
---|
| 1376 | |
---|
| 1377 | template<class Path> |
---|
| 1378 | basic_filesystem_error<Path>::basic_filesystem_error( |
---|
| 1379 | const std::string & what, system_error_type sys_err_code ) |
---|
| 1380 | : filesystem_error(what, sys_err_code) |
---|
| 1381 | { |
---|
| 1382 | try |
---|
| 1383 | { |
---|
| 1384 | m_imp_ptr.reset( new m_imp ); |
---|
| 1385 | } |
---|
| 1386 | catch (...) { m_imp_ptr.reset(); } |
---|
| 1387 | } |
---|
| 1388 | |
---|
| 1389 | template<class Path> |
---|
| 1390 | basic_filesystem_error<Path>::basic_filesystem_error( |
---|
| 1391 | const std::string & what, const path_type & path1, |
---|
| 1392 | system_error_type sys_err_code ) |
---|
| 1393 | : filesystem_error(what, sys_err_code) |
---|
| 1394 | { |
---|
| 1395 | try |
---|
| 1396 | { |
---|
| 1397 | m_imp_ptr.reset( new m_imp ); |
---|
| 1398 | m_imp_ptr->m_path1 = path1; |
---|
| 1399 | } |
---|
| 1400 | catch (...) { m_imp_ptr.reset(); } |
---|
| 1401 | } |
---|
| 1402 | |
---|
| 1403 | template<class Path> |
---|
| 1404 | basic_filesystem_error<Path>::basic_filesystem_error( |
---|
| 1405 | const std::string & what, const path_type & path1, |
---|
| 1406 | const path_type & path2, system_error_type sys_err_code ) |
---|
| 1407 | : filesystem_error(what, sys_err_code) |
---|
| 1408 | { |
---|
| 1409 | try |
---|
| 1410 | { |
---|
| 1411 | m_imp_ptr.reset( new m_imp ); |
---|
| 1412 | m_imp_ptr->m_path1 = path1; |
---|
| 1413 | m_imp_ptr->m_path2 = path2; |
---|
| 1414 | } |
---|
| 1415 | catch (...) { m_imp_ptr.reset(); } |
---|
| 1416 | } |
---|
| 1417 | |
---|
| 1418 | } // namespace BOOST_FILESYSTEM_NAMESPACE |
---|
| 1419 | } // namespace boost |
---|
| 1420 | |
---|
| 1421 | #include <boost/config/abi_suffix.hpp> // pops abi_prefix.hpp pragmas |
---|
| 1422 | |
---|
| 1423 | #endif // BOOST_FILESYSTEM_PATH_HPP |
---|