//===- Path.cpp -----------------------------------------------------------===// // // The MCLinker Project // // This file is distributed under the University of Illinois Open Source // License. See LICENSE.TXT for details. // //===----------------------------------------------------------------------===// #include "mcld/Config/Config.h" #include "mcld/Support/FileSystem.h" #include "mcld/Support/Path.h" #include <llvm/ADT/StringRef.h> #include <locale> #include <string.h> #include <istream> #include <ostream> using namespace mcld; using namespace mcld::sys::fs; //===--------------------------------------------------------------------===// // Path Path::Path() : m_PathName() { } Path::Path(const Path::ValueType* s ) : m_PathName(s) { } Path::Path(const Path::StringType &s ) : m_PathName(s) { } Path::Path(const Path& pCopy) : m_PathName(pCopy.m_PathName) { } Path::~Path() { } bool Path::isFromRoot() const { if (m_PathName.empty()) return false; return (separator == m_PathName[0]); } bool Path::isFromPWD() const { if (2 > m_PathName.size()) return false; return ('.' == m_PathName[0] && separator == m_PathName[1]); } Path& Path::assign(const Path::StringType &s) { m_PathName.assign(s); return *this; } Path& Path::assign(const Path::ValueType* s, unsigned int length) { if (0 == s || 0 == length) assert(0 && "assign a null or empty string to Path"); m_PathName.assign(s, length); return *this; } //a,/b a/,b a/,b/ a,b is a/b Path& Path::append(const Path& pPath) { //first path is a/,second path is /b if(m_PathName[m_PathName.length()-1] == separator && pPath.native()[0] == separator) { unsigned int old_size = m_PathName.size()-1; unsigned int new_size = old_size + pPath.native().size(); m_PathName.resize(new_size); strcpy(const_cast<char*>(m_PathName.data()+old_size), pPath.native().data()); } //first path is a,second path is b else if(this->string()[this->native().size()-1] != separator && pPath.string()[0] != separator) { m_PathName.append("/"); m_PathName.append(pPath.native()); } // a/,b or a,/b just append else { m_PathName.append(pPath.native()); } return *this; } bool Path::empty() const { return m_PathName.empty(); } std::string Path::string() const { return m_PathName; } Path::StringType Path::generic_string() const { std::string result = m_PathName; detail::canonicalize(result); return result; } bool Path::canonicalize() { return detail::canonicalize(m_PathName); } Path::StringType::size_type Path::m_append_separator_if_needed() { if (!m_PathName.empty() && #if defined(MCLD_ON_WIN32) *(m_PathName.end()-1) != colon && #endif !is_separator(*(m_PathName.end()-1))) { StringType::size_type tmp(m_PathName.size()); m_PathName += preferred_separator; return tmp; } return 0; } void Path::m_erase_redundant_separator(Path::StringType::size_type pSepPos) { size_t begin=pSepPos; // skip '/' while(separator == m_PathName[pSepPos]) ++pSepPos; if(begin!=pSepPos) m_PathName.erase(begin+1,pSepPos-begin-1); } Path Path::parent_path() const { size_t end_pos = m_PathName.find_last_of(separator); if (end_pos != StringType::npos) return Path(m_PathName.substr(0, end_pos)); return Path(); } Path Path::filename() const { size_t pos = m_PathName.find_last_of(separator); if (pos != StringType::npos) { ++pos; return Path(m_PathName.substr(pos)); } return Path(*this); } Path Path::stem() const { size_t begin_pos = m_PathName.find_last_of(separator)+1; size_t end_pos = m_PathName.find_last_of("."); Path result_path(m_PathName.substr(begin_pos, end_pos - begin_pos)); return result_path; } Path Path::extension() const { size_t begin_pos = m_PathName.find_last_of('.'); Path result_path(m_PathName.substr(begin_pos)); return result_path; } //===--------------------------------------------------------------------===// // non-member functions //===--------------------------------------------------------------------===// bool mcld::sys::fs::operator==(const Path& pLHS,const Path& pRHS) { return (pLHS.generic_string()==pRHS.generic_string()); } bool mcld::sys::fs::operator!=(const Path& pLHS,const Path& pRHS) { return !(pLHS==pRHS); } Path mcld::sys::fs::operator+(const Path& pLHS, const Path& pRHS) { mcld::sys::fs::Path result = pLHS; result.append(pRHS); return result; } bool mcld::sys::fs::is_separator(char value) { return (value == separator #if defined(MCLD_ON_WIN32) || value == preferred_separator #endif ); } bool mcld::sys::fs::exists(const Path &pPath) { FileStatus pFileStatus; detail::status(pPath, pFileStatus); return exists(pFileStatus); } bool mcld::sys::fs::is_directory(const Path &pPath) { FileStatus pFileStatus; detail::status(pPath, pFileStatus); return is_directory(pFileStatus); } std::ostream &mcld::sys::fs::operator<<(std::ostream& pOS, const Path& pPath) { return pOS << pPath.native(); } std::istream &mcld::sys::fs::operator>>(std::istream& pOS, Path& pPath) { return pOS >> pPath.native(); } llvm::raw_ostream &mcld::sys::fs::operator<<(llvm::raw_ostream &pOS, const Path &pPath) { return pOS << pPath.native(); }