cmake/Source/cmTestGenerator.cxx

232 lines
6.9 KiB
C++
Raw Normal View History

2016-10-30 18:24:19 +01:00
/* Distributed under the OSI-approved BSD 3-Clause License. See accompanying
file Copyright.txt or https://cmake.org/licensing for details. */
2009-10-04 10:30:41 +03:00
#include "cmTestGenerator.h"
2020-02-01 23:06:01 +01:00
#include <memory>
2017-04-14 19:02:05 +02:00
#include <ostream>
#include <utility>
2020-02-01 23:06:01 +01:00
#include <vector>
2017-04-14 19:02:05 +02:00
2009-10-04 10:30:41 +03:00
#include "cmGeneratorExpression.h"
2016-10-30 18:24:19 +01:00
#include "cmGeneratorTarget.h"
2019-11-11 23:01:05 +01:00
#include "cmListFileCache.h"
2015-11-17 17:22:37 +01:00
#include "cmLocalGenerator.h"
2016-07-09 11:21:54 +02:00
#include "cmOutputConverter.h"
2020-08-30 11:54:41 +02:00
#include "cmProperty.h"
2016-10-30 18:24:19 +01:00
#include "cmPropertyMap.h"
2019-11-11 23:01:05 +01:00
#include "cmRange.h"
2017-04-14 19:02:05 +02:00
#include "cmStateTypes.h"
2020-02-01 23:06:01 +01:00
#include "cmStringAlgorithms.h"
2009-10-04 10:30:41 +03:00
#include "cmSystemTools.h"
#include "cmTest.h"
2016-07-09 11:21:54 +02:00
cmTestGenerator::cmTestGenerator(
cmTest* test, std::vector<std::string> const& configurations)
: cmScriptGenerator("CTEST_CONFIGURATION_TYPE", configurations)
, Test(test)
2009-10-04 10:30:41 +03:00
{
this->ActionsPerConfig = !test->GetOldStyle();
this->TestGenerated = false;
2018-01-26 17:06:56 +01:00
this->LG = nullptr;
2009-10-04 10:30:41 +03:00
}
2019-11-11 23:01:05 +01:00
cmTestGenerator::~cmTestGenerator() = default;
2009-10-04 10:30:41 +03:00
2015-11-17 17:22:37 +01:00
void cmTestGenerator::Compute(cmLocalGenerator* lg)
{
this->LG = lg;
}
2018-04-23 21:13:27 +02:00
bool cmTestGenerator::TestsForConfig(const std::string& config)
{
return this->GeneratesForConfig(config);
}
cmTest* cmTestGenerator::GetTest() const
{
return this->Test;
}
2017-07-20 19:35:53 +02:00
void cmTestGenerator::GenerateScriptConfigs(std::ostream& os, Indent indent)
2009-10-04 10:30:41 +03:00
{
2014-08-03 19:52:23 +02:00
// Create the tests.
2009-10-04 10:30:41 +03:00
this->cmScriptGenerator::GenerateScriptConfigs(os, indent);
}
2017-07-20 19:35:53 +02:00
void cmTestGenerator::GenerateScriptActions(std::ostream& os, Indent indent)
2009-10-04 10:30:41 +03:00
{
2016-07-09 11:21:54 +02:00
if (this->ActionsPerConfig) {
2009-10-04 10:30:41 +03:00
// This is the per-config generation in a single-configuration
// build generator case. The superclass will call our per-config
// method.
this->cmScriptGenerator::GenerateScriptActions(os, indent);
2016-07-09 11:21:54 +02:00
} else {
2009-10-04 10:30:41 +03:00
// This is an old-style test, so there is only one config.
2016-07-09 11:21:54 +02:00
// assert(this->Test->GetOldStyle());
2009-10-04 10:30:41 +03:00
this->GenerateOldStyle(os, indent);
2016-07-09 11:21:54 +02:00
}
2009-10-04 10:30:41 +03:00
}
void cmTestGenerator::GenerateScriptForConfig(std::ostream& os,
2015-04-27 22:25:09 +02:00
const std::string& config,
2017-07-20 19:35:53 +02:00
Indent indent)
2009-10-04 10:30:41 +03:00
{
this->TestGenerated = true;
// Set up generator expression evaluation context.
2015-11-17 17:22:37 +01:00
cmGeneratorExpression ge(this->Test->GetBacktrace());
2009-10-04 10:30:41 +03:00
// Start the test command.
2014-08-03 19:52:23 +02:00
os << indent << "add_test(" << this->Test->GetName() << " ";
2009-10-04 10:30:41 +03:00
2020-02-01 23:06:01 +01:00
// Evaluate command line arguments
std::vector<std::string> argv =
EvaluateCommandLineArguments(this->Test->GetCommand(), ge, config);
// Expand arguments if COMMAND_EXPAND_LISTS is set
if (this->Test->GetCommandExpandLists()) {
argv = cmExpandedLists(argv.begin(), argv.end());
// Expanding lists on an empty command may have left it empty
if (argv.empty()) {
argv.emplace_back();
}
}
2009-10-04 10:30:41 +03:00
// Check whether the command executable is a target whose name is to
// be translated.
2020-02-01 23:06:01 +01:00
std::string exe = argv[0];
2016-07-09 11:21:54 +02:00
cmGeneratorTarget* target = this->LG->FindGeneratorTargetToUse(exe);
2017-04-14 19:02:05 +02:00
if (target && target->GetType() == cmStateEnums::EXECUTABLE) {
2009-10-04 10:30:41 +03:00
// Use the target file on disk.
exe = target->GetFullPath(config);
2015-08-17 11:37:30 +02:00
// Prepend with the emulator when cross compiling if required.
2020-08-30 11:54:41 +02:00
cmProp emulator = target->GetProperty("CROSSCOMPILING_EMULATOR");
if (emulator != nullptr && !emulator->empty()) {
std::vector<std::string> emulatorWithArgs = cmExpandedList(*emulator);
2015-08-17 11:37:30 +02:00
std::string emulatorExe(emulatorWithArgs[0]);
cmSystemTools::ConvertToUnixSlashes(emulatorExe);
2015-11-17 17:22:37 +01:00
os << cmOutputConverter::EscapeForCMake(emulatorExe) << " ";
2019-11-11 23:01:05 +01:00
for (std::string const& arg : cmMakeRange(emulatorWithArgs).advance(1)) {
os << cmOutputConverter::EscapeForCMake(arg) << " ";
2015-08-17 11:37:30 +02:00
}
2009-10-04 10:30:41 +03:00
}
2016-07-09 11:21:54 +02:00
} else {
2009-10-04 10:30:41 +03:00
// Use the command name given.
cmSystemTools::ConvertToUnixSlashes(exe);
2016-07-09 11:21:54 +02:00
}
2009-10-04 10:30:41 +03:00
// Generate the command line with full escapes.
2015-11-17 17:22:37 +01:00
os << cmOutputConverter::EscapeForCMake(exe);
2020-02-01 23:06:01 +01:00
for (auto const& arg : cmMakeRange(argv).advance(1)) {
os << " " << cmOutputConverter::EscapeForCMake(arg);
2016-07-09 11:21:54 +02:00
}
2009-10-04 10:30:41 +03:00
// Finish the test command.
os << ")\n";
2014-08-03 19:52:23 +02:00
// Output properties for the test.
2019-11-11 23:01:05 +01:00
os << indent << "set_tests_properties(" << this->Test->GetName()
<< " PROPERTIES ";
2020-02-01 23:06:01 +01:00
for (auto const& i : this->Test->GetProperties().GetList()) {
2019-11-11 23:01:05 +01:00
os << " " << i.first << " "
<< cmOutputConverter::EscapeForCMake(
2020-02-01 23:06:01 +01:00
ge.Parse(i.second)->Evaluate(this->LG, config));
2016-07-09 11:21:54 +02:00
}
2019-11-11 23:01:05 +01:00
this->GenerateInternalProperties(os);
2020-08-30 11:54:41 +02:00
os << ")\n";
2009-10-04 10:30:41 +03:00
}
2017-07-20 19:35:53 +02:00
void cmTestGenerator::GenerateScriptNoConfig(std::ostream& os, Indent indent)
2011-06-19 15:41:06 +03:00
{
2014-08-03 19:52:23 +02:00
os << indent << "add_test(" << this->Test->GetName() << " NOT_AVAILABLE)\n";
2011-06-19 15:41:06 +03:00
}
bool cmTestGenerator::NeedsScriptNoConfig() const
{
2016-07-09 11:21:54 +02:00
return (this->TestGenerated && // test generated for at least one config
2011-06-19 15:41:06 +03:00
this->ActionsPerConfig && // test is config-aware
2016-07-09 11:21:54 +02:00
this->Configurations.empty() && // test runs in all configs
2011-06-19 15:41:06 +03:00
!this->ConfigurationTypes->empty()); // config-dependent command
}
2017-07-20 19:35:53 +02:00
void cmTestGenerator::GenerateOldStyle(std::ostream& fout, Indent indent)
2009-10-04 10:30:41 +03:00
{
this->TestGenerated = true;
// Get the test command line to be executed.
std::vector<std::string> const& command = this->Test->GetCommand();
std::string exe = command[0];
cmSystemTools::ConvertToUnixSlashes(exe);
fout << indent;
2014-08-03 19:52:23 +02:00
fout << "add_test(";
2009-10-04 10:30:41 +03:00
fout << this->Test->GetName() << " \"" << exe << "\"";
2019-11-11 23:01:05 +01:00
for (std::string const& arg : cmMakeRange(command).advance(1)) {
2009-10-04 10:30:41 +03:00
// Just double-quote all arguments so they are re-parsed
// correctly by the test system.
fout << " \"";
2019-11-11 23:01:05 +01:00
for (char c : arg) {
2009-10-04 10:30:41 +03:00
// Escape quotes within arguments. We should escape
// backslashes too but we cannot because it makes the result
// inconsistent with previous behavior of this command.
2018-01-26 17:06:56 +01:00
if (c == '"') {
2009-10-04 10:30:41 +03:00
fout << '\\';
}
2018-01-26 17:06:56 +01:00
fout << c;
2009-10-04 10:30:41 +03:00
}
2020-08-30 11:54:41 +02:00
fout << '"';
2016-07-09 11:21:54 +02:00
}
2020-08-30 11:54:41 +02:00
fout << ")\n";
2014-08-03 19:52:23 +02:00
// Output properties for the test.
2019-11-11 23:01:05 +01:00
fout << indent << "set_tests_properties(" << this->Test->GetName()
<< " PROPERTIES ";
2020-02-01 23:06:01 +01:00
for (auto const& i : this->Test->GetProperties().GetList()) {
2019-11-11 23:01:05 +01:00
fout << " " << i.first << " "
2020-02-01 23:06:01 +01:00
<< cmOutputConverter::EscapeForCMake(i.second);
2019-11-11 23:01:05 +01:00
}
this->GenerateInternalProperties(fout);
2020-08-30 11:54:41 +02:00
fout << ")\n";
2019-11-11 23:01:05 +01:00
}
void cmTestGenerator::GenerateInternalProperties(std::ostream& os)
{
cmListFileBacktrace bt = this->Test->GetBacktrace();
if (bt.Empty()) {
return;
}
os << " "
<< "_BACKTRACE_TRIPLES"
<< " \"";
bool prependTripleSeparator = false;
while (!bt.Empty()) {
const auto& entry = bt.Top();
if (prependTripleSeparator) {
os << ";";
2014-08-03 19:52:23 +02:00
}
2019-11-11 23:01:05 +01:00
os << entry.FilePath << ";" << entry.Line << ";" << entry.Name;
bt = bt.Pop();
prependTripleSeparator = true;
2016-07-09 11:21:54 +02:00
}
2019-11-11 23:01:05 +01:00
2020-08-30 11:54:41 +02:00
os << '"';
2009-10-04 10:30:41 +03:00
}
2020-02-01 23:06:01 +01:00
std::vector<std::string> cmTestGenerator::EvaluateCommandLineArguments(
const std::vector<std::string>& argv, cmGeneratorExpression& ge,
const std::string& config) const
{
// Evaluate executable name and arguments
auto evaluatedRange =
cmMakeRange(argv).transform([&](const std::string& arg) {
return ge.Parse(arg)->Evaluate(this->LG, config);
});
return { evaluatedRange.begin(), evaluatedRange.end() };
}