os/ossrv/ossrv_pub/boost_apis/boost/filesystem/operations.hpp
author sl@SLION-WIN7.fritz.box
Fri, 15 Jun 2012 03:10:57 +0200
changeset 0 bde4ae8d615e
permissions -rw-r--r--
First public contribution.
sl@0
     1
//  boost/filesystem/operations.hpp  -----------------------------------------//
sl@0
     2
sl@0
     3
//  Copyright 2002-2005 Beman Dawes
sl@0
     4
//  Copyright 2002 Jan Langer
sl@0
     5
//  Copyright 2001 Dietmar Kuehl                                        
sl@0
     6
//  
sl@0
     7
//  Use, modification, and distribution is subject to the Boost Software
sl@0
     8
//  License, Version 1.0. (See accompanying file LICENSE_1_0.txt or copy
sl@0
     9
//  at http://www.boost.org/LICENSE_1_0.txt)                             
sl@0
    10
sl@0
    11
//  See library home page at http://www.boost.org/libs/filesystem
sl@0
    12
sl@0
    13
//----------------------------------------------------------------------------// 
sl@0
    14
sl@0
    15
#ifndef BOOST_FILESYSTEM_OPERATIONS_HPP
sl@0
    16
#define BOOST_FILESYSTEM_OPERATIONS_HPP
sl@0
    17
sl@0
    18
#include <boost/filesystem/path.hpp>  // includes <boost/filesystem/config.hpp>
sl@0
    19
sl@0
    20
#include <boost/shared_ptr.hpp>
sl@0
    21
#include <boost/utility/enable_if.hpp>
sl@0
    22
#include <boost/type_traits/is_same.hpp>
sl@0
    23
#include <boost/iterator.hpp>
sl@0
    24
#include <boost/cstdint.hpp>
sl@0
    25
#include <boost/assert.hpp>
sl@0
    26
sl@0
    27
#include <string>
sl@0
    28
#include <utility> // for pair
sl@0
    29
#include <ctime>
sl@0
    30
sl@0
    31
#ifdef BOOST_WINDOWS_API
sl@0
    32
#  include <fstream>
sl@0
    33
#  if !defined(_WIN32_WINNT) || _WIN32_WINNT >= 0x0500
sl@0
    34
#    define BOOST_FS_HARD_LINK // Default for Windows 2K or later 
sl@0
    35
#  endif
sl@0
    36
#endif
sl@0
    37
sl@0
    38
#include <boost/config/abi_prefix.hpp> // must be the last #include
sl@0
    39
sl@0
    40
# ifdef BOOST_NO_STDC_NAMESPACE
sl@0
    41
    namespace std { using ::time_t; }
sl@0
    42
# endif
sl@0
    43
sl@0
    44
# ifndef BOOST_FILESYSTEM_NARROW_ONLY
sl@0
    45
#   define BOOST_FS_FUNC(BOOST_FS_TYPE) \
sl@0
    46
      template<class Path> typename boost::enable_if<is_basic_path<Path>, \
sl@0
    47
      BOOST_FS_TYPE>::type
sl@0
    48
#   define BOOST_INLINE_FS_FUNC(BOOST_FS_TYPE) \
sl@0
    49
      template<class Path> inline typename boost::enable_if<is_basic_path<Path>, \
sl@0
    50
      BOOST_FS_TYPE>::type
sl@0
    51
#   define BOOST_FS_TYPENAME typename
sl@0
    52
# else
sl@0
    53
#   define BOOST_FS_FUNC(BOOST_FS_TYPE) inline BOOST_FS_TYPE
sl@0
    54
#   define BOOST_INLINE_FS_FUNC(BOOST_FS_TYPE) inline BOOST_FS_TYPE
sl@0
    55
    typedef boost::filesystem::path Path;
sl@0
    56
#   define BOOST_FS_TYPENAME
sl@0
    57
# endif
sl@0
    58
sl@0
    59
//----------------------------------------------------------------------------//
sl@0
    60
sl@0
    61
namespace boost
sl@0
    62
{
sl@0
    63
  namespace filesystem
sl@0
    64
  {
sl@0
    65
    template<class Path> class basic_directory_iterator;
sl@0
    66
sl@0
    67
    // BOOST_FILESYSTEM_NARROW_ONLY needs this:
sl@0
    68
    typedef basic_directory_iterator<path> directory_iterator;
sl@0
    69
sl@0
    70
    template<class Path> class basic_directory_entry;
sl@0
    71
sl@0
    72
    enum file_type
sl@0
    73
    { 
sl@0
    74
      status_unknown,
sl@0
    75
      file_not_found,
sl@0
    76
      regular_file,
sl@0
    77
      directory_file,
sl@0
    78
      // the following will never be reported by some operating or file systems
sl@0
    79
      symlink_file,
sl@0
    80
      block_file,
sl@0
    81
      character_file,
sl@0
    82
      fifo_file,
sl@0
    83
      socket_file,
sl@0
    84
      type_unknown // file does exist, but isn't one of the above types
sl@0
    85
    };
sl@0
    86
sl@0
    87
    class file_status
sl@0
    88
    {
sl@0
    89
    public:
sl@0
    90
      explicit file_status( file_type v = status_unknown ) : m_value(v) {}
sl@0
    91
sl@0
    92
      void type( file_type v )  { m_value = v; }
sl@0
    93
      file_type type() const    { return m_value; }
sl@0
    94
sl@0
    95
    private:
sl@0
    96
      // the internal representation is unspecified so that additional state
sl@0
    97
      // information such as permissions can be added in the future; this
sl@0
    98
      // implementation just uses status_type as the internal representation
sl@0
    99
sl@0
   100
      file_type m_value;
sl@0
   101
    };
sl@0
   102
sl@0
   103
    inline bool status_known( file_status f ) { return f.type() != status_unknown; }
sl@0
   104
    inline bool exists( file_status f )       { return f.type() != status_unknown && f.type() != file_not_found; }
sl@0
   105
    inline bool is_regular( file_status f )   { return f.type() == regular_file; }
sl@0
   106
    inline bool is_directory( file_status f ) { return f.type() == directory_file; }
sl@0
   107
    inline bool is_symlink( file_status f )   { return f.type() == symlink_file; }
sl@0
   108
    inline bool is_other( file_status f )     { return exists(f) && !is_regular(f) && !is_directory(f) && !is_symlink(f); }
sl@0
   109
sl@0
   110
    struct space_info
sl@0
   111
    {
sl@0
   112
      // all values are byte counts
sl@0
   113
      boost::uintmax_t capacity;
sl@0
   114
      boost::uintmax_t free;      // <= capacity
sl@0
   115
      boost::uintmax_t available; // <= free
sl@0
   116
    };
sl@0
   117
sl@0
   118
    namespace detail
sl@0
   119
    {
sl@0
   120
      typedef std::pair< boost::filesystem::system_error_type, bool >
sl@0
   121
        query_pair;
sl@0
   122
sl@0
   123
      typedef std::pair< boost::filesystem::system_error_type, boost::uintmax_t >
sl@0
   124
        uintmax_pair;
sl@0
   125
sl@0
   126
      typedef std::pair< boost::filesystem::system_error_type, std::time_t >
sl@0
   127
        time_pair;
sl@0
   128
sl@0
   129
      typedef std::pair< boost::filesystem::system_error_type, space_info >
sl@0
   130
        space_pair;
sl@0
   131
sl@0
   132
      template< class Path >
sl@0
   133
      struct directory_pair
sl@0
   134
      {
sl@0
   135
        typedef std::pair< boost::filesystem::system_error_type,
sl@0
   136
          typename Path::external_string_type > type;
sl@0
   137
      };
sl@0
   138
sl@0
   139
#   ifndef BOOST_FILESYSTEM_NO_DEPRECATED
sl@0
   140
      BOOST_FILESYSTEM_DECL bool
sl@0
   141
        symbolic_link_exists_api( const std::string & ); // deprecated
sl@0
   142
#   endif
sl@0
   143
sl@0
   144
      BOOST_FILESYSTEM_DECL file_status
sl@0
   145
        status_api( const std::string & ph, system_error_type & ec );
sl@0
   146
#   ifndef BOOST_WINDOWS_API
sl@0
   147
      BOOST_FILESYSTEM_DECL file_status
sl@0
   148
        symlink_status_api( const std::string & ph, system_error_type & ec );
sl@0
   149
#   endif
sl@0
   150
      BOOST_FILESYSTEM_DECL query_pair
sl@0
   151
        is_empty_api( const std::string & ph );
sl@0
   152
      BOOST_FILESYSTEM_DECL query_pair
sl@0
   153
        equivalent_api( const std::string & ph1, const std::string & ph2 );
sl@0
   154
      BOOST_FILESYSTEM_DECL uintmax_pair
sl@0
   155
        file_size_api( const std::string & ph );
sl@0
   156
      BOOST_FILESYSTEM_DECL space_pair
sl@0
   157
        space_api( const std::string & ph );
sl@0
   158
      BOOST_FILESYSTEM_DECL time_pair 
sl@0
   159
        last_write_time_api( const std::string & ph );
sl@0
   160
      BOOST_FILESYSTEM_DECL boost::filesystem::system_error_type
sl@0
   161
        last_write_time_api( const std::string & ph, std::time_t new_value );
sl@0
   162
      BOOST_FILESYSTEM_DECL boost::filesystem::system_error_type
sl@0
   163
        get_current_path_api( std::string & ph );
sl@0
   164
      BOOST_FILESYSTEM_DECL query_pair
sl@0
   165
        create_directory_api( const std::string & ph );
sl@0
   166
      BOOST_FILESYSTEM_DECL boost::filesystem::system_error_type
sl@0
   167
        create_hard_link_api( const std::string & to_ph,
sl@0
   168
          const std::string & from_ph );
sl@0
   169
      BOOST_FILESYSTEM_DECL boost::filesystem::system_error_type
sl@0
   170
        create_symlink_api( const std::string & to_ph,
sl@0
   171
          const std::string & from_ph );
sl@0
   172
      BOOST_FILESYSTEM_DECL boost::filesystem::system_error_type
sl@0
   173
        remove_api( const std::string & ph );
sl@0
   174
      BOOST_FILESYSTEM_DECL boost::filesystem::system_error_type
sl@0
   175
        rename_api( const std::string & from, const std::string & to );
sl@0
   176
      BOOST_FILESYSTEM_DECL boost::filesystem::system_error_type
sl@0
   177
        copy_file_api( const std::string & from, const std::string & to );
sl@0
   178
sl@0
   179
#   if defined(BOOST_WINDOWS_API)
sl@0
   180
      
sl@0
   181
      BOOST_FILESYSTEM_DECL boost::filesystem::system_error_type
sl@0
   182
        get_full_path_name_api( const std::string & ph, std::string & target );
sl@0
   183
sl@0
   184
#     if !defined(BOOST_FILESYSTEM_NARROW_ONLY)
sl@0
   185
sl@0
   186
      BOOST_FILESYSTEM_DECL  boost::filesystem::file_status
sl@0
   187
        status_api( const std::wstring & ph, system_error_type & ec );
sl@0
   188
      BOOST_FILESYSTEM_DECL query_pair 
sl@0
   189
        is_empty_api( const std::wstring & ph );
sl@0
   190
      BOOST_FILESYSTEM_DECL query_pair
sl@0
   191
        equivalent_api( const std::wstring & ph1, const std::wstring & ph2 );
sl@0
   192
      BOOST_FILESYSTEM_DECL uintmax_pair 
sl@0
   193
        file_size_api( const std::wstring & ph );
sl@0
   194
      BOOST_FILESYSTEM_DECL space_pair 
sl@0
   195
        space_api( const std::wstring & ph );
sl@0
   196
      BOOST_FILESYSTEM_DECL boost::filesystem::system_error_type
sl@0
   197
        get_full_path_name_api( const std::wstring & ph, std::wstring & target );
sl@0
   198
      BOOST_FILESYSTEM_DECL time_pair 
sl@0
   199
        last_write_time_api( const std::wstring & ph );
sl@0
   200
      BOOST_FILESYSTEM_DECL boost::filesystem::system_error_type
sl@0
   201
        last_write_time_api( const std::wstring & ph, std::time_t new_value );
sl@0
   202
      BOOST_FILESYSTEM_DECL boost::filesystem::system_error_type 
sl@0
   203
        get_current_path_api( std::wstring & ph );
sl@0
   204
      BOOST_FILESYSTEM_DECL query_pair
sl@0
   205
        create_directory_api( const std::wstring & ph );
sl@0
   206
# ifdef BOOST_FS_HARD_LINK
sl@0
   207
      BOOST_FILESYSTEM_DECL boost::filesystem::system_error_type
sl@0
   208
        create_hard_link_api( const std::wstring & existing_ph,
sl@0
   209
          const std::wstring & new_ph );
sl@0
   210
# endif
sl@0
   211
      BOOST_FILESYSTEM_DECL boost::filesystem::system_error_type
sl@0
   212
        create_symlink_api( const std::wstring & to_ph,
sl@0
   213
          const std::wstring & from_ph );
sl@0
   214
      BOOST_FILESYSTEM_DECL boost::filesystem::system_error_type
sl@0
   215
        remove_api( const std::wstring & ph );
sl@0
   216
      BOOST_FILESYSTEM_DECL boost::filesystem::system_error_type
sl@0
   217
        rename_api( const std::wstring & from, const std::wstring & to );
sl@0
   218
      BOOST_FILESYSTEM_DECL boost::filesystem::system_error_type
sl@0
   219
        copy_file_api( const std::wstring & from, const std::wstring & to );
sl@0
   220
sl@0
   221
#     endif
sl@0
   222
#   endif
sl@0
   223
sl@0
   224
      template<class Path>
sl@0
   225
      unsigned long remove_all_aux( const Path & ph );
sl@0
   226
sl@0
   227
    } // namespace detail
sl@0
   228
sl@0
   229
//  operations functions  ----------------------------------------------------//
sl@0
   230
sl@0
   231
    //  The non-template overloads enable automatic conversion from std and
sl@0
   232
    //  C-style strings. See basic_path constructors. The enable_if for the
sl@0
   233
    //  templates implements the famous "do-the-right-thing" rule.
sl@0
   234
sl@0
   235
//  query functions  ---------------------------------------------------------//
sl@0
   236
sl@0
   237
    BOOST_INLINE_FS_FUNC(file_status)
sl@0
   238
    status( const Path & ph, system_error_type & ec )
sl@0
   239
      { return detail::status_api( ph.external_file_string(), ec ); }
sl@0
   240
sl@0
   241
    BOOST_FS_FUNC(file_status)
sl@0
   242
    status( const Path & ph )
sl@0
   243
    { 
sl@0
   244
      system_error_type ec;
sl@0
   245
      file_status result( detail::status_api( ph.external_file_string(), ec ) );
sl@0
   246
      if ( ec )
sl@0
   247
        boost::throw_exception( basic_filesystem_error<Path>(
sl@0
   248
        "boost::filesystem::status", ph, ec ) );
sl@0
   249
      return result;
sl@0
   250
    }
sl@0
   251
sl@0
   252
    BOOST_INLINE_FS_FUNC(file_status)
sl@0
   253
    symlink_status( const Path & ph, system_error_type & ec )
sl@0
   254
#   ifdef BOOST_WINDOWS_API
sl@0
   255
      { return detail::status_api( ph.external_file_string(), ec ); }
sl@0
   256
#   else
sl@0
   257
      { return detail::symlink_status_api( ph.external_file_string(), ec ); }
sl@0
   258
#   endif
sl@0
   259
sl@0
   260
    BOOST_FS_FUNC(file_status)
sl@0
   261
    symlink_status( const Path & ph )
sl@0
   262
    { 
sl@0
   263
      system_error_type ec;
sl@0
   264
      file_status result( symlink_status( ph, ec ) );
sl@0
   265
      if ( ec )
sl@0
   266
        boost::throw_exception( basic_filesystem_error<Path>(
sl@0
   267
        "boost::filesystem::symlink_status", ph, ec ) );
sl@0
   268
      return result;
sl@0
   269
    }
sl@0
   270
sl@0
   271
# ifndef BOOST_FILESYSTEM_NO_DEPRECATED
sl@0
   272
    inline bool symbolic_link_exists( const path & ph )
sl@0
   273
      { return is_symlink( symlink_status(ph) ); }
sl@0
   274
#endif
sl@0
   275
sl@0
   276
    BOOST_FS_FUNC(bool) exists( const Path & ph )
sl@0
   277
    { 
sl@0
   278
      system_error_type ec;
sl@0
   279
      file_status result( detail::status_api( ph.external_file_string(), ec ) );
sl@0
   280
      if ( ec )
sl@0
   281
        boost::throw_exception( basic_filesystem_error<Path>(
sl@0
   282
          "boost::filesystem::exists", ph, ec ) );
sl@0
   283
      return exists( result );
sl@0
   284
    }
sl@0
   285
sl@0
   286
    BOOST_FS_FUNC(bool) is_directory( const Path & ph )
sl@0
   287
    { 
sl@0
   288
      system_error_type ec;
sl@0
   289
      file_status result( detail::status_api( ph.external_file_string(), ec ) );
sl@0
   290
      if ( ec )
sl@0
   291
        boost::throw_exception( basic_filesystem_error<Path>(
sl@0
   292
          "boost::filesystem::is_directory", ph, ec ) );
sl@0
   293
      return is_directory( result );
sl@0
   294
    }
sl@0
   295
sl@0
   296
    BOOST_FS_FUNC(bool) is_regular( const Path & ph )
sl@0
   297
    { 
sl@0
   298
      system_error_type ec;
sl@0
   299
      file_status result( detail::status_api( ph.external_file_string(), ec ) );
sl@0
   300
      if ( ec )
sl@0
   301
        boost::throw_exception( basic_filesystem_error<Path>(
sl@0
   302
          "boost::filesystem::is_regular", ph, ec ) );
sl@0
   303
      return is_regular( result );
sl@0
   304
    }
sl@0
   305
sl@0
   306
    BOOST_FS_FUNC(bool) is_other( const Path & ph )
sl@0
   307
    { 
sl@0
   308
      system_error_type ec;
sl@0
   309
      file_status result( detail::status_api( ph.external_file_string(), ec ) );
sl@0
   310
      if ( ec )
sl@0
   311
        boost::throw_exception( basic_filesystem_error<Path>(
sl@0
   312
          "boost::filesystem::is_other", ph, ec ) );
sl@0
   313
      return is_other( result );
sl@0
   314
    }
sl@0
   315
sl@0
   316
    BOOST_FS_FUNC(bool) is_symlink(
sl@0
   317
#   ifdef BOOST_WINDOWS_API
sl@0
   318
      const Path & )
sl@0
   319
    {
sl@0
   320
      return false;
sl@0
   321
#   else
sl@0
   322
      const Path & ph)
sl@0
   323
    {
sl@0
   324
      system_error_type ec;
sl@0
   325
      file_status result( detail::symlink_status_api( ph.external_file_string(), ec ) );
sl@0
   326
      if ( ec )
sl@0
   327
        boost::throw_exception( basic_filesystem_error<Path>(
sl@0
   328
          "boost::filesystem::is_symlink", ph, ec ) );
sl@0
   329
      return is_symlink( result );
sl@0
   330
#   endif
sl@0
   331
    }
sl@0
   332
sl@0
   333
    // VC++ 7.0 and earlier has a serious namespace bug that causes a clash
sl@0
   334
    // between boost::filesystem::is_empty and the unrelated type trait
sl@0
   335
    // boost::is_empty.
sl@0
   336
sl@0
   337
# if !defined( BOOST_MSVC ) || BOOST_MSVC > 1300
sl@0
   338
    BOOST_FS_FUNC(bool) is_empty( const Path & ph )
sl@0
   339
# else
sl@0
   340
    BOOST_FS_FUNC(bool) _is_empty( const Path & ph )
sl@0
   341
# endif
sl@0
   342
    {
sl@0
   343
      detail::query_pair result = detail::is_empty_api( ph.external_file_string() );
sl@0
   344
      if ( result.first != 0 )
sl@0
   345
        boost::throw_exception( basic_filesystem_error<Path>(
sl@0
   346
          "boost::filesystem::is_empty", ph, result.first ) );
sl@0
   347
      return result.second;
sl@0
   348
    }
sl@0
   349
sl@0
   350
    BOOST_FS_FUNC(bool) equivalent( const Path & ph1, const Path & ph2 )
sl@0
   351
    {
sl@0
   352
      detail::query_pair result = detail::equivalent_api(
sl@0
   353
        ph1.external_file_string(), ph2.external_file_string() );
sl@0
   354
      if ( result.first != 0 )
sl@0
   355
        boost::throw_exception( basic_filesystem_error<Path>(
sl@0
   356
          "boost::filesystem::equivalent", ph1, ph2, result.first ) );
sl@0
   357
      return result.second;
sl@0
   358
    }
sl@0
   359
sl@0
   360
    BOOST_FS_FUNC(boost::uintmax_t) file_size( const Path & ph )
sl@0
   361
    {
sl@0
   362
      detail::uintmax_pair result
sl@0
   363
        = detail::file_size_api( ph.external_file_string() );
sl@0
   364
      if ( result.first != 0 )
sl@0
   365
        boost::throw_exception( basic_filesystem_error<Path>(
sl@0
   366
          "boost::filesystem::file_size", ph, result.first ) );
sl@0
   367
      return result.second;
sl@0
   368
    }
sl@0
   369
sl@0
   370
    BOOST_FS_FUNC(space_info) space( const Path & ph )
sl@0
   371
    {
sl@0
   372
      detail::space_pair result
sl@0
   373
        = detail::space_api( ph.external_file_string() );
sl@0
   374
      if ( result.first != 0 )
sl@0
   375
        boost::throw_exception( basic_filesystem_error<Path>(
sl@0
   376
          "boost::filesystem::space", ph, result.first ) );
sl@0
   377
      return result.second;
sl@0
   378
    }
sl@0
   379
sl@0
   380
    BOOST_FS_FUNC(std::time_t) last_write_time( const Path & ph )
sl@0
   381
    {
sl@0
   382
      detail::time_pair result
sl@0
   383
        = detail::last_write_time_api( ph.external_file_string() );
sl@0
   384
      if ( result.first != 0 )
sl@0
   385
        boost::throw_exception( basic_filesystem_error<Path>(
sl@0
   386
          "boost::filesystem::last_write_time", ph, result.first ) );
sl@0
   387
      return result.second;
sl@0
   388
    }
sl@0
   389
sl@0
   390
sl@0
   391
//  operations  --------------------------------------------------------------//
sl@0
   392
sl@0
   393
    BOOST_FS_FUNC(bool) create_directory( const Path & dir_ph )
sl@0
   394
    {
sl@0
   395
      detail::query_pair result(
sl@0
   396
        detail::create_directory_api( dir_ph.external_directory_string() ) );
sl@0
   397
      if ( result.first != 0 )
sl@0
   398
        boost::throw_exception( basic_filesystem_error<Path>(
sl@0
   399
          "boost::filesystem::create_directory",
sl@0
   400
          dir_ph, result.first ) );
sl@0
   401
      return result.second;
sl@0
   402
    }
sl@0
   403
sl@0
   404
#if !defined(BOOST_WINDOWS_API) || defined(BOOST_FS_HARD_LINK)
sl@0
   405
    BOOST_FS_FUNC(void)
sl@0
   406
    create_hard_link( const Path & to_ph, const Path & from_ph )
sl@0
   407
    {
sl@0
   408
      system_error_type result( 
sl@0
   409
        detail::create_hard_link_api(
sl@0
   410
          to_ph.external_file_string(),
sl@0
   411
          from_ph.external_file_string() ) );
sl@0
   412
      if ( result != 0 )
sl@0
   413
        boost::throw_exception( basic_filesystem_error<Path>(
sl@0
   414
          "boost::filesystem::create_hard_link",
sl@0
   415
          to_ph, from_ph, result ) );
sl@0
   416
    }
sl@0
   417
sl@0
   418
    BOOST_FS_FUNC(system_error_type)
sl@0
   419
    create_hard_link( const Path & to_ph, const Path & from_ph,
sl@0
   420
      system_error_type & ec )
sl@0
   421
    {
sl@0
   422
      ec = detail::create_hard_link_api(
sl@0
   423
            to_ph.external_file_string(),
sl@0
   424
            from_ph.external_file_string() );
sl@0
   425
      return ec;
sl@0
   426
    }
sl@0
   427
#endif
sl@0
   428
sl@0
   429
    BOOST_FS_FUNC(void)
sl@0
   430
    create_symlink( const Path & to_ph, const Path & from_ph )
sl@0
   431
    {
sl@0
   432
      system_error_type result( 
sl@0
   433
        detail::create_symlink_api(
sl@0
   434
          to_ph.external_file_string(),
sl@0
   435
          from_ph.external_file_string() ) );
sl@0
   436
      if ( result )
sl@0
   437
        boost::throw_exception( basic_filesystem_error<Path>(
sl@0
   438
          "boost::filesystem::create_symlink",
sl@0
   439
          to_ph, from_ph, result ) );
sl@0
   440
    }
sl@0
   441
sl@0
   442
    BOOST_FS_FUNC(system_error_type)
sl@0
   443
    create_symlink( const Path & to_ph, const Path & from_ph,
sl@0
   444
      system_error_type & ec )
sl@0
   445
    {
sl@0
   446
      ec = detail::create_symlink_api(
sl@0
   447
             to_ph.external_file_string(),
sl@0
   448
             from_ph.external_file_string() );
sl@0
   449
      return ec;
sl@0
   450
    }
sl@0
   451
sl@0
   452
    BOOST_FS_FUNC(bool) remove( const Path & ph )
sl@0
   453
    {
sl@0
   454
      if ( exists( ph )
sl@0
   455
        || is_symlink( ph ) ) // handle dangling symbolic links
sl@0
   456
        // note that the POSIX behavior for symbolic links is what we want;
sl@0
   457
        // the link rather than what it points to is deleted. Windows behavior
sl@0
   458
        // doesn't matter; is_symlink() is always false on Windows.
sl@0
   459
      {
sl@0
   460
        system_error_type result = detail::remove_api( ph.external_file_string() );
sl@0
   461
        if ( result != 0 )
sl@0
   462
          boost::throw_exception( basic_filesystem_error<Path>(
sl@0
   463
            "boost::filesystem::remove",
sl@0
   464
            ph, result ) );
sl@0
   465
        return true;
sl@0
   466
      }
sl@0
   467
      return false;
sl@0
   468
    }
sl@0
   469
sl@0
   470
    BOOST_FS_FUNC(unsigned long) remove_all( const Path & ph )
sl@0
   471
    {
sl@0
   472
      return exists( ph )|| is_symlink( ph )
sl@0
   473
        ? detail::remove_all_aux( ph ) : 0;
sl@0
   474
    }
sl@0
   475
sl@0
   476
    BOOST_FS_FUNC(void) rename( const Path & from_path, const Path & to_path )
sl@0
   477
    {
sl@0
   478
      system_error_type result = detail::rename_api(
sl@0
   479
        from_path.external_directory_string(),
sl@0
   480
        to_path.external_directory_string() );
sl@0
   481
      if ( result != 0 )
sl@0
   482
        boost::throw_exception( basic_filesystem_error<Path>(
sl@0
   483
          "boost::filesystem::rename",
sl@0
   484
          from_path, to_path, result ) );
sl@0
   485
    }
sl@0
   486
sl@0
   487
    BOOST_FS_FUNC(void) copy_file( const Path & from_path, const Path & to_path )
sl@0
   488
    {
sl@0
   489
      system_error_type result = detail::copy_file_api(
sl@0
   490
        from_path.external_directory_string(),
sl@0
   491
        to_path.external_directory_string() );
sl@0
   492
      if ( result != 0 )
sl@0
   493
        boost::throw_exception( basic_filesystem_error<Path>(
sl@0
   494
          "boost::filesystem::copy_file",
sl@0
   495
          from_path, to_path, result ) );
sl@0
   496
    }
sl@0
   497
sl@0
   498
    template< class Path >
sl@0
   499
    Path current_path()
sl@0
   500
    {
sl@0
   501
      typename Path::external_string_type ph;
sl@0
   502
      boost::filesystem::system_error_type result;
sl@0
   503
      if ( (result = detail::get_current_path_api( ph )) != 0 )
sl@0
   504
          boost::throw_exception( basic_filesystem_error<Path>(
sl@0
   505
            "boost::filesystem::current_path", result ) );
sl@0
   506
      return Path( Path::traits_type::to_internal( ph ) );
sl@0
   507
    }
sl@0
   508
sl@0
   509
    template< class Path >
sl@0
   510
    const Path & initial_path()
sl@0
   511
    {
sl@0
   512
      static Path init_path;
sl@0
   513
      if ( init_path.empty() ) init_path = current_path<Path>();
sl@0
   514
      return init_path;
sl@0
   515
    }
sl@0
   516
sl@0
   517
# ifndef BOOST_FILESYSTEM_NO_DEPRECATED
sl@0
   518
    // legacy support
sl@0
   519
    inline path current_path()  // overload supports pre-i18n apps
sl@0
   520
      { return current_path<boost::filesystem::path>(); }
sl@0
   521
    inline const path & initial_path() // overload supports pre-i18n apps
sl@0
   522
      { return initial_path<boost::filesystem::path>(); }
sl@0
   523
# endif
sl@0
   524
sl@0
   525
    BOOST_FS_FUNC(Path) system_complete( const Path & ph )
sl@0
   526
    {
sl@0
   527
# ifdef BOOST_WINDOWS_API
sl@0
   528
      if ( ph.empty() ) return ph;
sl@0
   529
      BOOST_FS_TYPENAME Path::external_string_type sys_ph;
sl@0
   530
      boost::filesystem::system_error_type result;
sl@0
   531
      if ( (result = detail::get_full_path_name_api( ph.external_file_string(),
sl@0
   532
              sys_ph )) != 0 )
sl@0
   533
          boost::throw_exception( basic_filesystem_error<Path>(
sl@0
   534
            "boost::filesystem::system_complete", ph, result ) );
sl@0
   535
      return Path( Path::traits_type::to_internal( sys_ph ) );
sl@0
   536
# else
sl@0
   537
      return (ph.empty() || ph.is_complete())
sl@0
   538
        ? ph : current_path<Path>() / ph;
sl@0
   539
# endif
sl@0
   540
    }
sl@0
   541
sl@0
   542
    BOOST_FS_FUNC(Path)
sl@0
   543
    complete( const Path & ph,
sl@0
   544
      const Path & base/* = initial_path<Path>() */)
sl@0
   545
    {
sl@0
   546
      BOOST_ASSERT( base.is_complete()
sl@0
   547
        && (ph.is_complete() || !ph.has_root_name())
sl@0
   548
        && "boost::filesystem::complete() precondition not met" );
sl@0
   549
#   ifdef BOOST_WINDOWS_PATH
sl@0
   550
      if (ph.empty() || ph.is_complete()) return ph;
sl@0
   551
      if ( !ph.has_root_name() )
sl@0
   552
        return ph.has_root_directory()
sl@0
   553
          ? Path( base.root_name() ) / ph
sl@0
   554
          : base / ph;
sl@0
   555
      return base / ph;
sl@0
   556
#   else
sl@0
   557
      return (ph.empty() || ph.is_complete()) ? ph : base / ph;
sl@0
   558
#   endif
sl@0
   559
    }
sl@0
   560
sl@0
   561
    // VC++ 7.1 had trouble with default arguments, so separate one argument
sl@0
   562
    // signatures are provided as workarounds; the effect is the same.
sl@0
   563
    BOOST_FS_FUNC(Path) complete( const Path & ph )
sl@0
   564
      { return complete( ph, initial_path<Path>() ); }
sl@0
   565
sl@0
   566
    BOOST_FS_FUNC(void)
sl@0
   567
    last_write_time( const Path & ph, const std::time_t new_time )
sl@0
   568
    {
sl@0
   569
      boost::filesystem::system_error_type result;
sl@0
   570
      if ( (result = detail::last_write_time_api( ph.external_file_string(),
sl@0
   571
          new_time )) != 0 )
sl@0
   572
        boost::throw_exception( basic_filesystem_error<Path>(
sl@0
   573
          "boost::filesystem::last_write_time", ph, result ) );
sl@0
   574
    }
sl@0
   575
sl@0
   576
# ifndef BOOST_FILESYSTEM_NARROW_ONLY
sl@0
   577
sl@0
   578
    // "do-the-right-thing" overloads  ---------------------------------------//
sl@0
   579
sl@0
   580
    inline file_status status( const path & ph )
sl@0
   581
      { return status<path>( ph ); }
sl@0
   582
    inline file_status status( const wpath & ph )
sl@0
   583
      { return status<wpath>( ph ); }
sl@0
   584
sl@0
   585
    inline file_status status( const path & ph, system_error_type & ec )
sl@0
   586
      { return status<path>( ph, ec ); }
sl@0
   587
    inline file_status status( const wpath & ph, system_error_type & ec )
sl@0
   588
      { return status<wpath>( ph, ec ); }
sl@0
   589
sl@0
   590
    inline file_status symlink_status( const path & ph )
sl@0
   591
      { return symlink_status<path>( ph ); }
sl@0
   592
    inline file_status symlink_status( const wpath & ph )
sl@0
   593
      { return symlink_status<wpath>( ph ); }
sl@0
   594
sl@0
   595
    inline file_status symlink_status( const path & ph, system_error_type & ec )
sl@0
   596
      { return symlink_status<path>( ph, ec ); }
sl@0
   597
    inline file_status symlink_status( const wpath & ph, system_error_type & ec )
sl@0
   598
      { return symlink_status<wpath>( ph, ec ); }
sl@0
   599
sl@0
   600
    inline bool exists( const path & ph ) { return exists<path>( ph ); }
sl@0
   601
    inline bool exists( const wpath & ph ) { return exists<wpath>( ph ); }
sl@0
   602
sl@0
   603
    inline bool is_directory( const path & ph )
sl@0
   604
      { return is_directory<path>( ph ); }
sl@0
   605
    inline bool is_directory( const wpath & ph )
sl@0
   606
      { return is_directory<wpath>( ph ); }
sl@0
   607
 
sl@0
   608
    inline bool is_regular( const path & ph )
sl@0
   609
      { return is_regular<path>( ph ); }
sl@0
   610
    inline bool is_regular( const wpath & ph )
sl@0
   611
      { return is_regular<wpath>( ph ); }
sl@0
   612
sl@0
   613
    inline bool is_other( const path & ph )
sl@0
   614
      { return is_other<path>( ph ); }
sl@0
   615
    inline bool is_other( const wpath & ph )
sl@0
   616
      { return is_other<wpath>( ph ); }
sl@0
   617
sl@0
   618
    inline bool is_symlink( const path & ph )
sl@0
   619
      { return is_symlink<path>( ph ); }
sl@0
   620
    inline bool is_symlink( const wpath & ph )
sl@0
   621
      { return is_symlink<wpath>( ph ); }
sl@0
   622
sl@0
   623
    inline bool is_empty( const path & ph )
sl@0
   624
      { return is_empty<path>( ph ); }
sl@0
   625
    inline bool is_empty( const wpath & ph )
sl@0
   626
      { return is_empty<wpath>( ph ); }
sl@0
   627
sl@0
   628
    inline bool equivalent( const path & ph1, const path & ph2 )
sl@0
   629
      { return equivalent<path>( ph1, ph2 ); }
sl@0
   630
    inline bool equivalent( const wpath & ph1, const wpath & ph2 )
sl@0
   631
      { return equivalent<wpath>( ph1, ph2 ); }
sl@0
   632
sl@0
   633
    inline boost::uintmax_t file_size( const path & ph )
sl@0
   634
      { return file_size<path>( ph ); }
sl@0
   635
    inline boost::uintmax_t file_size( const wpath & ph )
sl@0
   636
      { return file_size<wpath>( ph ); }
sl@0
   637
sl@0
   638
    inline space_info space( const path & ph )
sl@0
   639
      { return space<path>( ph ); }
sl@0
   640
    inline space_info space( const wpath & ph )
sl@0
   641
      { return space<wpath>( ph ); }
sl@0
   642
sl@0
   643
    inline std::time_t last_write_time( const path & ph )
sl@0
   644
      { return last_write_time<path>( ph ); }
sl@0
   645
    inline std::time_t last_write_time( const wpath & ph )
sl@0
   646
      { return last_write_time<wpath>( ph ); }
sl@0
   647
sl@0
   648
    inline bool create_directory( const path & dir_ph )
sl@0
   649
      { return create_directory<path>( dir_ph ); }
sl@0
   650
    inline bool create_directory( const wpath & dir_ph )
sl@0
   651
      { return create_directory<wpath>( dir_ph ); }
sl@0
   652
sl@0
   653
#if !defined(BOOST_WINDOWS_API) || defined(BOOST_FS_HARD_LINK)
sl@0
   654
    inline void create_hard_link( const path & to_ph,
sl@0
   655
      const path & from_ph )
sl@0
   656
      { return create_hard_link<path>( to_ph, from_ph ); }
sl@0
   657
    inline void create_hard_link( const wpath & to_ph,
sl@0
   658
      const wpath & from_ph )
sl@0
   659
      { return create_hard_link<wpath>( to_ph, from_ph ); }
sl@0
   660
sl@0
   661
    inline system_error_type create_hard_link( const path & to_ph,
sl@0
   662
      const path & from_ph, system_error_type & ec )
sl@0
   663
      { return create_hard_link<path>( to_ph, from_ph, ec ); }
sl@0
   664
    inline system_error_type create_hard_link( const wpath & to_ph,
sl@0
   665
      const wpath & from_ph, system_error_type & ec )
sl@0
   666
      { return create_hard_link<wpath>( to_ph, from_ph, ec ); }
sl@0
   667
#endif
sl@0
   668
    
sl@0
   669
    inline void create_symlink( const path & to_ph,
sl@0
   670
      const path & from_ph )
sl@0
   671
      { return create_symlink<path>( to_ph, from_ph ); }
sl@0
   672
    inline void create_symlink( const wpath & to_ph,
sl@0
   673
      const wpath & from_ph )
sl@0
   674
      { return create_symlink<wpath>( to_ph, from_ph ); }
sl@0
   675
sl@0
   676
    inline system_error_type create_symlink( const path & to_ph,
sl@0
   677
      const path & from_ph, system_error_type & ec )
sl@0
   678
      { return create_symlink<path>( to_ph, from_ph, ec ); }
sl@0
   679
    inline system_error_type create_symlink( const wpath & to_ph,
sl@0
   680
      const wpath & from_ph, system_error_type & ec )
sl@0
   681
      { return create_symlink<wpath>( to_ph, from_ph, ec ); }
sl@0
   682
sl@0
   683
    inline bool remove( const path & ph )
sl@0
   684
      { return remove<path>( ph ); }
sl@0
   685
    inline bool remove( const wpath & ph )
sl@0
   686
      { return remove<wpath>( ph ); }
sl@0
   687
sl@0
   688
    inline unsigned long remove_all( const path & ph )
sl@0
   689
      { return remove_all<path>( ph ); }
sl@0
   690
    inline unsigned long remove_all( const wpath & ph )
sl@0
   691
      { return remove_all<wpath>( ph ); }
sl@0
   692
sl@0
   693
    inline void rename( const path & from_path, const path & to_path )
sl@0
   694
      { return rename<path>( from_path, to_path ); }
sl@0
   695
    inline void rename( const wpath & from_path, const wpath & to_path )
sl@0
   696
      { return rename<wpath>( from_path, to_path ); }
sl@0
   697
sl@0
   698
    inline void copy_file( const path & from_path, const path & to_path )
sl@0
   699
      { return copy_file<path>( from_path, to_path ); }
sl@0
   700
    inline void copy_file( const wpath & from_path, const wpath & to_path )
sl@0
   701
      { return copy_file<wpath>( from_path, to_path ); }
sl@0
   702
sl@0
   703
    inline path system_complete( const path & ph )
sl@0
   704
      { return system_complete<path>( ph ); }
sl@0
   705
    inline wpath system_complete( const wpath & ph )
sl@0
   706
      { return system_complete<wpath>( ph ); }
sl@0
   707
sl@0
   708
    inline path complete( const path & ph,
sl@0
   709
      const path & base/* = initial_path<path>()*/ )
sl@0
   710
      { return complete<path>( ph, base ); }
sl@0
   711
    inline wpath complete( const wpath & ph,
sl@0
   712
      const wpath & base/* = initial_path<wpath>()*/ )
sl@0
   713
      { return complete<wpath>( ph, base ); }
sl@0
   714
sl@0
   715
    inline path complete( const path & ph )
sl@0
   716
      { return complete<path>( ph, initial_path<path>() ); }
sl@0
   717
    inline wpath complete( const wpath & ph )
sl@0
   718
      { return complete<wpath>( ph, initial_path<wpath>() ); }
sl@0
   719
sl@0
   720
    inline void last_write_time( const path & ph, const std::time_t new_time )
sl@0
   721
      { last_write_time<path>( ph, new_time ); }
sl@0
   722
    inline void last_write_time( const wpath & ph, const std::time_t new_time )
sl@0
   723
      { last_write_time<wpath>( ph, new_time ); }
sl@0
   724
sl@0
   725
# endif // BOOST_FILESYSTEM_NARROW_ONLY
sl@0
   726
sl@0
   727
    namespace detail
sl@0
   728
    {
sl@0
   729
      template<class Path>
sl@0
   730
      unsigned long remove_all_aux( const Path & ph )
sl@0
   731
      {
sl@0
   732
        static const boost::filesystem::basic_directory_iterator<Path> end_itr;
sl@0
   733
        unsigned long count = 1;
sl@0
   734
        if ( !boost::filesystem::is_symlink( ph ) // don't recurse symbolic links
sl@0
   735
          && boost::filesystem::is_directory( ph ) )
sl@0
   736
        {
sl@0
   737
          for ( boost::filesystem::basic_directory_iterator<Path> itr( ph );
sl@0
   738
                itr != end_itr; ++itr )
sl@0
   739
          {
sl@0
   740
            count += remove_all_aux( itr->path() );
sl@0
   741
          }
sl@0
   742
        }
sl@0
   743
        boost::filesystem::remove( ph );
sl@0
   744
        return count;
sl@0
   745
      }
sl@0
   746
sl@0
   747
//  test helper  -------------------------------------------------------------//
sl@0
   748
sl@0
   749
    // not part of the documented interface because false positives are possible;
sl@0
   750
    // there is no law that says that an OS that has large stat.st_size
sl@0
   751
    // actually supports large file sizes.
sl@0
   752
      BOOST_FILESYSTEM_DECL bool possible_large_file_size_support();
sl@0
   753
sl@0
   754
//  directory_iterator helpers  ----------------------------------------------//
sl@0
   755
sl@0
   756
//    forwarding functions avoid need for BOOST_FILESYSTEM_DECL for class
sl@0
   757
//    basic_directory_iterator, and so avoid iterator_facade DLL template
sl@0
   758
//    problems. They also overload to the proper external path character type.
sl@0
   759
sl@0
   760
      BOOST_FILESYSTEM_DECL boost::filesystem::system_error_type
sl@0
   761
        dir_itr_first( void *& handle,
sl@0
   762
#if       defined(BOOST_POSIX_API)
sl@0
   763
            void *& buffer,
sl@0
   764
#endif
sl@0
   765
          const std::string & dir_path,
sl@0
   766
          std::string & target, file_status & fs, file_status & symlink_fs );
sl@0
   767
      // eof: return==0 && handle==0
sl@0
   768
sl@0
   769
      BOOST_FILESYSTEM_DECL boost::filesystem::system_error_type
sl@0
   770
        dir_itr_increment( void *& handle,
sl@0
   771
#if       defined(BOOST_POSIX_API)
sl@0
   772
            void *& buffer,
sl@0
   773
#endif
sl@0
   774
          std::string & target, file_status & fs, file_status & symlink_fs );
sl@0
   775
      // eof: return==0 && handle==0
sl@0
   776
sl@0
   777
      BOOST_FILESYSTEM_DECL boost::filesystem::system_error_type
sl@0
   778
        dir_itr_close( void *& handle
sl@0
   779
#if       defined(BOOST_POSIX_API)
sl@0
   780
            , void *& buffer
sl@0
   781
#endif
sl@0
   782
          );
sl@0
   783
      // Effects: none if handle==0, otherwise close handle, set handle=0
sl@0
   784
sl@0
   785
#     if defined(BOOST_WINDOWS_API) && !defined(BOOST_FILESYSTEM_NARROW_ONLY)
sl@0
   786
      BOOST_FILESYSTEM_DECL boost::filesystem::system_error_type
sl@0
   787
        dir_itr_first( void *& handle, const std::wstring & ph,
sl@0
   788
          std::wstring & target, file_status & fs, file_status & symlink_fs );
sl@0
   789
      BOOST_FILESYSTEM_DECL boost::filesystem::system_error_type
sl@0
   790
        dir_itr_increment( void *& handle, std::wstring & target,
sl@0
   791
          file_status & fs, file_status & symlink_fs );
sl@0
   792
#     endif
sl@0
   793
sl@0
   794
      template< class Path >
sl@0
   795
      class dir_itr_imp
sl@0
   796
      {
sl@0
   797
      public:  
sl@0
   798
        basic_directory_entry<Path> m_directory_entry;
sl@0
   799
        void *        m_handle;
sl@0
   800
#       ifdef BOOST_POSIX_API
sl@0
   801
          void *      m_buffer;  // see dir_itr_increment implementation
sl@0
   802
#       endif
sl@0
   803
        dir_itr_imp() : m_handle(0)
sl@0
   804
#       ifdef BOOST_POSIX_API
sl@0
   805
          , m_buffer(0)
sl@0
   806
#       endif
sl@0
   807
        {}
sl@0
   808
sl@0
   809
        ~dir_itr_imp() { dir_itr_close( m_handle
sl@0
   810
#if       defined(BOOST_POSIX_API)
sl@0
   811
            , m_buffer
sl@0
   812
#endif
sl@0
   813
          ); }
sl@0
   814
      };
sl@0
   815
sl@0
   816
    BOOST_FILESYSTEM_DECL extern system_error_type not_found_error;
sl@0
   817
    } // namespace detail
sl@0
   818
sl@0
   819
//  basic_directory_iterator  ------------------------------------------------//
sl@0
   820
sl@0
   821
    template< class Path >
sl@0
   822
    class basic_directory_iterator
sl@0
   823
      : public boost::iterator_facade<
sl@0
   824
          basic_directory_iterator<Path>,
sl@0
   825
          basic_directory_entry<Path>,
sl@0
   826
          boost::single_pass_traversal_tag >
sl@0
   827
    {
sl@0
   828
    public:
sl@0
   829
      typedef Path path_type;
sl@0
   830
sl@0
   831
      basic_directory_iterator(){}  // creates the "end" iterator
sl@0
   832
sl@0
   833
      explicit basic_directory_iterator( const Path & dir_path );
sl@0
   834
      basic_directory_iterator( const Path & dir_path, system_error_type & ec );
sl@0
   835
sl@0
   836
    private:
sl@0
   837
sl@0
   838
      // shared_ptr provides shallow-copy semantics required for InputIterators.
sl@0
   839
      // m_imp.get()==0 indicates the end iterator.
sl@0
   840
      boost::shared_ptr< detail::dir_itr_imp< Path > >  m_imp;
sl@0
   841
sl@0
   842
      friend class boost::iterator_core_access;
sl@0
   843
sl@0
   844
      typename boost::iterator_facade<
sl@0
   845
        basic_directory_iterator<Path>,
sl@0
   846
        basic_directory_entry<Path>,
sl@0
   847
        boost::single_pass_traversal_tag >::reference dereference() const 
sl@0
   848
      {
sl@0
   849
        BOOST_ASSERT( m_imp.get() && "attempt to dereference end iterator" );
sl@0
   850
        return m_imp->m_directory_entry;
sl@0
   851
      }
sl@0
   852
sl@0
   853
      void increment();
sl@0
   854
sl@0
   855
      bool equal( const basic_directory_iterator & rhs ) const
sl@0
   856
        { return m_imp == rhs.m_imp; }
sl@0
   857
sl@0
   858
      system_error_type m_init( const Path & dir_path );
sl@0
   859
    };
sl@0
   860
sl@0
   861
    typedef basic_directory_iterator< path > directory_iterator;
sl@0
   862
# ifndef BOOST_FILESYSTEM_NARROW_ONLY
sl@0
   863
    typedef basic_directory_iterator< wpath > wdirectory_iterator;
sl@0
   864
# endif
sl@0
   865
sl@0
   866
    //  basic_directory_iterator implementation  ---------------------------//
sl@0
   867
sl@0
   868
    template<class Path>
sl@0
   869
    system_error_type basic_directory_iterator<Path>::m_init(
sl@0
   870
      const Path & dir_path )
sl@0
   871
    {
sl@0
   872
      if ( dir_path.empty() )
sl@0
   873
      {
sl@0
   874
        m_imp.reset();
sl@0
   875
        return detail::not_found_error;
sl@0
   876
      }
sl@0
   877
      system_error_type sys_err;
sl@0
   878
      typename Path::external_string_type name;
sl@0
   879
      file_status fs, symlink_fs;
sl@0
   880
sl@0
   881
      if ( (sys_err = detail::dir_itr_first( m_imp->m_handle,
sl@0
   882
#if   defined(BOOST_POSIX_API)
sl@0
   883
        m_imp->m_buffer,
sl@0
   884
#endif
sl@0
   885
        dir_path.external_directory_string(),
sl@0
   886
        name, fs, symlink_fs )) != 0 )
sl@0
   887
      {
sl@0
   888
        m_imp.reset();
sl@0
   889
        return sys_err;
sl@0
   890
      }
sl@0
   891
      
sl@0
   892
      if ( m_imp->m_handle == 0 ) m_imp.reset(); // eof, so make end iterator
sl@0
   893
      else // not eof
sl@0
   894
      {
sl@0
   895
        m_imp->m_directory_entry.assign( dir_path
sl@0
   896
          / Path::traits_type::to_internal( name ), fs, symlink_fs );
sl@0
   897
        if ( name[0] == dot<Path>::value // dot or dot-dot
sl@0
   898
          && (name.size() == 1
sl@0
   899
            || (name[1] == dot<Path>::value
sl@0
   900
              && name.size() == 2)) )
sl@0
   901
          {  increment(); }
sl@0
   902
      }
sl@0
   903
      return 0;
sl@0
   904
    }
sl@0
   905
sl@0
   906
    template<class Path>
sl@0
   907
    basic_directory_iterator<Path>::basic_directory_iterator(
sl@0
   908
      const Path & dir_path )
sl@0
   909
      : m_imp( new detail::dir_itr_imp<Path> )
sl@0
   910
    {
sl@0
   911
      system_error_type ec( m_init(dir_path) );
sl@0
   912
      if ( ec != 0 )
sl@0
   913
      {
sl@0
   914
        boost::throw_exception( basic_filesystem_error<Path>(  
sl@0
   915
          "boost::filesystem::basic_directory_iterator constructor",
sl@0
   916
          dir_path, ec ) );
sl@0
   917
      }
sl@0
   918
    }
sl@0
   919
sl@0
   920
    template<class Path>
sl@0
   921
    basic_directory_iterator<Path>::basic_directory_iterator(
sl@0
   922
      const Path & dir_path, system_error_type & ec )
sl@0
   923
      : m_imp( new detail::dir_itr_imp<Path> )
sl@0
   924
    {
sl@0
   925
      ec = m_init(dir_path);
sl@0
   926
    }
sl@0
   927
sl@0
   928
    template<class Path>
sl@0
   929
    void basic_directory_iterator<Path>::increment()
sl@0
   930
    {
sl@0
   931
      BOOST_ASSERT( m_imp.get() && "attempt to increment end iterator" );
sl@0
   932
      BOOST_ASSERT( m_imp->m_handle != 0 && "internal program error" );
sl@0
   933
      
sl@0
   934
      system_error_type sys_err(0);
sl@0
   935
      typename Path::external_string_type name;
sl@0
   936
      file_status fs, symlink_fs;
sl@0
   937
sl@0
   938
      for (;;)
sl@0
   939
      {
sl@0
   940
        if ( (sys_err = detail::dir_itr_increment( m_imp->m_handle,
sl@0
   941
#if     defined(BOOST_POSIX_API)
sl@0
   942
          m_imp->m_buffer,
sl@0
   943
#endif
sl@0
   944
          name, fs, symlink_fs )) != 0 )
sl@0
   945
        {
sl@0
   946
          boost::throw_exception( basic_filesystem_error<Path>(  
sl@0
   947
            "boost::filesystem::basic_directory_iterator increment",
sl@0
   948
            m_imp->m_directory_entry.path().branch_path(), sys_err ) );
sl@0
   949
        }
sl@0
   950
        if ( m_imp->m_handle == 0 ) { m_imp.reset(); return; } // eof, make end
sl@0
   951
        if ( !(name[0] == dot<Path>::value // !(dot or dot-dot)
sl@0
   952
          && (name.size() == 1
sl@0
   953
            || (name[1] == dot<Path>::value
sl@0
   954
              && name.size() == 2))) )
sl@0
   955
        {
sl@0
   956
          m_imp->m_directory_entry.replace_leaf(
sl@0
   957
            Path::traits_type::to_internal( name ), fs, symlink_fs );
sl@0
   958
          return;
sl@0
   959
        }
sl@0
   960
      }
sl@0
   961
    }
sl@0
   962
sl@0
   963
    //  basic_directory_entry  -----------------------------------------------//
sl@0
   964
    
sl@0
   965
    template<class Path>
sl@0
   966
    class basic_directory_entry
sl@0
   967
    {
sl@0
   968
    public:
sl@0
   969
      typedef Path path_type;
sl@0
   970
      typedef typename Path::string_type string_type;
sl@0
   971
sl@0
   972
      // compiler generated copy-ctor, copy assignment, and destructor apply
sl@0
   973
sl@0
   974
      basic_directory_entry() {}
sl@0
   975
      explicit basic_directory_entry( const path_type & p,
sl@0
   976
        file_status st = file_status(), file_status symlink_st=file_status() )
sl@0
   977
        : m_path(p), m_status(st), m_symlink_status(symlink_st)
sl@0
   978
        {}
sl@0
   979
sl@0
   980
      void assign( const path_type & p,
sl@0
   981
        file_status st, file_status symlink_st )
sl@0
   982
        { m_path = p; m_status = st; m_symlink_status = symlink_st; }
sl@0
   983
sl@0
   984
      void replace_leaf( const string_type & s,
sl@0
   985
        file_status st, file_status symlink_st )
sl@0
   986
     {
sl@0
   987
       m_path.remove_leaf();
sl@0
   988
       m_path /= s;
sl@0
   989
       m_status = st;
sl@0
   990
       m_symlink_status = symlink_st;
sl@0
   991
     }
sl@0
   992
sl@0
   993
      const Path &   path() const { return m_path; }
sl@0
   994
      file_status   status() const;
sl@0
   995
      file_status   status( system_error_type & ec ) const;
sl@0
   996
      file_status   symlink_status() const;
sl@0
   997
      file_status   symlink_status( system_error_type & ec ) const;
sl@0
   998
sl@0
   999
      // conversion simplifies the most common use of basic_directory_entry
sl@0
  1000
      operator const path_type &() const { return m_path; }
sl@0
  1001
sl@0
  1002
#   ifndef BOOST_FILESYSTEM_NO_DEPRECATED
sl@0
  1003
      // deprecated functions preserve common use cases in legacy code
sl@0
  1004
      typename Path::string_type leaf() const
sl@0
  1005
      {
sl@0
  1006
        return path().leaf();
sl@0
  1007
      }
sl@0
  1008
      typename Path::string_type string() const
sl@0
  1009
      {
sl@0
  1010
        return path().string();
sl@0
  1011
      }
sl@0
  1012
#   endif
sl@0
  1013
sl@0
  1014
    private:
sl@0
  1015
      path_type             m_path;
sl@0
  1016
      mutable file_status  m_status;           // stat()-like
sl@0
  1017
      mutable file_status  m_symlink_status;   // lstat()-like
sl@0
  1018
        // note: m_symlink_status is not used by Windows implementation
sl@0
  1019
sl@0
  1020
    }; // basic_directory_status
sl@0
  1021
sl@0
  1022
    typedef basic_directory_entry<path> directory_entry;
sl@0
  1023
# ifndef BOOST_FILESYSTEM_NARROW_ONLY
sl@0
  1024
    typedef basic_directory_entry<wpath> wdirectory_entry;
sl@0
  1025
# endif
sl@0
  1026
sl@0
  1027
    //  basic_directory_entry implementation  --------------------------------//
sl@0
  1028
sl@0
  1029
    template<class Path>
sl@0
  1030
    file_status
sl@0
  1031
    basic_directory_entry<Path>::status() const
sl@0
  1032
    {
sl@0
  1033
      if ( !status_known( m_status ) )
sl@0
  1034
      {
sl@0
  1035
#     ifndef BOOST_WINDOWS_API
sl@0
  1036
        if ( status_known( m_symlink_status )
sl@0
  1037
          && !is_symlink( m_symlink_status ) )
sl@0
  1038
          { m_status = m_symlink_status; }
sl@0
  1039
        else { m_status = boost::filesystem::status( m_path ); }
sl@0
  1040
#     else
sl@0
  1041
        m_status = boost::filesystem::status( m_path );
sl@0
  1042
#     endif
sl@0
  1043
      }
sl@0
  1044
      return m_status;
sl@0
  1045
    }
sl@0
  1046
sl@0
  1047
    template<class Path>
sl@0
  1048
    file_status
sl@0
  1049
    basic_directory_entry<Path>::status( system_error_type & ec ) const
sl@0
  1050
    {
sl@0
  1051
      if ( !status_known( m_status ) )
sl@0
  1052
      {
sl@0
  1053
#     ifndef BOOST_WINDOWS_API
sl@0
  1054
        if ( status_known( m_symlink_status )
sl@0
  1055
          && !is_symlink( m_symlink_status ) )
sl@0
  1056
          { ec = 0; m_status = m_symlink_status; }
sl@0
  1057
        else { m_status = boost::filesystem::status( m_path, ec ); }
sl@0
  1058
#     else
sl@0
  1059
        m_status = boost::filesystem::status( m_path, ec );
sl@0
  1060
#     endif
sl@0
  1061
      }
sl@0
  1062
      else ec = 0;
sl@0
  1063
      return m_status;
sl@0
  1064
    }
sl@0
  1065
sl@0
  1066
    template<class Path>
sl@0
  1067
    file_status
sl@0
  1068
    basic_directory_entry<Path>::symlink_status() const
sl@0
  1069
    {
sl@0
  1070
#   ifndef BOOST_WINDOWS_API
sl@0
  1071
      if ( !status_known( m_symlink_status ) )
sl@0
  1072
        { m_symlink_status = boost::filesystem::symlink_status( m_path ); }
sl@0
  1073
      return m_symlink_status;
sl@0
  1074
#   else
sl@0
  1075
      return status();
sl@0
  1076
#   endif
sl@0
  1077
    }
sl@0
  1078
sl@0
  1079
    template<class Path>
sl@0
  1080
    file_status
sl@0
  1081
    basic_directory_entry<Path>::symlink_status( system_error_type & ec ) const
sl@0
  1082
    {
sl@0
  1083
#   ifndef BOOST_WINDOWS_API
sl@0
  1084
      if ( !status_known( m_symlink_status ) )
sl@0
  1085
        { m_symlink_status = boost::filesystem::symlink_status( m_path, ec ); }
sl@0
  1086
      else ec = 0;
sl@0
  1087
      return m_symlink_status;
sl@0
  1088
#   else
sl@0
  1089
      return status( ec );
sl@0
  1090
#   endif
sl@0
  1091
    }
sl@0
  1092
  } // namespace filesystem
sl@0
  1093
} // namespace boost
sl@0
  1094
sl@0
  1095
#undef BOOST_FS_FUNC
sl@0
  1096
sl@0
  1097
sl@0
  1098
#include <boost/config/abi_suffix.hpp> // pops abi_prefix.hpp pragmas
sl@0
  1099
#endif // BOOST_FILESYSTEM_OPERATIONS_HPP