//===- Path.h -------------------------------------------------------------===// // // The MCLinker Project // // This file is distributed under the University of Illinois Open Source // License. See LICENSE.TXT for details. // //===----------------------------------------------------------------------===// // This file declares the mcld::sys::fs::Path. It follows TR2/boost // filesystem (v3), but modified to remove exception handling and the // path class. //===----------------------------------------------------------------------===// #ifndef MCLD_SUPPORT_PATH_H_ #define MCLD_SUPPORT_PATH_H_ #include "mcld/Config/Config.h" #include <llvm/Support/raw_ostream.h> #include <iosfwd> #include <functional> #include <string> #include <locale> namespace mcld { namespace sys { namespace fs { #if defined(MCLD_ON_WIN32) const char preferred_separator = '/'; const char separator = '/'; #else const char preferred_separator = '/'; const char separator = '/'; #endif const char colon = ':'; const char dot = '.'; /** \class Path * \brief Path provides an abstraction for the path to a file or directory in * the operating system's filesystem. */ class Path { public: typedef char ValueType; typedef std::string StringType; public: Path(); explicit Path(const ValueType* s); explicit Path(const StringType& s); Path(const Path& pCopy); virtual ~Path(); // ----- assignments ----- // template <class InputIterator> Path& assign(InputIterator begin, InputIterator end); Path& assign(const StringType& s); Path& assign(const ValueType* s, unsigned int length); // ----- appends ----- // template <class InputIterator> Path& append(InputIterator begin, InputIterator end); Path& append(const Path& pPath); Path& append(const StringType& pPath); // ----- observers ----- // bool empty() const; bool isFromRoot() const; bool isFromPWD() const; const StringType& native() const { return m_PathName; } StringType& native() { return m_PathName; } const ValueType* c_str() const { return m_PathName.c_str(); } // ----- decomposition ----- // Path parent_path() const; Path filename() const; Path stem() const; Path extension() const; // ----- generic form observers ----- // StringType generic_string() const; bool canonicalize(); public: StringType::size_type m_append_separator_if_needed(); void m_erase_redundant_separator(StringType::size_type sep_pos); protected: StringType m_PathName; }; bool operator==(const Path& pLHS, const Path& pRHS); bool operator!=(const Path& pLHS, const Path& pRHS); Path operator+(const Path& pLHS, const Path& pRHS); //===----------------------------------------------------------------------===// // Non-member Functions //===----------------------------------------------------------------------===// bool exists(const Path& pPath); bool is_directory(const Path& pPath); template <class Char, class Traits> inline std::basic_ostream<Char, Traits>& operator<<( std::basic_ostream<Char, Traits>& pOS, const Path& pPath) { return pOS << pPath.native(); } template <class Char, class Traits> inline std::basic_istream<Char, Traits>& operator>>( std::basic_istream<Char, Traits>& pOS, Path& pPath) { return pOS >> pPath.native(); } inline llvm::raw_ostream& operator<<(llvm::raw_ostream& pOS, const Path& pPath) { return pOS << pPath.native(); } //===----------------------------------------------------------------------===// // class path member template implementation //===----------------------------------------------------------------------===// template <class InputIterator> Path& Path::assign(InputIterator begin, InputIterator end) { m_PathName.clear(); if (begin != end) m_PathName.append<InputIterator>(begin, end); return *this; } template <class InputIterator> Path& Path::append(InputIterator begin, InputIterator end) { if (begin == end) return *this; StringType::size_type sep_pos(m_append_separator_if_needed()); m_PathName.append<InputIterator>(begin, end); if (sep_pos) m_erase_redundant_separator(sep_pos); return *this; } } // namespace fs } // namespace sys } // namespace mcld //===----------------------------------------------------------------------===// // STL compatible functions //===----------------------------------------------------------------------===// namespace std { template <> struct less<mcld::sys::fs::Path> : public binary_function<mcld::sys::fs::Path, mcld::sys::fs::Path, bool> { bool operator()(const mcld::sys::fs::Path& pX, const mcld::sys::fs::Path& pY) const { if (pX.generic_string().size() < pY.generic_string().size()) return true; return (pX.generic_string() < pY.generic_string()); } }; } // namespace std #endif // MCLD_SUPPORT_PATH_H_