mirror of
https://github.com/Kitware/CMake.git
synced 2026-04-22 14:23:10 -05:00
fcbc883fa3
Adds the ``--project-file`` command-line option to modify the default script name loaded by CMake and ``add_subdirectory`` to values other than ``CMakeLists.txt``. Fixes: #21570
172 lines
5.8 KiB
C++
172 lines
5.8 KiB
C++
/* Distributed under the OSI-approved BSD 3-Clause License. See accompanying
|
|
file Copyright.txt or https://cmake.org/licensing for details. */
|
|
#include "cmCTestConfigureCommand.h"
|
|
|
|
#include <cstring>
|
|
#include <sstream>
|
|
#include <utility>
|
|
#include <vector>
|
|
|
|
#include <cm/memory>
|
|
#include <cmext/string_view>
|
|
|
|
#include "cmArgumentParser.h"
|
|
#include "cmCTest.h"
|
|
#include "cmCTestConfigureHandler.h"
|
|
#include "cmCTestGenericHandler.h"
|
|
#include "cmExecutionStatus.h"
|
|
#include "cmGlobalGenerator.h"
|
|
#include "cmList.h"
|
|
#include "cmMakefile.h"
|
|
#include "cmStringAlgorithms.h"
|
|
#include "cmSystemTools.h"
|
|
#include "cmValue.h"
|
|
#include "cmake.h"
|
|
|
|
std::unique_ptr<cmCTestGenericHandler>
|
|
cmCTestConfigureCommand::InitializeHandler(HandlerArguments& arguments,
|
|
cmExecutionStatus& status) const
|
|
{
|
|
cmMakefile& mf = status.GetMakefile();
|
|
auto const& args = static_cast<ConfigureArguments&>(arguments);
|
|
cmList options;
|
|
|
|
if (!args.Options.empty()) {
|
|
options.assign(args.Options);
|
|
}
|
|
|
|
if (this->CTest->GetCTestConfiguration("BuildDirectory").empty()) {
|
|
status.SetError(
|
|
"Build directory not specified. Either use BUILD "
|
|
"argument to CTEST_CONFIGURE command or set CTEST_BINARY_DIRECTORY "
|
|
"variable");
|
|
return nullptr;
|
|
}
|
|
|
|
cmValue ctestConfigureCommand = mf.GetDefinition("CTEST_CONFIGURE_COMMAND");
|
|
|
|
if (cmNonempty(ctestConfigureCommand)) {
|
|
this->CTest->SetCTestConfiguration("ConfigureCommand",
|
|
*ctestConfigureCommand, args.Quiet);
|
|
} else {
|
|
cmValue cmakeGeneratorName = mf.GetDefinition("CTEST_CMAKE_GENERATOR");
|
|
if (cmNonempty(cmakeGeneratorName)) {
|
|
const std::string& source_dir =
|
|
this->CTest->GetCTestConfiguration("SourceDirectory");
|
|
if (source_dir.empty()) {
|
|
status.SetError(
|
|
"Source directory not specified. Either use SOURCE "
|
|
"argument to CTEST_CONFIGURE command or set CTEST_SOURCE_DIRECTORY "
|
|
"variable");
|
|
return nullptr;
|
|
}
|
|
|
|
const std::string cmlName = mf.GetSafeDefinition("CMAKE_LIST_FILE_NAME");
|
|
const std::string cmakelists_file = cmStrCat(
|
|
source_dir, "/", cmlName.empty() ? "CMakeLists.txt" : cmlName);
|
|
if (!cmSystemTools::FileExists(cmakelists_file)) {
|
|
std::ostringstream e;
|
|
e << "CMakeLists.txt file does not exist [" << cmakelists_file << "]";
|
|
status.SetError(e.str());
|
|
return nullptr;
|
|
}
|
|
|
|
bool multiConfig = false;
|
|
bool cmakeBuildTypeInOptions = false;
|
|
|
|
auto gg =
|
|
mf.GetCMakeInstance()->CreateGlobalGenerator(*cmakeGeneratorName);
|
|
if (gg) {
|
|
multiConfig = gg->IsMultiConfig();
|
|
gg.reset();
|
|
}
|
|
|
|
std::string cmakeConfigureCommand =
|
|
cmStrCat('"', cmSystemTools::GetCMakeCommand(), '"');
|
|
|
|
for (std::string const& option : options) {
|
|
cmakeConfigureCommand += " \"";
|
|
cmakeConfigureCommand += option;
|
|
cmakeConfigureCommand += "\"";
|
|
|
|
if ((nullptr != strstr(option.c_str(), "CMAKE_BUILD_TYPE=")) ||
|
|
(nullptr != strstr(option.c_str(), "CMAKE_BUILD_TYPE:STRING="))) {
|
|
cmakeBuildTypeInOptions = true;
|
|
}
|
|
}
|
|
|
|
if (!multiConfig && !cmakeBuildTypeInOptions &&
|
|
!this->CTest->GetConfigType().empty()) {
|
|
cmakeConfigureCommand += " \"-DCMAKE_BUILD_TYPE:STRING=";
|
|
cmakeConfigureCommand += this->CTest->GetConfigType();
|
|
cmakeConfigureCommand += "\"";
|
|
}
|
|
|
|
if (mf.IsOn("CTEST_USE_LAUNCHERS")) {
|
|
cmakeConfigureCommand += " \"-DCTEST_USE_LAUNCHERS:BOOL=TRUE\"";
|
|
}
|
|
|
|
cmakeConfigureCommand += " \"-G";
|
|
cmakeConfigureCommand += *cmakeGeneratorName;
|
|
cmakeConfigureCommand += "\"";
|
|
|
|
cmValue cmakeGeneratorPlatform =
|
|
mf.GetDefinition("CTEST_CMAKE_GENERATOR_PLATFORM");
|
|
if (cmNonempty(cmakeGeneratorPlatform)) {
|
|
cmakeConfigureCommand += " \"-A";
|
|
cmakeConfigureCommand += *cmakeGeneratorPlatform;
|
|
cmakeConfigureCommand += "\"";
|
|
}
|
|
|
|
cmValue cmakeGeneratorToolset =
|
|
mf.GetDefinition("CTEST_CMAKE_GENERATOR_TOOLSET");
|
|
if (cmNonempty(cmakeGeneratorToolset)) {
|
|
cmakeConfigureCommand += " \"-T";
|
|
cmakeConfigureCommand += *cmakeGeneratorToolset;
|
|
cmakeConfigureCommand += "\"";
|
|
}
|
|
|
|
cmakeConfigureCommand += " \"-S";
|
|
cmakeConfigureCommand += source_dir;
|
|
cmakeConfigureCommand += "\"";
|
|
|
|
cmakeConfigureCommand += " \"-B";
|
|
cmakeConfigureCommand +=
|
|
this->CTest->GetCTestConfiguration("BuildDirectory");
|
|
cmakeConfigureCommand += "\"";
|
|
|
|
this->CTest->SetCTestConfiguration("ConfigureCommand",
|
|
cmakeConfigureCommand, args.Quiet);
|
|
} else {
|
|
status.SetError(
|
|
"Configure command is not specified. If this is a "
|
|
"\"built with CMake\" project, set CTEST_CMAKE_GENERATOR. If not, "
|
|
"set CTEST_CONFIGURE_COMMAND.");
|
|
return nullptr;
|
|
}
|
|
}
|
|
|
|
if (cmValue labelsForSubprojects =
|
|
mf.GetDefinition("CTEST_LABELS_FOR_SUBPROJECTS")) {
|
|
this->CTest->SetCTestConfiguration("LabelsForSubprojects",
|
|
*labelsForSubprojects, args.Quiet);
|
|
}
|
|
|
|
auto handler = cm::make_unique<cmCTestConfigureHandler>(this->CTest);
|
|
handler->SetQuiet(args.Quiet);
|
|
return std::unique_ptr<cmCTestGenericHandler>(std::move(handler));
|
|
}
|
|
|
|
bool cmCTestConfigureCommand::InitialPass(std::vector<std::string> const& args,
|
|
cmExecutionStatus& status) const
|
|
{
|
|
using Args = ConfigureArguments;
|
|
static auto const parser =
|
|
cmArgumentParser<Args>{ MakeHandlerParser<Args>() } //
|
|
.Bind("OPTIONS"_s, &ConfigureArguments::Options);
|
|
|
|
return this->Invoke(parser, args, status, [&](ConfigureArguments& a) {
|
|
return this->ExecuteHandlerCommand(a, status);
|
|
});
|
|
}
|