| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145 |
- /* Distributed under the OSI-approved BSD 3-Clause License. See accompanying
- file Copyright.txt or https://cmake.org/licensing for details. */
- #include "cmCTestBuildCommand.h"
- #include <sstream>
- #include <utility>
- #include <cm/memory>
- #include <cmext/string_view>
- #include "cmArgumentParser.h"
- #include "cmCTest.h"
- #include "cmCTestBuildHandler.h"
- #include "cmCTestGenericHandler.h"
- #include "cmExecutionStatus.h"
- #include "cmGlobalGenerator.h"
- #include "cmMakefile.h"
- #include "cmMessageType.h"
- #include "cmStringAlgorithms.h"
- #include "cmSystemTools.h"
- #include "cmValue.h"
- #include "cmake.h"
- bool cmCTestBuildCommand::InitialPass(std::vector<std::string> const& args,
- cmExecutionStatus& status) const
- {
- static auto const parser =
- cmArgumentParser<BuildArguments>{ MakeHandlerParser<BuildArguments>() }
- .Bind("NUMBER_ERRORS"_s, &BuildArguments::NumberErrors)
- .Bind("NUMBER_WARNINGS"_s, &BuildArguments::NumberWarnings)
- .Bind("TARGET"_s, &BuildArguments::Target)
- .Bind("CONFIGURATION"_s, &BuildArguments::Configuration)
- .Bind("FLAGS"_s, &BuildArguments::Flags)
- .Bind("PROJECT_NAME"_s, &BuildArguments::ProjectName)
- .Bind("PARALLEL_LEVEL"_s, &BuildArguments::ParallelLevel);
- return this->Invoke(parser, args, status, [&](BuildArguments& a) {
- return this->ExecuteHandlerCommand(a, status);
- });
- }
- std::unique_ptr<cmCTestGenericHandler> cmCTestBuildCommand::InitializeHandler(
- HandlerArguments& arguments, cmExecutionStatus& status) const
- {
- cmMakefile& mf = status.GetMakefile();
- auto const& args = static_cast<BuildArguments&>(arguments);
- auto handler = cm::make_unique<cmCTestBuildHandler>(this->CTest);
- cmValue ctestBuildCommand = mf.GetDefinition("CTEST_BUILD_COMMAND");
- if (cmNonempty(ctestBuildCommand)) {
- this->CTest->SetCTestConfiguration("MakeCommand", *ctestBuildCommand,
- args.Quiet);
- } else {
- cmValue cmakeGeneratorName = mf.GetDefinition("CTEST_CMAKE_GENERATOR");
- // Build configuration is determined by: CONFIGURATION argument,
- // or CTEST_BUILD_CONFIGURATION script variable, or
- // CTEST_CONFIGURATION_TYPE script variable, or ctest -C command
- // line argument... in that order.
- //
- cmValue ctestBuildConfiguration =
- mf.GetDefinition("CTEST_BUILD_CONFIGURATION");
- std::string cmakeBuildConfiguration = cmNonempty(args.Configuration)
- ? args.Configuration
- : cmNonempty(ctestBuildConfiguration) ? *ctestBuildConfiguration
- : this->CTest->GetConfigType();
- const std::string& cmakeBuildAdditionalFlags = cmNonempty(args.Flags)
- ? args.Flags
- : mf.GetSafeDefinition("CTEST_BUILD_FLAGS");
- const std::string& cmakeBuildTarget = cmNonempty(args.Target)
- ? args.Target
- : mf.GetSafeDefinition("CTEST_BUILD_TARGET");
- if (cmNonempty(cmakeGeneratorName)) {
- if (cmakeBuildConfiguration.empty()) {
- cmakeBuildConfiguration = "Release";
- }
- auto globalGenerator =
- mf.GetCMakeInstance()->CreateGlobalGenerator(*cmakeGeneratorName);
- if (!globalGenerator) {
- std::string e = cmStrCat("could not create generator named \"",
- *cmakeGeneratorName, '"');
- mf.IssueMessage(MessageType::FATAL_ERROR, e);
- cmSystemTools::SetFatalErrorOccurred();
- return nullptr;
- }
- if (cmakeBuildConfiguration.empty()) {
- cmakeBuildConfiguration = "Debug";
- }
- std::string dir = this->CTest->GetCTestConfiguration("BuildDirectory");
- std::string buildCommand = globalGenerator->GenerateCMakeBuildCommand(
- cmakeBuildTarget, cmakeBuildConfiguration, args.ParallelLevel,
- cmakeBuildAdditionalFlags, mf.IgnoreErrorsCMP0061());
- cmCTestOptionalLog(this->CTest, HANDLER_VERBOSE_OUTPUT,
- "SetMakeCommand:" << buildCommand << "\n",
- args.Quiet);
- this->CTest->SetCTestConfiguration("MakeCommand", buildCommand,
- args.Quiet);
- } else {
- std::ostringstream ostr;
- /* clang-format off */
- ostr << "has no project to build. If this is a "
- "\"built with CMake\" project, verify that CTEST_CMAKE_GENERATOR "
- "is set. Otherwise, set CTEST_BUILD_COMMAND to build the project "
- "with a custom command line.";
- /* clang-format on */
- status.SetError(ostr.str());
- return nullptr;
- }
- }
- if (cmValue useLaunchers = mf.GetDefinition("CTEST_USE_LAUNCHERS")) {
- this->CTest->SetCTestConfiguration("UseLaunchers", *useLaunchers,
- args.Quiet);
- }
- if (cmValue labelsForSubprojects =
- mf.GetDefinition("CTEST_LABELS_FOR_SUBPROJECTS")) {
- this->CTest->SetCTestConfiguration("LabelsForSubprojects",
- *labelsForSubprojects, args.Quiet);
- }
- handler->SetQuiet(args.Quiet);
- return std::unique_ptr<cmCTestGenericHandler>(std::move(handler));
- }
- void cmCTestBuildCommand::ProcessAdditionalValues(
- cmCTestGenericHandler* generic, HandlerArguments const& arguments,
- cmExecutionStatus& status) const
- {
- cmMakefile& mf = status.GetMakefile();
- auto const& args = static_cast<BuildArguments const&>(arguments);
- auto const* handler = static_cast<cmCTestBuildHandler*>(generic);
- if (!args.NumberErrors.empty()) {
- mf.AddDefinition(args.NumberErrors,
- std::to_string(handler->GetTotalErrors()));
- }
- if (!args.NumberWarnings.empty()) {
- mf.AddDefinition(args.NumberWarnings,
- std::to_string(handler->GetTotalWarnings()));
- }
- }
|