| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222 | /* Distributed under the OSI-approved BSD 3-Clause License.  See accompanying   file Copyright.txt or https://cmake.org/licensing for details.  */#include "cmMacroCommand.h"#include <sstream>#include <stdio.h>#include <utility>#include "cmAlgorithms.h"#include "cmExecutionStatus.h"#include "cmMakefile.h"#include "cmPolicies.h"#include "cmState.h"#include "cmSystemTools.h"// define the class for macro commandsclass cmMacroHelperCommand : public cmCommand{public:  cmMacroHelperCommand() {}  ///! clean up any memory allocated by the macro  ~cmMacroHelperCommand() override {}  /**   * This is a virtual constructor for the command.   */  cmCommand* Clone() override  {    cmMacroHelperCommand* newC = new cmMacroHelperCommand;    // we must copy when we clone    newC->Args = this->Args;    newC->Functions = this->Functions;    newC->FilePath = this->FilePath;    newC->Policies = this->Policies;    return newC;  }  /**   * This is called when the command is first encountered in   * the CMakeLists.txt file.   */  bool InvokeInitialPass(const std::vector<cmListFileArgument>& args,                         cmExecutionStatus&) override;  bool InitialPass(std::vector<std::string> const&,                   cmExecutionStatus&) override  {    return false;  }  std::vector<std::string> Args;  std::vector<cmListFileFunction> Functions;  cmPolicies::PolicyMap Policies;  std::string FilePath;};bool cmMacroHelperCommand::InvokeInitialPass(  const std::vector<cmListFileArgument>& args, cmExecutionStatus& inStatus){  // Expand the argument list to the macro.  std::vector<std::string> expandedArgs;  this->Makefile->ExpandArguments(args, expandedArgs);  // make sure the number of arguments passed is at least the number  // required by the signature  if (expandedArgs.size() < this->Args.size() - 1) {    std::string errorMsg =      "Macro invoked with incorrect arguments for macro named: ";    errorMsg += this->Args[0];    this->SetError(errorMsg);    return false;  }  cmMakefile::MacroPushPop macroScope(this->Makefile, this->FilePath,                                      this->Policies);  // set the value of argc  std::ostringstream argcDefStream;  argcDefStream << expandedArgs.size();  std::string argcDef = argcDefStream.str();  std::vector<std::string>::const_iterator eit =    expandedArgs.begin() + (this->Args.size() - 1);  std::string expandedArgn = cmJoin(cmMakeRange(eit, expandedArgs.end()), ";");  std::string expandedArgv = cmJoin(expandedArgs, ";");  std::vector<std::string> variables;  variables.reserve(this->Args.size() - 1);  for (unsigned int j = 1; j < this->Args.size(); ++j) {    variables.push_back("${" + this->Args[j] + "}");  }  std::vector<std::string> argVs;  argVs.reserve(expandedArgs.size());  char argvName[60];  for (unsigned int j = 0; j < expandedArgs.size(); ++j) {    sprintf(argvName, "${ARGV%u}", j);    argVs.push_back(argvName);  }  // Invoke all the functions that were collected in the block.  cmListFileFunction newLFF;  // for each function  for (cmListFileFunction const& func : this->Functions) {    // Replace the formal arguments and then invoke the command.    newLFF.Arguments.clear();    newLFF.Arguments.reserve(func.Arguments.size());    newLFF.Name = func.Name;    newLFF.Line = func.Line;    // for each argument of the current function    for (cmListFileArgument const& k : func.Arguments) {      cmListFileArgument arg;      arg.Value = k.Value;      if (k.Delim != cmListFileArgument::Bracket) {        // replace formal arguments        for (unsigned int j = 0; j < variables.size(); ++j) {          cmSystemTools::ReplaceString(arg.Value, variables[j],                                       expandedArgs[j]);        }        // replace argc        cmSystemTools::ReplaceString(arg.Value, "${ARGC}", argcDef);        cmSystemTools::ReplaceString(arg.Value, "${ARGN}", expandedArgn);        cmSystemTools::ReplaceString(arg.Value, "${ARGV}", expandedArgv);        // if the current argument of the current function has ${ARGV in it        // then try replacing ARGV values        if (arg.Value.find("${ARGV") != std::string::npos) {          for (unsigned int t = 0; t < expandedArgs.size(); ++t) {            cmSystemTools::ReplaceString(arg.Value, argVs[t], expandedArgs[t]);          }        }      }      arg.Delim = k.Delim;      arg.Line = k.Line;      newLFF.Arguments.push_back(std::move(arg));    }    cmExecutionStatus status;    if (!this->Makefile->ExecuteCommand(newLFF, status) ||        status.GetNestedError()) {      // The error message should have already included the call stack      // so we do not need to report an error here.      macroScope.Quiet();      inStatus.SetNestedError();      return false;    }    if (status.GetReturnInvoked()) {      inStatus.SetReturnInvoked();      return true;    }    if (status.GetBreakInvoked()) {      inStatus.SetBreakInvoked();      return true;    }  }  return true;}bool cmMacroFunctionBlocker::IsFunctionBlocked(const cmListFileFunction& lff,                                               cmMakefile& mf,                                               cmExecutionStatus&){  // record commands until we hit the ENDMACRO  // at the ENDMACRO call we shift gears and start looking for invocations  if (lff.Name.Lower == "macro") {    this->Depth++;  } else if (lff.Name.Lower == "endmacro") {    // if this is the endmacro for this macro then execute    if (!this->Depth) {      mf.AppendProperty("MACROS", this->Args[0].c_str());      // create a new command and add it to cmake      cmMacroHelperCommand* f = new cmMacroHelperCommand();      f->Args = this->Args;      f->Functions = this->Functions;      f->FilePath = this->GetStartingContext().FilePath;      mf.RecordPolicies(f->Policies);      mf.GetState()->AddScriptedCommand(this->Args[0], f);      // remove the function blocker now that the macro is defined      mf.RemoveFunctionBlocker(this, lff);      return true;    }    // decrement for each nested macro that ends    this->Depth--;  }  // if it wasn't an endmacro and we are not executing then we must be  // recording  this->Functions.push_back(lff);  return true;}bool cmMacroFunctionBlocker::ShouldRemove(const cmListFileFunction& lff,                                          cmMakefile& mf){  if (lff.Name.Lower == "endmacro") {    std::vector<std::string> expandedArguments;    mf.ExpandArguments(lff.Arguments, expandedArguments,                       this->GetStartingContext().FilePath.c_str());    // if the endmacro has arguments make sure they    // match the arguments of the macro    if ((expandedArguments.empty() ||         (expandedArguments[0] == this->Args[0]))) {      return true;    }  }  return false;}bool cmMacroCommand::InitialPass(std::vector<std::string> const& args,                                 cmExecutionStatus&){  if (args.empty()) {    this->SetError("called with incorrect number of arguments");    return false;  }  // create a function blocker  cmMacroFunctionBlocker* f = new cmMacroFunctionBlocker();  f->Args.insert(f->Args.end(), args.begin(), args.end());  this->Makefile->AddFunctionBlocker(f);  return true;}
 |