CMake/Source/cmAddTestCommand.cxx
Brad King 9862f383d0 ENH: Add NAME mode to ADD_TEST command
This creates command mode add_test(NAME ...).  This signature is
extensible with more keyword arguments later.  The main purpose is to
enable automatic replacement of target names with built target file
locations.  A side effect of this feature is support for tests that only
run under specific configurations.
2009-03-16 10:51:30 -04:00

166 lines
4.3 KiB
C++

/*=========================================================================
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 "cmAddTestCommand.h"
#include "cmTestGenerator.h"
#include "cmTest.h"
// cmExecutableCommand
bool cmAddTestCommand
::InitialPass(std::vector<std::string> const& args, cmExecutionStatus &)
{
if(!args.empty() && args[0] == "NAME")
{
return this->HandleNameMode(args);
}
// First argument is the name of the test Second argument is the name of
// the executable to run (a target or external program) Remaining arguments
// are the arguments to pass to the executable
if(args.size() < 2 )
{
this->SetError("called with incorrect number of arguments");
return false;
}
// Collect the command with arguments.
std::vector<std::string> command;
for(std::vector<std::string>::const_iterator it = args.begin() + 1;
it != args.end(); ++it)
{
command.push_back(*it);
}
// Create the test but add a generator only the first time it is
// seen. This preserves behavior from before test generators.
cmTest* test = this->Makefile->GetTest(args[0].c_str());
if(test)
{
// If the test was already added by a new-style signature do not
// allow it to be duplicated.
if(!test->GetOldStyle())
{
cmOStringStream e;
e << " given test name \"" << args[0]
<< "\" which already exists in this directory.";
this->SetError(e.str().c_str());
return false;
}
}
else
{
test = this->Makefile->CreateTest(args[0].c_str());
test->SetOldStyle(true);
this->Makefile->AddTestGenerator(new cmTestGenerator(test));
}
test->SetCommand(command);
return true;
}
//----------------------------------------------------------------------------
bool cmAddTestCommand::HandleNameMode(std::vector<std::string> const& args)
{
std::string name;
std::vector<std::string> configurations;
std::vector<std::string> command;
// Read the arguments.
enum Doing {
DoingName,
DoingCommand,
DoingConfigs,
DoingNone
};
Doing doing = DoingName;
for(unsigned int i=1; i < args.size(); ++i)
{
if(args[i] == "COMMAND")
{
if(!command.empty())
{
this->SetError(" may be given at most one COMMAND.");
return false;
}
doing = DoingCommand;
}
else if(args[i] == "CONFIGURATIONS")
{
if(!configurations.empty())
{
this->SetError(" may be given at most one set of CONFIGURATIONS.");
return false;
}
doing = DoingConfigs;
}
else if(doing == DoingName)
{
name = args[i];
doing = DoingNone;
}
else if(doing == DoingCommand)
{
command.push_back(args[i]);
}
else if(doing == DoingConfigs)
{
configurations.push_back(args[i]);
}
else
{
cmOStringStream e;
e << " given unknown argument:\n " << args[i] << "\n";
this->SetError(e.str().c_str());
return false;
}
}
// Require a test name.
if(name.empty())
{
this->SetError(" must be given non-empty NAME.");
return false;
}
// Require a command.
if(command.empty())
{
this->SetError(" must be given non-empty COMMAND.");
return false;
}
// Require a unique test name within the directory.
if(this->Makefile->GetTest(name.c_str()))
{
cmOStringStream e;
e << " given test NAME \"" << name
<< "\" which already exists in this directory.";
this->SetError(e.str().c_str());
return false;
}
// Add the test.
cmTest* test = this->Makefile->CreateTest(name.c_str());
test->SetOldStyle(false);
test->SetCommand(command);
this->Makefile->AddTestGenerator(new cmTestGenerator(test, configurations));
return true;
}