CMake/Source/cmMacroCommand.cxx

249 lines
C++
Raw Normal View History

2002-08-13 23:46:33 +04:00
/*=========================================================================
Program: CMake - Cross-Platform Makefile Generator
2002-08-13 23:46:33 +04:00
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.
2002-08-13 23:46:33 +04:00
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 "cmMacroCommand.h"
bool cmMacroFunctionBlocker::
IsFunctionBlocked(const cmListFileFunction& lff, cmMakefile &mf)
2002-08-13 23:46:33 +04:00
{
// record commands until we hit the ENDMACRO
2003-06-24 23:11:21 +04:00
if (!m_Executing)
2002-08-13 23:46:33 +04:00
{
2003-06-24 23:11:21 +04:00
// at the ENDMACRO call we shift gears and start looking for invocations
if(lff.m_Name == "ENDMACRO")
{
2003-06-24 23:11:21 +04:00
std::vector<std::string> expandedArguments;
mf.ExpandArguments(lff.m_Arguments, expandedArguments);
if(!expandedArguments.empty() && (expandedArguments[0] == m_Args[0]))
{
m_Executing = true;
std::string name = m_Args[0];
std::vector<std::string>::size_type cc;
name += "(";
for ( cc = 0; cc < m_Args.size(); cc ++ )
{
name += " " + m_Args[cc];
}
name += " )";
mf.AddMacro(m_Args[0].c_str(), name.c_str());
2003-06-24 23:11:21 +04:00
return true;
}
}
2002-08-13 23:46:33 +04:00
// if it wasn't an endmacro and we are not executing then we must be
// recording
m_Functions.push_back(lff);
2002-08-13 23:46:33 +04:00
return true;
}
// otherwise the macro has been recorded and we are executing
// so we look for macro invocations
if(lff.m_Name == m_Args[0])
2002-08-13 23:46:33 +04:00
{
2003-06-24 00:26:48 +04:00
std::string tmps;
cmListFileArgument arg;
std::string variable;
// Expand the argument list to the macro.
std::vector<std::string> expandedArguments;
mf.ExpandArguments(lff.m_Arguments, expandedArguments);
2004-04-26 19:11:57 +04:00
// make sure the number of arguments passed is at least the number
// required by the signature
if (expandedArguments.size() < m_Args.size() - 1)
2002-08-13 23:46:33 +04:00
{
cmOStringStream error;
error << "Error in cmake code at\n"
<< lff.m_FilePath << ":" << lff.m_Line << ":\n"
<< "Invocation of macro \""
<< lff.m_Name.c_str() << "\" with incorrect number of arguments.";
cmSystemTools::Error(error.str().c_str());
return true;
2002-08-13 23:46:33 +04:00
}
2004-05-20 00:04:48 +04:00
// set the value of argc
2004-04-27 16:30:25 +04:00
cmOStringStream argcDefStream;
argcDefStream << expandedArguments.size();
std::string argcDef = argcDefStream.str();
2004-04-26 19:11:57 +04:00
2004-05-20 00:04:48 +04:00
// declare varuiables for ARGV ARGN but do not compute until needed
std::string argvDef;
std::string argnDef;
bool argnDefInitialized = false;
bool argvDefInitialized = false;
// Invoke all the functions that were collected in the block.
2003-06-24 23:11:21 +04:00
cmListFileFunction newLFF;
2004-05-20 00:04:48 +04:00
// for each function
for(unsigned int c = 0; c < m_Functions.size(); ++c)
2002-08-13 23:46:33 +04:00
{
// Replace the formal arguments and then invoke the command.
2003-06-24 23:11:21 +04:00
newLFF.m_Arguments.clear();
newLFF.m_Arguments.reserve(m_Functions[c].m_Arguments.size());
newLFF.m_Name = m_Functions[c].m_Name;
newLFF.m_FilePath = m_Functions[c].m_FilePath;
newLFF.m_Line = m_Functions[c].m_Line;
2004-05-20 00:04:48 +04:00
// for each argument of the current function
for (std::vector<cmListFileArgument>::const_iterator k =
m_Functions[c].m_Arguments.begin();
k != m_Functions[c].m_Arguments.end(); ++k)
2002-08-13 23:46:33 +04:00
{
2003-06-24 00:26:48 +04:00
tmps = k->Value;
2004-04-26 19:11:57 +04:00
// replace formal arguments
2002-08-13 23:46:33 +04:00
for (unsigned int j = 1; j < m_Args.size(); ++j)
{
2003-06-24 00:26:48 +04:00
variable = "${";
2002-08-13 23:46:33 +04:00
variable += m_Args[j];
variable += "}";
cmSystemTools::ReplaceString(tmps, variable.c_str(),
expandedArguments[j-1].c_str());
2002-08-13 23:46:33 +04:00
}
2004-05-20 00:04:48 +04:00
// replace argc
cmSystemTools::ReplaceString(tmps, "${ARGC}",argcDef.c_str());
// repleace ARGN
if (tmps.find("${ARGN}") != std::string::npos)
{
2004-05-20 00:04:48 +04:00
if (!argnDefInitialized)
{
2004-05-20 00:04:48 +04:00
std::vector<std::string>::iterator eit;
std::vector<std::string>::size_type cnt = 0;
for ( eit = expandedArguments.begin();
eit != expandedArguments.end();
++ eit )
{
2004-05-20 00:04:48 +04:00
if ( cnt >= m_Args.size()-1 )
{
if ( argnDef.size() > 0 )
{
argnDef += ";";
}
argnDef += *eit;
}
cnt ++;
}
2004-05-20 00:04:48 +04:00
argnDefInitialized = true;
}
2004-05-20 00:04:48 +04:00
cmSystemTools::ReplaceString(tmps, "${ARGN}", argnDef.c_str());
}
2004-05-20 00:04:48 +04:00
// if the current argument of the current function has ${ARGV in it
// then try replacing ARGV values
if (tmps.find("${ARGV") != std::string::npos)
{
2004-05-20 00:04:48 +04:00
char argvName[60];
// repleace ARGV, compute it only once
if (!argvDefInitialized)
{
2004-05-20 00:04:48 +04:00
std::vector<std::string>::iterator eit;
for ( eit = expandedArguments.begin();
eit != expandedArguments.end();
++ eit )
{
2004-05-20 00:04:48 +04:00
if ( argvDef.size() > 0 )
{
2004-05-20 00:04:48 +04:00
argvDef += ";";
}
2004-05-20 00:04:48 +04:00
argvDef += *eit;
}
2004-05-20 00:04:48 +04:00
argvDefInitialized = true;
}
2004-05-20 00:04:48 +04:00
cmSystemTools::ReplaceString(tmps, "${ARGV}", argvDef.c_str());
// also replace the ARGV1 ARGV2 ... etc
for (unsigned int t = 0; t < expandedArguments.size(); ++t)
2004-04-26 19:11:57 +04:00
{
2004-05-20 00:04:48 +04:00
sprintf(argvName,"${ARGV%i}",t);
cmSystemTools::ReplaceString(tmps, argvName,
expandedArguments[t].c_str());
2004-04-26 19:11:57 +04:00
}
}
2003-06-24 00:26:48 +04:00
arg.Value = tmps;
arg.Quoted = k->Quoted;
const char* def =
mf.GetDefinition("CMAKE_MACRO_REPORT_DEFINITION_LOCATION");
if(def && !cmSystemTools::IsOff(def))
{
// Report the location of the argument where the macro was
// defined.
arg.FilePath = k->FilePath;
arg.Line = k->Line;
}
else
{
// Report the location of the argument where the macro was
// invoked.
arg.FilePath = lff.m_FilePath;
arg.Line = lff.m_Line;
}
newLFF.m_Arguments.push_back(arg);
2002-08-13 23:46:33 +04:00
}
if(!mf.ExecuteCommand(newLFF))
{
cmOStringStream error;
error << "Error in cmake code at\n"
<< lff.m_FilePath << ":" << lff.m_Line << ":\n"
<< "A command failed during the invocation of macro \""
<< lff.m_Name.c_str() << "\".";
cmSystemTools::Error(error.str().c_str());
}
2002-08-13 23:46:33 +04:00
}
return true;
}
// if not an invocation then it is just an ordinary line
return false;
}
bool cmMacroFunctionBlocker::
ShouldRemove(const cmListFileFunction&, cmMakefile &)
2002-08-13 23:46:33 +04:00
{
return false;
}
void cmMacroFunctionBlocker::
2003-04-28 21:16:18 +04:00
ScopeEnded(cmMakefile &mf)
2002-08-13 23:46:33 +04:00
{
2003-04-28 21:16:18 +04:00
// macros never leave scope but we should have seen the ENDMACRO call by now
if (m_Executing != true)
{
cmSystemTools::Error("The end of a CMakeLists file was reached with a MACRO statement that was not closed properly. Within the directory: ",
mf.GetCurrentDirectory(), " with macro ",
m_Args[0].c_str());
}
2002-08-13 23:46:33 +04:00
}
bool cmMacroCommand::InitialPass(std::vector<std::string> const& args)
2002-08-13 23:46:33 +04:00
{
if(args.size() < 1)
{
this->SetError("called with incorrect number of arguments");
return false;
}
// create a function blocker
cmMacroFunctionBlocker *f = new cmMacroFunctionBlocker();
for(std::vector<std::string>::const_iterator j = args.begin();
j != args.end(); ++j)
{
f->m_Args.push_back(*j);
}
m_Makefile->AddFunctionBlocker(f);
return true;
}