| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159 |
- /*============================================================================
- CMake - Cross Platform Makefile Generator
- Copyright 2000-2009 Kitware, Inc., Insight Software Consortium
- Distributed under the OSI-approved BSD License (the "License");
- see accompanying file Copyright.txt for details.
- This software is distributed WITHOUT ANY WARRANTY; without even the
- implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
- See the License for more information.
- ============================================================================*/
- #include "cmVariableWatchCommand.h"
- #include "cmVariableWatch.h"
- //----------------------------------------------------------------------------
- struct cmVariableWatchCallbackData
- {
- bool InCallback;
- std::string Command;
- };
- //----------------------------------------------------------------------------
- static void cmVariableWatchCommandVariableAccessed(
- const std::string& variable, int access_type, void* client_data,
- const char* newValue, const cmMakefile* mf)
- {
- cmVariableWatchCallbackData* data
- = static_cast<cmVariableWatchCallbackData*>(client_data);
- if ( data->InCallback )
- {
- return;
- }
- data->InCallback = true;
- cmListFileFunction newLFF;
- cmListFileArgument arg;
- bool processed = false;
- const char* accessString = cmVariableWatch::GetAccessAsString(access_type);
- const char* currentListFile = mf->GetDefinition("CMAKE_CURRENT_LIST_FILE");
- /// Ultra bad!!
- cmMakefile* makefile = const_cast<cmMakefile*>(mf);
- std::string stack = makefile->GetProperty("LISTFILE_STACK");
- if ( !data->Command.empty() )
- {
- newLFF.Arguments.clear();
- newLFF.Arguments.push_back(
- cmListFileArgument(variable, cmListFileArgument::Quoted,
- "unknown", 9999));
- newLFF.Arguments.push_back(
- cmListFileArgument(accessString, cmListFileArgument::Quoted,
- "unknown", 9999));
- newLFF.Arguments.push_back(
- cmListFileArgument(newValue?newValue:"", cmListFileArgument::Quoted,
- "unknown", 9999));
- newLFF.Arguments.push_back(
- cmListFileArgument(currentListFile, cmListFileArgument::Quoted,
- "unknown", 9999));
- newLFF.Arguments.push_back(
- cmListFileArgument(stack, cmListFileArgument::Quoted,
- "unknown", 9999));
- newLFF.Name = data->Command;
- newLFF.FilePath = "Some weird path";
- newLFF.Line = 9999;
- cmExecutionStatus status;
- if(!makefile->ExecuteCommand(newLFF,status))
- {
- arg.FilePath = "Unknown";
- arg.Line = 0;
- cmOStringStream error;
- error << "Error in cmake code at\n"
- << arg.FilePath << ":" << arg.Line << ":\n"
- << "A command failed during the invocation of callback \""
- << data->Command << "\".";
- cmSystemTools::Error(error.str().c_str());
- data->InCallback = false;
- return;
- }
- processed = true;
- }
- if ( !processed )
- {
- cmOStringStream msg;
- msg << "Variable \"" << variable.c_str() << "\" was accessed using "
- << accessString << " with value \"" << (newValue?newValue:"") << "\".";
- makefile->IssueMessage(cmake::LOG, msg.str());
- }
- data->InCallback = false;
- }
- //----------------------------------------------------------------------------
- static void deleteVariableWatchCallbackData(void* client_data)
- {
- cmVariableWatchCallbackData* data
- = static_cast<cmVariableWatchCallbackData*>(client_data);
- delete data;
- }
- //----------------------------------------------------------------------------
- cmVariableWatchCommand::cmVariableWatchCommand()
- {
- }
- //----------------------------------------------------------------------------
- cmVariableWatchCommand::~cmVariableWatchCommand()
- {
- std::set<std::string>::const_iterator it;
- for ( it = this->WatchedVariables.begin();
- it != this->WatchedVariables.end();
- ++it )
- {
- this->Makefile->GetCMakeInstance()->GetVariableWatch()->RemoveWatch(
- *it, cmVariableWatchCommandVariableAccessed);
- }
- }
- //----------------------------------------------------------------------------
- bool cmVariableWatchCommand
- ::InitialPass(std::vector<std::string> const& args, cmExecutionStatus &)
- {
- if ( args.size() < 1 )
- {
- this->SetError("must be called with at least one argument.");
- return false;
- }
- std::string variable = args[0];
- std::string command;
- if ( args.size() > 1 )
- {
- command = args[1];
- }
- if ( variable == "CMAKE_CURRENT_LIST_FILE" )
- {
- cmOStringStream ostr;
- ostr << "cannot be set on the variable: " << variable.c_str();
- this->SetError(ostr.str().c_str());
- return false;
- }
- cmVariableWatchCallbackData* data = new cmVariableWatchCallbackData;
- data->InCallback = false;
- data->Command = command;
- this->WatchedVariables.insert(variable);
- if ( !this->Makefile->GetCMakeInstance()->GetVariableWatch()->AddWatch(
- variable, cmVariableWatchCommandVariableAccessed,
- data, deleteVariableWatchCallbackData) )
- {
- deleteVariableWatchCallbackData(data);
- return false;
- }
- return true;
- }
|