ENH: Add globbing to FILE command
This commit is contained in:
parent
1a4ec91593
commit
db77d2e019
|
@ -15,6 +15,7 @@ cmCustomCommand.cxx
|
||||||
cmCacheManager.cxx
|
cmCacheManager.cxx
|
||||||
cmSourceGroup.cxx
|
cmSourceGroup.cxx
|
||||||
cmListFileCache.cxx
|
cmListFileCache.cxx
|
||||||
|
cmGlob.cxx
|
||||||
cmGlobalGenerator.cxx
|
cmGlobalGenerator.cxx
|
||||||
cmGlobalUnixMakefileGenerator.cxx
|
cmGlobalUnixMakefileGenerator.cxx
|
||||||
cmLocalGenerator.cxx
|
cmLocalGenerator.cxx
|
||||||
|
|
|
@ -16,6 +16,8 @@
|
||||||
=========================================================================*/
|
=========================================================================*/
|
||||||
#include "cmFileCommand.h"
|
#include "cmFileCommand.h"
|
||||||
|
|
||||||
|
#include "cmGlob.h"
|
||||||
|
|
||||||
// cmLibraryCommand
|
// cmLibraryCommand
|
||||||
bool cmFileCommand::InitialPass(std::vector<std::string> const& args)
|
bool cmFileCommand::InitialPass(std::vector<std::string> const& args)
|
||||||
{
|
{
|
||||||
|
@ -37,6 +39,10 @@ bool cmFileCommand::InitialPass(std::vector<std::string> const& args)
|
||||||
{
|
{
|
||||||
return this->HandleReadCommand(args);
|
return this->HandleReadCommand(args);
|
||||||
}
|
}
|
||||||
|
else if ( subCommand == "GLOB" )
|
||||||
|
{
|
||||||
|
return this->HandleGlobCommand(args);
|
||||||
|
}
|
||||||
|
|
||||||
std::string e = "does not recognize sub-command "+subCommand;
|
std::string e = "does not recognize sub-command "+subCommand;
|
||||||
this->SetError(e.c_str());
|
this->SetError(e.c_str());
|
||||||
|
@ -53,6 +59,12 @@ bool cmFileCommand::HandleWriteCommand(std::vector<std::string> const& args,
|
||||||
i++; // Get rid of subcommand
|
i++; // Get rid of subcommand
|
||||||
|
|
||||||
std::string fileName = *i;
|
std::string fileName = *i;
|
||||||
|
if ( !cmsys::SystemTools::FileIsFullPath(i->c_str()) )
|
||||||
|
{
|
||||||
|
fileName = m_Makefile->GetCurrentDirectory();
|
||||||
|
fileName += "/" + *i;
|
||||||
|
}
|
||||||
|
|
||||||
i++;
|
i++;
|
||||||
|
|
||||||
for(;i != args.end(); ++i)
|
for(;i != args.end(); ++i)
|
||||||
|
@ -82,9 +94,16 @@ bool cmFileCommand::HandleReadCommand(std::vector<std::string> const& args)
|
||||||
if ( args.size() != 3 )
|
if ( args.size() != 3 )
|
||||||
{
|
{
|
||||||
this->SetError("READ must be called with two additional arguments");
|
this->SetError("READ must be called with two additional arguments");
|
||||||
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
std::string fileName = args[1];
|
std::string fileName = args[1];
|
||||||
|
if ( !cmsys::SystemTools::FileIsFullPath(args[1].c_str()) )
|
||||||
|
{
|
||||||
|
fileName = m_Makefile->GetCurrentDirectory();
|
||||||
|
fileName += "/" + args[1];
|
||||||
|
}
|
||||||
|
|
||||||
std::string variable = args[2];
|
std::string variable = args[2];
|
||||||
std::ifstream file(fileName.c_str(), std::ios::in);
|
std::ifstream file(fileName.c_str(), std::ios::in);
|
||||||
if ( !file )
|
if ( !file )
|
||||||
|
@ -111,3 +130,48 @@ bool cmFileCommand::HandleReadCommand(std::vector<std::string> const& args)
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
//----------------------------------------------------------------------------
|
||||||
|
bool cmFileCommand::HandleGlobCommand(std::vector<std::string> const& args)
|
||||||
|
{
|
||||||
|
if ( args.size() < 2 )
|
||||||
|
{
|
||||||
|
this->SetError("GLOB requires at least a variable name");
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
std::vector<std::string>::const_iterator i = args.begin();
|
||||||
|
|
||||||
|
i++; // Get rid of subcommand
|
||||||
|
|
||||||
|
std::string variable = *i;
|
||||||
|
i++;
|
||||||
|
cmGlob g;
|
||||||
|
std::string output = "";
|
||||||
|
bool first = true;
|
||||||
|
for ( ; i != args.end(); ++i )
|
||||||
|
{
|
||||||
|
if ( !cmsys::SystemTools::FileIsFullPath(i->c_str()) )
|
||||||
|
{
|
||||||
|
std::string expr = m_Makefile->GetCurrentDirectory();
|
||||||
|
expr += "/" + *i;
|
||||||
|
g.FindFiles(expr);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
g.FindFiles(*i);
|
||||||
|
}
|
||||||
|
std::vector<std::string>::size_type cc;
|
||||||
|
std::vector<std::string>& files = g.GetFiles();
|
||||||
|
for ( cc = 0; cc < files.size(); cc ++ )
|
||||||
|
{
|
||||||
|
if ( !first )
|
||||||
|
{
|
||||||
|
output += ";";
|
||||||
|
}
|
||||||
|
output += files[cc];
|
||||||
|
first = false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
m_Makefile->AddDefinition(variable.c_str(), output.c_str());
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
|
@ -69,19 +69,23 @@ public:
|
||||||
" FILE(WRITE filename \"message to write\"... )\n"
|
" FILE(WRITE filename \"message to write\"... )\n"
|
||||||
" FILE(APPEND filename \"message to write\"... )\n"
|
" FILE(APPEND filename \"message to write\"... )\n"
|
||||||
" FILE(READ filename variable)\n"
|
" FILE(READ filename variable)\n"
|
||||||
|
" FILE(GLOB variable [globbing expressions]...)\n"
|
||||||
"WRITE will write a message into a file called 'filename'. It "
|
"WRITE will write a message into a file called 'filename'. It "
|
||||||
"overwrites the file if it already exists, and creates the file "
|
"overwrites the file if it already exists, and creates the file "
|
||||||
"if it does not exists.\n\n"
|
"if it does not exists.\n\n"
|
||||||
"APPEND will write a message into a file same as WRITE, except "
|
"APPEND will write a message into a file same as WRITE, except "
|
||||||
"it will append it to the end of the file\n\n"
|
"it will append it to the end of the file\n\n"
|
||||||
"READ will read the content of a file and store it into a "
|
"READ will read the content of a file and store it into the "
|
||||||
"variable.\n\n"; }
|
"variable.\n\n"
|
||||||
|
"GLOB will generate a list of all files that match the expressions "
|
||||||
|
"and store it into the variable.\n\n"; }
|
||||||
|
|
||||||
cmTypeMacro(cmFileCommand, cmCommand);
|
cmTypeMacro(cmFileCommand, cmCommand);
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
bool HandleWriteCommand(std::vector<std::string> const& args, bool append);
|
bool HandleWriteCommand(std::vector<std::string> const& args, bool append);
|
||||||
bool HandleReadCommand(std::vector<std::string> const& args);
|
bool HandleReadCommand(std::vector<std::string> const& args);
|
||||||
|
bool HandleGlobCommand(std::vector<std::string> const& args);
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
|
|
|
@ -0,0 +1,240 @@
|
||||||
|
/*=========================================================================
|
||||||
|
|
||||||
|
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 <cmsys/Directory.hxx>
|
||||||
|
#include <cmsys/RegularExpression.hxx>
|
||||||
|
#include <cmsys/SystemTools.hxx>
|
||||||
|
|
||||||
|
#include <stdio.h>
|
||||||
|
|
||||||
|
class cmGlobInternal
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
std::vector<std::string> Files;
|
||||||
|
std::vector<cmsys::RegularExpression> Expressions;
|
||||||
|
};
|
||||||
|
|
||||||
|
cmGlob::cmGlob()
|
||||||
|
{
|
||||||
|
m_Internals = new cmGlobInternal;
|
||||||
|
}
|
||||||
|
|
||||||
|
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
|
||||||
|
{
|
||||||
|
sprintf(buffer, "%c", ch);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
std::vector<std::string>& 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] == '!' )
|
||||||
|
{
|
||||||
|
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 = '^' + 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::ProcessDirectory(std::string::size_type start,
|
||||||
|
const std::string& dir, bool dir_only)
|
||||||
|
{
|
||||||
|
cmsys::Directory d;
|
||||||
|
if ( !d.Load(dir.c_str()) )
|
||||||
|
{
|
||||||
|
//std::cout << "Cannot open directory: " << dir.c_str() << std::endl;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
unsigned long cc;
|
||||||
|
std::string fullname;
|
||||||
|
bool last = ( start == m_Internals->Expressions.size()-1 );
|
||||||
|
//std::cout << "Last: " << last << " Dironly: " << dir_only << std::endl;
|
||||||
|
for ( cc = 0; cc < d.GetNumberOfFiles(); cc ++ )
|
||||||
|
{
|
||||||
|
if ( strcmp(d.GetFile(cc), ".") == 0 ||
|
||||||
|
strcmp(d.GetFile(cc), "..") == 0 )
|
||||||
|
{
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
if ( start == 0 )
|
||||||
|
{
|
||||||
|
fullname = dir + d.GetFile(cc);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
fullname = dir + "/" + d.GetFile(cc);
|
||||||
|
}
|
||||||
|
|
||||||
|
if ( (!dir_only || !last) && !cmsys::SystemTools::FileIsDirectory(fullname.c_str()) )
|
||||||
|
{
|
||||||
|
//std::cout << " Ignore: " << fullname.c_str() << std::endl;
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
if ( m_Internals->Expressions[start].find(d.GetFile(cc)) )
|
||||||
|
{
|
||||||
|
//std::cout << " Matches: " << fullname.c_str() << std::endl;
|
||||||
|
if ( last )
|
||||||
|
{
|
||||||
|
//std::cout << "--- find file: " << fullname.c_str() << "---" << std::endl;
|
||||||
|
m_Internals->Files.push_back(fullname);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
this->ProcessDirectory(start+1, fullname, dir_only);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
//std::cout << " Not Matches: " << fullname.c_str() << std::endl;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
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;
|
||||||
|
}
|
||||||
|
std::cout << "Expr: " << expr << std::endl;
|
||||||
|
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());
|
||||||
|
}
|
||||||
|
|
||||||
|
this->ProcessDirectory(0, "/", true);
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
void cmGlob::AddExpression(const char* expr)
|
||||||
|
{
|
||||||
|
m_Internals->Expressions.push_back(
|
||||||
|
cmsys::RegularExpression(
|
||||||
|
this->ConvertExpression(expr).c_str()));
|
||||||
|
}
|
||||||
|
|
|
@ -0,0 +1,61 @@
|
||||||
|
/*=========================================================================
|
||||||
|
|
||||||
|
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.
|
||||||
|
|
||||||
|
=========================================================================*/
|
||||||
|
#ifndef cmGlob_h
|
||||||
|
#define cmGlob_h
|
||||||
|
|
||||||
|
#include "cmStandardIncludes.h"
|
||||||
|
|
||||||
|
class cmGlobInternal;
|
||||||
|
|
||||||
|
/** \class cmGlob
|
||||||
|
* \brief Helper class for performing globbing searches.
|
||||||
|
*
|
||||||
|
* Finds all files that match a given globbing expression.
|
||||||
|
*/
|
||||||
|
class cmGlob
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
cmGlob();
|
||||||
|
~cmGlob();
|
||||||
|
|
||||||
|
//! Find all files that match the pattern.
|
||||||
|
bool FindFiles(const std::string& inexpr);
|
||||||
|
|
||||||
|
//! Return the list of files that matched.
|
||||||
|
std::vector<std::string>& GetFiles();
|
||||||
|
|
||||||
|
protected:
|
||||||
|
//! Process directory
|
||||||
|
void ProcessDirectory(std::string::size_type start,
|
||||||
|
const std::string& dir, bool dir_only);
|
||||||
|
|
||||||
|
//! Escape all non-alphanumeric characters in pattern.
|
||||||
|
void Escape(int ch, char* buffer);
|
||||||
|
|
||||||
|
//!
|
||||||
|
// Translate a shell PATTERN to a regular expression.
|
||||||
|
// There is no way to quote meta-characters.
|
||||||
|
std::string ConvertExpression(const std::string& expr);
|
||||||
|
|
||||||
|
//! Add regular expression
|
||||||
|
void AddExpression(const char* expr);
|
||||||
|
|
||||||
|
cmGlobInternal* m_Internals;
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
|
#endif
|
Loading…
Reference in New Issue