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