Planet
navi homePPSaboutscreenshotsdownloaddevelopmentforum

source: downloads/boost_1_33_1/libs/filesystem/src/path_posix_windows.cpp @ 20

Last change on this file since 20 was 12, checked in by landauf, 18 years ago

added boost

File size: 19.9 KB
Line 
1//  path implementation  -----------------------------------------------------//
2
3//  © Copyright Beman Dawes 2002-2003
4//  Use, modification, and distribution is subject to the Boost Software
5//  License, Version 1.0. (See accompanying file LICENSE_1_0.txt or copy at
6//  http://www.boost.org/LICENSE_1_0.txt)
7
8//  See library home page at http://www.boost.org/libs/filesystem
9
10
11//****************************************************************************//
12
13//  WARNING: This code was hacked time and time again as different library
14//  designs were tried.  Thus portions may be residue from the earlier
15//  experiments, and be totally stupid or wrong in light of the final library
16//  specifications.  The code needs to be reviewed line-by-line to elmininate
17//  such problems.
18
19//****************************************************************************//
20
21// define BOOST_FILESYSTEM_SOURCE so that <boost/filesystem/config.hpp> knows
22// the library is being built (possibly exporting rather than importing code)
23#define BOOST_FILESYSTEM_SOURCE
24
25#include <boost/filesystem/path.hpp>
26#include <boost/filesystem/exception.hpp>
27
28// BOOST_POSIX or BOOST_WINDOWS specify which API to use.
29# if !defined( BOOST_WINDOWS ) && !defined( BOOST_POSIX )
30#   if defined(_WIN32) || defined(__WIN32__) || defined(WIN32) || defined(__CYGWIN__)
31#     define BOOST_WINDOWS
32#   else
33#     define BOOST_POSIX
34#   endif
35# endif
36
37
38namespace fs = boost::filesystem;
39
40#include <boost/config.hpp>
41#ifdef BOOST_NO_STDC_NAMESPACE
42  namespace std { using ::strlen; }
43#endif
44
45#include <boost/throw_exception.hpp>
46#include <cstring>  // SGI MIPSpro compilers need this
47#include <vector>
48#include <algorithm> // for lexicographical_compare
49#include <cassert>
50
51#include <boost/config/abi_prefix.hpp> // must be the last header
52
53//  helpers  -----------------------------------------------------------------//
54
55namespace
56{
57  // POSIX & Windows cases: "", "/", "/foo", "foo", "foo/bar"
58  // Windows only cases: "c:", "c:/", "c:foo", "c:/foo",
59  //                     "prn:", "//share", "//share/", "//share/foo"
60
61  std::string::size_type leaf_pos( const std::string & str,
62    std::string::size_type end_pos ) // end_pos is past-the-end position
63  // return 0 if str itself is leaf (or empty)
64  {
65    if ( end_pos && str[end_pos-1] == '/' ) return end_pos-1;
66   
67    std::string::size_type pos( str.find_last_of( '/', end_pos-1 ) );
68#   ifdef BOOST_WINDOWS
69    if ( pos == std::string::npos ) pos = str.find_last_of( ':', end_pos-2 );
70#   endif
71
72    return ( pos == std::string::npos // path itself must be a leaf (or empty)
73#     ifdef BOOST_WINDOWS
74      || (pos == 1 && str[0] == '/') // or share
75#     endif
76      ) ? 0 // so leaf is entire string
77        : pos + 1; // or starts after delimiter
78  }
79
80  void first_name( const std::string & src, std::string & target )
81  {
82    target = ""; // VC++ 6.0 doesn't have string::clear()
83    std::string::const_iterator itr( src.begin() );
84
85#   ifdef BOOST_WINDOWS
86    // deal with //share
87    if ( src.size() >= 2 && src[0] == '/' && src[1] == '/' )
88      { target = "//"; itr += 2; }
89#   endif
90
91    while ( itr != src.end()
92#     ifdef BOOST_WINDOWS
93      && *itr != ':'
94#     endif
95      && *itr != '/' ) { target += *itr++; }
96
97    if ( itr == src.end() ) return;
98
99#   ifdef BOOST_WINDOWS
100    if ( *itr == ':' )
101    {
102      target += *itr++;
103      return;
104    }
105#   endif
106
107    // *itr is '/'
108    if ( itr == src.begin() ) { target += '/'; }
109    return;
110  }
111
112  const char invalid_chars[] =
113    "\x01\x02\x03\x04\x05\x06\x07\x08\x09\x0A\x0B\x0C\x0D\x0E\x0F"
114    "\x10\x11\x12\x13\x14\x15\x16\x17\x18\x19\x1A\x1B\x1C\x1D\x1E\x1F"
115    "<>:\"/\\|";
116  // note that the terminating '\0' is part of the string - thus the size below
117  // is sizeof(invalid_chars) rather than sizeof(invalid_chars)-1.  I
118  const std::string windows_invalid_chars( invalid_chars, sizeof(invalid_chars) );
119
120  const std::string valid_posix(
121    "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789._-" );
122
123  fs::path::name_check default_check = fs::portable_name;
124  bool safe_to_write_check = true; // write-once-before-read allowed
125
126} // unnamed namespace
127
128//----------------------------------------------------------------------------//
129
130namespace boost
131{
132  namespace filesystem
133  {
134    //  name_check functions  ----------------------------------------------//
135
136#   ifdef BOOST_WINDOWS
137    BOOST_FILESYSTEM_DECL bool native( const std::string & name )
138    {
139      return windows_name( name );
140    }
141#   else
142    BOOST_FILESYSTEM_DECL bool native( const std::string & )
143    {
144      return true;
145    }
146#   endif
147
148    BOOST_FILESYSTEM_DECL bool no_check( const std::string & ) { return true; }
149
150    BOOST_FILESYSTEM_DECL bool portable_posix_name( const std::string & name )
151    {
152      return name.size() != 0
153        && name.find_first_not_of( valid_posix ) == std::string::npos;     
154    }
155
156    BOOST_FILESYSTEM_DECL bool windows_name( const std::string & name )
157    {
158      return name.size() != 0
159        && name.find_first_of( windows_invalid_chars ) == std::string::npos
160        && *(name.end()-1) != ' '
161        && (*(name.end()-1) != '.'
162          || name.length() == 1 || name == "..");
163    }
164
165    BOOST_FILESYSTEM_DECL bool portable_name( const std::string & name )
166    {
167      return
168        name.size() == 0
169        || name == "."
170        || name == ".."
171        || (windows_name( name )
172        && portable_posix_name( name )
173        && name[0] != '.' && name[0] != '-');
174    }
175
176    BOOST_FILESYSTEM_DECL bool portable_directory_name( const std::string & name )
177    {
178      return
179        name == "."
180        || name == ".."
181        || (portable_name( name )
182          && name.find('.') == std::string::npos);
183    }
184
185    BOOST_FILESYSTEM_DECL bool portable_file_name( const std::string & name )
186    {
187      std::string::size_type pos;
188      return
189         name == "."
190        || name == ".."
191        || (portable_name( name )
192          && ( (pos = name.find( '.' )) == std::string::npos
193            || (name.find( '.', pos+1 )== std::string::npos
194              && (pos + 5) > name.length() )))
195        ;
196    }
197
198
199//  path implementation  -----------------------------------------------------//
200
201    path::path( const std::string & src )
202    {
203      m_path_append( src, default_name_check() );
204    }
205
206    path::path( const char * src )
207    {
208      assert( src != 0 );
209      m_path_append( src, default_name_check() );
210    }
211
212    path::path( const std::string & src, name_check checker )
213    {
214      m_path_append( src, checker );
215    }
216
217    path::path( const char * src, name_check checker )
218    {
219      assert( src != 0 );
220      m_path_append( src, checker );
221    }
222
223    path & path::operator /=( const path & rhs )
224    {
225      m_path_append( rhs.m_path, no_check );
226      return *this;
227    }
228
229    void path::m_path_append( const std::string & src, name_check checker )
230    {
231      // convert backslash to forward slash if checker==native
232      // allow system-specific-root if checker==no_check || checker==native
233
234      assert( checker );
235      assert( src.size() == std::strlen( src.c_str() ) ); // no embedded 0
236
237      if ( src.size() == 0 ) return;
238
239      std::string::const_iterator itr( src.begin() );
240
241      // [root-filesystem]
242#     ifdef BOOST_WINDOWS
243      if ( (checker == no_check || checker == native) && src.size() >= 2 )
244      {
245        // drive or device
246        if ( src[1] == ':' || src[src.size()-1] == ':' )
247        {       
248          for ( ; *itr != ':'; ++itr ) m_path += *itr;
249          m_path += ':';
250          ++itr;
251        }
252
253        // share
254        else if ( (*itr == '/' || (*itr == '\\' && checker == native))
255          && (*(itr+1) == '/' || (*(itr+1) == '\\' && checker == native)) )
256        {
257          m_path += "//";
258          for ( itr += 2;
259                itr != src.end() && *itr != '/' && *itr != '\\';
260                ++itr ) m_path += *itr;
261        }
262      }
263#     endif
264
265      // root directory [ "/" ]
266      if ( itr != src.end() && (*itr == '/'
267#         ifdef BOOST_WINDOWS
268          || (*itr == '\\' && checker == native)
269#         endif
270          ) )
271      {
272        ++itr;
273        if ( m_path.size() == 0
274#         ifdef BOOST_WINDOWS
275          || m_path[m_path.size()-1] == ':' // drive or device
276          || (  // share
277             m_path.size() > 2
278             && m_path[0] == '/'
279             && m_path[1] == '/'
280             && m_path.find( '/', 2 ) == std::string::npos
281             )
282#         endif
283          ) m_path += '/';
284      }
285
286      // element { "/" element } [ "/" ]
287      while ( itr != src.end() )
288      {
289        if ( m_path == "." ) m_path = "";
290
291        // directory-placeholder
292        if ( *itr == '.' && ((itr+1) == src.end() || *(itr+1) == '/'
293#         ifdef BOOST_WINDOWS
294          || *(itr+1) == '\\'
295#         endif
296          ) )
297        {
298          if ( empty() ) m_path += '.';
299          ++itr;
300        }
301
302        // parent-directory or name
303        else
304        {
305          // append '/' if needed
306          if ( !empty()
307#         ifdef BOOST_WINDOWS
308            && *(m_path.end()-1) != ':'
309#         endif
310            && *(m_path.end()-1) != '/' )
311              m_path += '/';
312
313          // parent-directory
314          if ( *itr == '.'
315            && (itr+1) != src.end() && *(itr+1) == '.'
316            && ((itr+2) == src.end() || *(itr+2) == '/'
317#           ifdef BOOST_WINDOWS
318            || *(itr+2) == '\\'
319#           endif
320           ) )
321          {
322            m_path += "..";
323            ++itr;
324            ++itr;
325          } // parent-directory
326
327          // name
328          else
329          {
330            std::string name;
331            do
332              { name += *itr; }
333            while ( ++itr != src.end() && *itr != '/'
334#             ifdef BOOST_WINDOWS
335              && (*itr != '\\' || checker != native)
336#             endif
337              );
338
339            if ( !checker( name ) )
340            {
341              boost::throw_exception( filesystem_error(
342                "boost::filesystem::path",
343                "invalid name \"" + name + "\" in path: \"" + src + "\"" ) );
344            }
345
346            m_path += name;
347          }
348        } // parent-directory or name
349
350        // end or "/"
351        if ( itr != src.end() )
352        {
353          if ( *itr == '/'
354#         ifdef BOOST_WINDOWS
355          || (*itr == '\\' && checker == native)
356#         endif
357          ) ++itr;
358          else 
359            boost::throw_exception( filesystem_error(
360              "boost::filesystem::path",
361              "invalid path syntax: \"" + src + "\"" ) );
362        }
363
364      } // while more elements
365
366      // special case: remove one or more leading "/.."
367
368      std::string::size_type pos = 0, sz = m_path.size();
369
370#     ifdef BOOST_WINDOWS
371      if ( sz > 2 && m_path[pos] != '/' && m_path[pos+1] == ':' ) // drive
372        { pos += 2;  sz  -= 2; }
373#     endif
374
375      while ( sz >= 3 && m_path[pos] == '/'
376         && m_path[pos+1] == '.' && m_path[pos+2] == '.'
377         && (sz == 3 || m_path[pos+3] == '/') )
378      {
379        m_path.erase( pos+1, 3 ); // "..[/]"
380        sz -= 3; // on last, 3 should be 2; that doesn't matter
381      }
382    }
383
384// path conversion functions  ------------------------------------------------//
385
386    std::string path::native_file_string() const
387    {
388#   ifdef BOOST_WINDOWS
389      std::string s( m_path );
390      for ( std::string::iterator itr( s.begin() );
391        itr != s.end(); ++itr )
392        if ( *itr == '/' ) *itr = '\\';
393      return s;
394#   else
395      return m_path;
396#   endif
397    }
398
399    std::string path::native_directory_string() const
400      { return native_file_string(); }
401
402// path modification functions -----------------------------------------------//
403
404      path & path::normalize()
405      {
406        if ( m_path.empty() ) return *this;
407        std::string::size_type end, beg(0), start(0);
408
409#       ifdef BOOST_WINDOWS
410          if ( m_path.size() > 2
411            && m_path[0] != '/' && m_path[1] == ':' ) start = 2; // drive
412#       endif
413
414        while ( (beg=m_path.find( "/..", beg )) != std::string::npos )
415        {
416          end = beg + 3;
417          if ( (beg == 1 && m_path[0] == '.')
418            || (beg == 2 && m_path[0] == '.' && m_path[1] == '.')
419            || (beg > 2 && m_path[beg-3] == '/'
420                        && m_path[beg-2] == '.' && m_path[beg-1] == '.') )
421          {
422            beg = end;
423            continue;
424          }
425          if ( end < m_path.size() )
426          {
427            if ( m_path[end] == '/' ) ++end;
428            else { beg = end; continue; } // name starts with ..
429          }
430
431          // end is one past end of substr to be erased; now set beg
432          while ( beg > start && m_path[--beg] != '/' ) {}
433          if ( m_path[beg] == '/') ++beg;
434          m_path.erase( beg, end-beg );
435          if ( beg ) --beg;
436        }
437
438        if ( m_path.empty() ) m_path = ".";
439        else
440        { // remove trailing '/' if not root directory
441          std::string::size_type sz = m_path.size();
442
443#       ifdef BOOST_WINDOWS
444          if ( start ) sz  -= 2; // drive
445#       endif
446
447          if ( sz > 1 && m_path[m_path.size()-1] == '/' )
448            { m_path.erase( m_path.size()-1 ); }
449        }
450        return *this;
451      }
452
453 // path decomposition functions  ---------------------------------------------//
454
455    path::iterator path::begin() const
456    {
457      iterator itr;
458      itr.m_path_ptr = this;
459      first_name( m_path, itr.m_name );
460      itr.m_pos = 0;
461      return itr;
462    }
463
464    void path::m_replace_leaf( const char * new_leaf )
465    {
466      m_path.erase( leaf_pos( m_path, m_path.size() ) );
467      m_path += new_leaf;
468    }
469
470    std::string path::leaf() const
471    {
472      return m_path.substr( leaf_pos( m_path, m_path.size() ) );
473    }
474
475    namespace detail
476    {
477      inline bool is_absolute_root( const std::string & s,
478        std::string::size_type len )
479      {
480        return
481          len && s[len-1] == '/'
482          &&
483          (
484            len == 1 // "/"
485#       ifdef BOOST_WINDOWS
486            || ( len > 1
487                 && ( s[len-2] == ':' // drive or device
488                   || ( s[0] == '/'   // share
489                     && s[1] == '/'
490                     && s.find( '/', 2 ) == len-1
491                      )
492                    )
493                )
494#       endif
495          );
496      }
497    }
498
499    path path::branch_path() const
500    {
501      std::string::size_type end_pos( leaf_pos( m_path, m_path.size() ) );
502
503      // skip a '/' unless it is a root directory
504      if ( end_pos && m_path[end_pos-1] == '/'
505        && !detail::is_absolute_root( m_path, end_pos ) ) --end_pos;
506      return path( m_path.substr( 0, end_pos ), no_check );
507    }
508
509    path path::relative_path() const
510    {
511      std::string::size_type pos( 0 );
512      if ( m_path.size() && m_path[0] == '/' )
513      { pos = 1;
514#     ifdef BOOST_WINDOWS
515        if ( m_path.size()>1 && m_path[1] == '/' ) // share
516        {
517          if ( (pos = m_path.find( '/', 2 )) != std::string::npos ) ++pos;
518          else return path();
519        }
520      }
521      else if ( (pos = m_path.find( ':' )) == std::string::npos ) pos = 0;
522      else // has ':'
523      {
524        if ( ++pos < m_path.size() && m_path[pos] == '/' ) ++pos;
525#     endif
526      }
527      return path( m_path.substr( pos ), no_check );
528    }
529
530    std::string path::root_name() const
531    {
532#   ifdef BOOST_WINDOWS
533      std::string::size_type pos( m_path.find( ':' ) );
534      if ( pos != std::string::npos ) return m_path.substr( 0, pos+1 );
535      if ( m_path.size() > 2 && m_path[0] == '/' && m_path[1] == '/' )
536      {
537        pos = m_path.find( '/', 2 );
538        return m_path.substr( 0, pos );
539      }
540#   endif
541      return std::string();
542    }
543
544    std::string path::root_directory() const
545    {
546      return std::string(
547        ( m_path.size() && m_path[0] == '/' )  // covers both "/" and "//share"
548#       ifdef BOOST_WINDOWS
549          || ( m_path.size() > 2
550               && m_path[1] == ':'
551               && m_path[2] == '/' )  // "c:/"
552#       endif
553               ? "/" : "" );
554    }
555
556    path path::root_path() const
557    {
558      return path(
559#   ifdef BOOST_WINDOWS
560        root_name(), no_check ) /= root_directory();
561#   else
562        root_directory() );
563#   endif
564    }
565
566// path query functions  -----------------------------------------------------//
567
568    bool path::is_complete() const
569    {
570#   ifdef BOOST_WINDOWS
571      return m_path.size() > 2
572        && ( (m_path[1] == ':' && m_path[2] == '/') // "c:/"
573          || (m_path[0] == '/' && m_path[1] == '/') // "//share"
574          || m_path[m_path.size()-1] == ':' );
575#   else
576      return m_path.size() && m_path[0] == '/';
577#   endif
578    }
579
580    bool path::has_root_path() const
581    {
582      return ( m_path.size() 
583               && m_path[0] == '/' )  // covers both "/" and "//share"
584#            ifdef BOOST_WINDOWS
585               || ( m_path.size() > 1 && m_path[1] == ':' ) // "c:" and "c:/"
586               || ( m_path.size() > 3
587                    && m_path[m_path.size()-1] == ':' ) // "device:"
588#            endif
589               ;
590    }
591
592    bool path::has_root_name() const
593    {
594#   ifdef BOOST_WINDOWS
595      return m_path.size() > 1
596        && ( m_path[1] == ':' // "c:"
597          || m_path[m_path.size()-1] == ':' // "prn:"
598          || (m_path[0] == '/' && m_path[1] == '/') // "//share"
599           );
600#   else
601      return false;
602#   endif
603    }
604
605    bool path::has_root_directory() const
606    {
607      return ( m_path.size() 
608               && m_path[0] == '/' )  // covers both "/" and "//share"
609#            ifdef BOOST_WINDOWS
610               || ( m_path.size() > 2
611                    && m_path[1] == ':' && m_path[2] == '/' ) // "c:/"
612#            endif
613               ;
614    }
615
616    bool path::has_relative_path() const { return !relative_path().empty(); }
617    bool path::has_branch_path() const { return !branch_path().empty(); }
618
619    //  default name_check mechanism  ----------------------------------------//
620
621    bool path::default_name_check_writable()
622    {
623      return safe_to_write_check;
624    }
625
626    void path::default_name_check( name_check new_check )
627    {
628      assert( new_check );
629      if ( !safe_to_write_check )
630        boost::throw_exception(
631          filesystem_error( "boost::filesystem::default_name_check",
632                            "default name check already set" ));
633      default_check = new_check;
634      safe_to_write_check = false;
635    }
636
637    path::name_check path::default_name_check()
638    {
639      safe_to_write_check = false;
640      return default_check;
641    }
642
643    //  path operator<  ------------------------------------------------------//
644    bool path::operator<( const path & that ) const
645    {
646      return std::lexicographical_compare(
647        begin(), end(), that.begin(), that.end() );
648    }
649
650
651// path::iterator implementation  --------------------------------------------//
652
653    BOOST_FILESYSTEM_DECL void path::iterator::increment()
654    {
655      assert( m_pos < m_path_ptr->m_path.size() ); // detect increment past end
656      m_pos += m_name.size();
657      if ( m_pos == m_path_ptr->m_path.size() )
658      {
659        m_name = "";  // not strictly required, but might aid debugging
660        return;
661      }
662      if ( m_path_ptr->m_path[m_pos] == '/' )
663      {
664#       ifdef BOOST_WINDOWS
665        if ( m_name[m_name.size()-1] == ':' // drive or device
666          || (m_name[0] == '/' && m_name[1] == '/') ) // share
667        {
668          m_name = "/";
669          return;
670        }
671#       endif
672        ++m_pos;
673      }
674      std::string::size_type end_pos( m_path_ptr->m_path.find( '/', m_pos ) );
675      if ( end_pos == std::string::npos )
676        end_pos = m_path_ptr->m_path.size();
677      m_name = m_path_ptr->m_path.substr( m_pos, end_pos - m_pos );
678    }
679
680    BOOST_FILESYSTEM_DECL void path::iterator::decrement()
681    {                                                                               
682      assert( m_pos ); // detect decrement of begin
683      std::string::size_type end_pos( m_pos );
684
685      // skip a '/' unless it is a root directory
686      if ( m_path_ptr->m_path[end_pos-1] == '/'
687        && !detail::is_absolute_root( m_path_ptr->m_path, end_pos ) ) --end_pos;
688      m_pos = leaf_pos( m_path_ptr->m_path, end_pos );
689      m_name = m_path_ptr->m_path.substr( m_pos, end_pos - m_pos );
690    }
691  } // namespace filesystem
692} // namespace boost
Note: See TracBrowser for help on using the repository browser.