/*========================================================================= Program: CMake - Cross-Platform Makefile Generator Module: $RCSfile$ Language: C++ Date: $Date$ Version: $Revision$ Copyright (c) 2002 Kitware, Inc., Insight Consortium. All rights reserved. See Copyright.txt or http://www.cmake.org/HTML/Copyright.html for details. This software is distributed WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the above copyright notices for more information. =========================================================================*/ #include "cmGlob.h" #include #include #include #include #include #if defined( _WIN32 ) || defined( APPLE ) || defined( __CYGWIN__ ) // On Windows and apple, no difference between lower and upper case #define CM_GLOB_CASE_INDEPENDENT #endif #if defined( _WIN32 ) || defined( __CYGWIN__ ) // Handle network paths #define CM_GLOB_SUPPORT_NETWORK_PATHS #endif class cmGlobInternal { public: std::vector Files; std::vector Expressions; }; cmGlob::cmGlob() { m_Internals = new cmGlobInternal; m_Recurse = false; } cmGlob::~cmGlob() { delete m_Internals; } void cmGlob::Escape(int ch, char* buffer) { if (! ( 'a' <= ch && ch <= 'z' || 'A' <= ch && ch <= 'Z' || '0' <= ch && ch <= '9') ) { sprintf(buffer, "\\%c", ch); } else { #if defined( CM_GLOB_CASE_INDEPENDENT ) // On Windows and apple, no difference between lower and upper case sprintf(buffer, "%c", tolower(ch)); #else sprintf(buffer, "%c", ch); #endif } } std::vector& cmGlob::GetFiles() { return m_Internals->Files; } std::string cmGlob::ConvertExpression(const std::string& expr) { std::string::size_type i = 0; std::string::size_type n = expr.size(); std::string res = "^"; std::string stuff = ""; while ( i < n ) { int c = expr[i]; i = i+1; if ( c == '*' ) { res = res + ".*"; } else if ( c == '?' ) { res = res + "."; } else if ( c == '[' ) { std::string::size_type j = i; if ( j < n && ( expr[j] == '!' || expr[j] == '^' ) ) { j = j+1; } if ( j < n && expr[j] == ']' ) { j = j+1; } while ( j < n && expr[j] != ']' ) { j = j+1; } if ( j >= n ) { res = res + "\\["; } else { stuff = ""; std::string::size_type cc; for ( cc = i; cc < j; cc ++ ) { if ( expr[cc] == '\\' ) { stuff += "\\\\"; } else { stuff += expr[cc]; } } i = j+1; if ( stuff[0] == '!' || stuff[0] == '^' ) { stuff = '^' + stuff.substr(1); } else if ( stuff[0] == '^' ) { stuff = '\\' + stuff; } res = res + "[" + stuff + "]"; } } else { char buffer[100]; buffer[0] = 0; this->Escape(c, buffer); res = res + buffer; } } return res + "$"; } void cmGlob::RecurseDirectory(const std::string& dir, bool dir_only) { cmsys::Directory d; if ( !d.Load(dir.c_str()) ) { return; } unsigned long cc; std::string fullname; std::string fname; for ( cc = 0; cc < d.GetNumberOfFiles(); cc ++ ) { fname = d.GetFile(cc); if ( strcmp(fname.c_str(), ".") == 0 || strcmp(fname.c_str(), "..") == 0 ) { continue; } #if defined( CM_GLOB_CASE_INDEPENDENT ) // On Windows and apple, no difference between lower and upper case fname = cmsys::SystemTools::LowerCase(fname); #endif fullname = dir + "/" + fname; if ( !dir_only || !cmsys::SystemTools::FileIsDirectory(fullname.c_str()) ) { if ( m_Internals->Expressions[m_Internals->Expressions.size()-1].find(fname.c_str()) ) { m_Internals->Files.push_back(fullname); } } if ( cmsys::SystemTools::FileIsDirectory(fullname.c_str()) ) { this->RecurseDirectory(fullname, dir_only); } } } void cmGlob::ProcessDirectory(std::string::size_type start, const std::string& dir, bool dir_only) { bool last = ( start == m_Internals->Expressions.size()-1 ); if ( last && m_Recurse ) { this->RecurseDirectory(dir, dir_only); return; } cmsys::Directory d; if ( !d.Load(dir.c_str()) ) { return; } unsigned long cc; std::string fullname; std::string fname; for ( cc = 0; cc < d.GetNumberOfFiles(); cc ++ ) { fname = d.GetFile(cc); if ( strcmp(fname.c_str(), ".") == 0 || strcmp(fname.c_str(), "..") == 0 ) { continue; } #if defined( CM_GLOB_CASE_INDEPENDENT ) // On Windows and apple, no difference between lower and upper case fname = cmsys::SystemTools::LowerCase(fname); #endif if ( start == 0 ) { fullname = dir + fname; } else { fullname = dir + "/" + fname; } if ( (!dir_only || !last) && !cmsys::SystemTools::FileIsDirectory(fullname.c_str()) ) { continue; } if ( m_Internals->Expressions[start].find(fname.c_str()) ) { if ( last ) { m_Internals->Files.push_back(fullname); } else { this->ProcessDirectory(start+1, fullname, dir_only); } } } } bool cmGlob::FindFiles(const std::string& inexpr) { std::string cexpr; std::string::size_type cc; std::string expr = inexpr; m_Internals->Expressions.empty(); m_Internals->Files.empty(); if ( !cmsys::SystemTools::FileIsFullPath(expr.c_str()) ) { expr = cmsys::SystemTools::GetCurrentWorkingDirectory(); expr += "/" + inexpr; } int skip = 0; #if defined( CM_GLOB_SUPPORT_NETWORK_PATHS ) // Handle network paths if ( expr[0] == '/' && expr[1] == '/' ) { int cnt = 0; for ( cc = 2; cc < expr.size(); cc ++ ) { if ( expr[cc] == '/' ) { cnt ++; if ( cnt == 2 ) { break; } } } skip = cc + 1; expr = expr.substr(skip); } else #endif if ( expr[1] == ':' && expr[0] != '/' ) { expr = expr.substr(2); } cexpr = ""; for ( cc = 0; cc < expr.size(); cc ++ ) { int ch = expr[cc]; if ( ch == '/' ) { if ( cexpr.size() > 0 ) { this->AddExpression(cexpr.c_str()); } cexpr = ""; } else { cexpr.append(1, (char)ch); } } if ( cexpr.size() > 0 ) { this->AddExpression(cexpr.c_str()); } #ifdef _WIN32 // Handle network paths if ( skip > 0 ) { this->ProcessDirectory(0, inexpr.substr(0, skip), true); } else #endif if ( inexpr[1] == ':' && inexpr[0] != '/' ) { std::string startdir = "A:/"; if ( inexpr[0] >= 'a' && inexpr[0] <= 'z' || inexpr[0] >= 'A' && inexpr[0] <= 'Z') { startdir[0] = inexpr[0]; this->ProcessDirectory(0, startdir, true); } else { return false; } } else { this->ProcessDirectory(0, "/", true); } return true; } void cmGlob::AddExpression(const char* expr) { m_Internals->Expressions.push_back( cmsys::RegularExpression( this->ConvertExpression(expr).c_str())); }