ctest: Allow passing -j without value to choose a contextual default

Under job server integration, added by commit 80fe56c481 (ctest: Add
support for running under a make job server on POSIX systems,
2023-11-15, v3.29.0-rc1~324^2), use a very high default so that
parallelism is effectively limited only by available job server tokens.

Otherwise, choose a default limit based on the number of processors.

Also allow passing `0` to specify unbounded parallelism.

Fixes: #25739
This commit is contained in:
Brad King
2024-03-06 11:16:33 -05:00
parent bbcbcff7d9
commit 5de1e21659
40 changed files with 372 additions and 51 deletions

View File

@@ -18,7 +18,7 @@ Perform the :ref:`CTest Test Step` as a :ref:`Dashboard Client`.
[EXCLUDE_FIXTURE <regex>]
[EXCLUDE_FIXTURE_SETUP <regex>]
[EXCLUDE_FIXTURE_CLEANUP <regex>]
[PARALLEL_LEVEL <level>]
[PARALLEL_LEVEL [<level>]]
[RESOURCE_SPEC_FILE <file>]
[TEST_LOAD <threshold>]
[SCHEDULE_RANDOM <ON|OFF>]
@@ -104,9 +104,14 @@ The options are:
Same as ``EXCLUDE_FIXTURE`` except only matching cleanup tests are excluded.
``PARALLEL_LEVEL <level>``
Specify a positive number representing the number of tests to
be run in parallel.
``PARALLEL_LEVEL [<level>]``
Run tests in parallel, limited to a given level of parallelism.
.. versionadded:: 3.29
The ``<level>`` may be omitted, or ``0``, to let ctest use a default
level of parallelism, or unbounded parallelism, respectively, as
documented by the :option:`ctest --parallel` option.
``RESOURCE_SPEC_FILE <file>``
.. versionadded:: 3.16

View File

@@ -8,4 +8,15 @@ For example, if ``CTEST_PARALLEL_LEVEL`` is set to 8, CTest will run
up to 8 tests concurrently as if ``ctest`` were invoked with the
:option:`--parallel 8 <ctest --parallel>` option.
.. versionchanged:: 3.29
The value may be empty, or ``0``, to let ctest use a default level of
parallelism, or unbounded parallelism, respectively, as documented by
the :option:`ctest --parallel` option.
On Windows, environment variables cannot be set to an empty string.
CTest will interpret a whitespace-only string as empty.
In CMake 3.28 and earlier, an empty or ``0`` value was equivalent to ``1``.
See :manual:`ctest(1)` for more information on parallel test execution.

View File

@@ -118,17 +118,27 @@ Run Tests
previously interrupted. If no interruption occurred, the ``-F`` option
will have no effect.
.. option:: -j <jobs>, --parallel <jobs>
.. option:: -j [<level>], --parallel [<level>]
Run the tests in parallel using the given number of jobs.
Run tests in parallel, optionally limited to a given level of parallelism.
This option tells CTest to run the tests in parallel using given
number of jobs. This option can also be set by setting the
:envvar:`CTEST_PARALLEL_LEVEL` environment variable.
.. versionadded:: 3.29
The ``<level>`` may be omitted, or ``0``, in which case:
* Under `Job Server Integration`_, parallelism is limited by
available job tokens.
* Otherwise, if the value is omitted, parallelism is limited
by the number of processors, or 2, whichever is larger.
* Otherwise, if the value is ``0``, parallelism is unbounded.
This option may instead be specified by the :envvar:`CTEST_PARALLEL_LEVEL`
environment variable.
This option can be used with the :prop_test:`PROCESSORS` test property.
See `Label and Subproject Summary`_.
See the `Label and Subproject Summary`_.
.. option:: --resource-spec-file <file>

View File

@@ -150,6 +150,12 @@ CTest
* :manual:`ctest(1)` now supports :ref:`job server integration
<ctest-job-server-integration>` on POSIX systems.
* The :option:`ctest -j` option may now be given without a value to let
ctest choose a default level of parallelism, or with ``0`` to let ctest
use unbounded parallelism. The corresponding :envvar:`CTEST_PARALLEL_LEVEL`
environment variable, if set to the empty string, is now equivalent to
passing ``-j`` with no value.
* The :command:`ctest_test` command gained options
``INCLUDE_FROM_FILE`` and ``EXCLUDE_FROM_FILE`` to run or exclude
tests named in a file.

View File

@@ -43,6 +43,17 @@
#include "cmUVJobServerClient.h"
#include "cmWorkingDirectory.h"
namespace {
// For unspecified parallelism, limit to the number of processors,
// but with a minimum greater than 1 so there is some parallelism.
constexpr unsigned long kParallelLevelMinimum = 2u;
// For "unbounded" parallelism, limit to a very high value.
// Under a job server, parallelism is effectively limited
// only by available job server tokens.
constexpr unsigned long kParallelLevelUnbounded = 0x10000u;
}
namespace cmsys {
class RegularExpression;
}
@@ -72,6 +83,7 @@ cmCTestMultiProcessHandler::cmCTestMultiProcessHandler(
, TestHandler(handler)
, ProcessorsAvailable(cmAffinity::GetProcessorsAvailable())
, HaveAffinity(this->ProcessorsAvailable.size())
, ParallelLevelDefault(kParallelLevelMinimum)
{
}
@@ -97,9 +109,43 @@ void cmCTestMultiProcessHandler::SetTests(TestMap tests,
}
// Set the max number of tests that can be run at the same time.
void cmCTestMultiProcessHandler::SetParallelLevel(size_t level)
void cmCTestMultiProcessHandler::SetParallelLevel(cm::optional<size_t> level)
{
this->ParallelLevel = level < 1 ? 1 : level;
this->ParallelLevel = level;
if (!this->ParallelLevel) {
// '-j' was given with no value. Limit by number of processors.
cmsys::SystemInformation info;
info.RunCPUCheck();
unsigned long processorCount = info.GetNumberOfLogicalCPU();
if (cm::optional<std::string> fakeProcessorCount =
cmSystemTools::GetEnvVar(
"__CTEST_FAKE_PROCESSOR_COUNT_FOR_TESTING")) {
unsigned long pc = 0;
if (cmStrToULong(*fakeProcessorCount, &pc)) {
processorCount = pc;
} else {
cmSystemTools::Error("Failed to parse fake processor count: " +
*fakeProcessorCount);
}
}
this->ParallelLevelDefault =
std::max(kParallelLevelMinimum, processorCount);
}
}
size_t cmCTestMultiProcessHandler::GetParallelLevel() const
{
if ((this->ParallelLevel && *this->ParallelLevel == 0) ||
(!this->ParallelLevel && this->JobServerClient)) {
return kParallelLevelUnbounded;
}
if (this->ParallelLevel) {
return *this->ParallelLevel;
}
return this->ParallelLevelDefault;
}
void cmCTestMultiProcessHandler::SetTestLoad(unsigned long load)
@@ -446,10 +492,11 @@ void cmCTestMultiProcessHandler::UnlockResources(int index)
inline size_t cmCTestMultiProcessHandler::GetProcessorsUsed(int test)
{
size_t processors = static_cast<int>(this->Properties[test]->Processors);
size_t const parallelLevel = this->GetParallelLevel();
// If processors setting is set higher than the -j
// setting, we default to using all of the process slots.
if (processors > this->ParallelLevel) {
processors = this->ParallelLevel;
if (processors > parallelLevel) {
processors = parallelLevel;
}
// Cap tests that want affinity to the maximum affinity available.
if (this->HaveAffinity && processors > this->HaveAffinity &&
@@ -503,8 +550,9 @@ void cmCTestMultiProcessHandler::StartNextTests()
size_t numToStart = 0;
if (this->RunningCount < this->ParallelLevel) {
numToStart = this->ParallelLevel - this->RunningCount;
size_t const parallelLevel = this->GetParallelLevel();
if (this->RunningCount < parallelLevel) {
numToStart = parallelLevel - this->RunningCount;
}
if (numToStart == 0) {
@@ -518,7 +566,7 @@ void cmCTestMultiProcessHandler::StartNextTests()
}
bool allTestsFailedTestLoadCheck = false;
size_t minProcessorsRequired = this->ParallelLevel;
size_t minProcessorsRequired = this->GetParallelLevel();
std::string testWithMinProcessors;
cmsys::SystemInformation info;
@@ -813,7 +861,7 @@ void cmCTestMultiProcessHandler::ReadCostData()
this->Properties[index]->PreviousRuns = prev;
// When not running in parallel mode, don't use cost data
if (this->ParallelLevel > 1 && this->Properties[index] &&
if (this->GetParallelLevel() > 1 && this->Properties[index] &&
this->Properties[index]->Cost == 0) {
this->Properties[index]->Cost = cost;
}
@@ -842,7 +890,7 @@ int cmCTestMultiProcessHandler::SearchByName(std::string const& name)
void cmCTestMultiProcessHandler::CreateTestCostList()
{
if (this->ParallelLevel > 1) {
if (this->GetParallelLevel() > 1) {
this->CreateParallelTestCostList();
} else {
this->CreateSerialTestCostList();

View File

@@ -63,7 +63,7 @@ public:
// Set the tests
void SetTests(TestMap tests, PropertiesMap properties);
// Set the max number of tests that can be run at the same time.
void SetParallelLevel(size_t);
void SetParallelLevel(cm::optional<size_t> level);
void SetTestLoad(unsigned long load);
virtual void RunTests();
void PrintOutputAsJson();
@@ -201,7 +201,15 @@ private:
ResourceAvailabilityErrors;
cmCTestResourceAllocator ResourceAllocator;
std::vector<cmCTestTestHandler::cmCTestTestResult>* TestResults;
size_t ParallelLevel = 1; // max number of process that can be run at once
// Get the maximum number of processors that may be used at once.
size_t GetParallelLevel() const;
// With no '-j' option, default to serial testing.
cm::optional<size_t> ParallelLevel = 1;
// Fallback parallelism limit when '-j' is given with no value.
size_t ParallelLevelDefault;
// 'make' jobserver client. If connected, we acquire a token
// for each test before running its process.

View File

@@ -105,8 +105,8 @@ cmCTestGenericHandler* cmCTestTestCommand::InitializeHandler()
if (this->StopOnFailure) {
handler->SetOption("StopOnFailure", "ON");
}
if (!this->ParallelLevel.empty()) {
handler->SetOption("ParallelLevel", this->ParallelLevel);
if (this->ParallelLevel) {
handler->SetOption("ParallelLevel", *this->ParallelLevel);
}
if (!this->Repeat.empty()) {
handler->SetOption("Repeat", this->Repeat);

View File

@@ -8,7 +8,9 @@
#include <utility>
#include <cm/memory>
#include <cm/optional>
#include "cmArgumentParserTypes.h"
#include "cmCTestHandlerCommand.h"
#include "cmCommand.h"
@@ -56,7 +58,7 @@ protected:
std::string ExcludeFixture;
std::string ExcludeFixtureSetup;
std::string ExcludeFixtureCleanup;
std::string ParallelLevel;
cm::optional<ArgumentParser::Maybe<std::string>> ParallelLevel;
std::string Repeat;
std::string ScheduleRandom;
std::string StopTime;

View File

@@ -548,9 +548,21 @@ bool cmCTestTestHandler::ProcessOptions()
return false;
}
}
if (this->GetOption("ParallelLevel")) {
this->CTest->SetParallelLevel(
std::stoi(*this->GetOption("ParallelLevel")));
if (cmValue parallelLevel = this->GetOption("ParallelLevel")) {
if (parallelLevel.IsEmpty()) {
// An empty value tells ctest to choose a default.
this->CTest->SetParallelLevel(cm::nullopt);
} else {
// A non-empty value must be a non-negative integer.
unsigned long plevel = 0;
if (!cmStrToULong(*parallelLevel, &plevel)) {
cmCTestLog(this->CTest, ERROR_MESSAGE,
"ParallelLevel invalid value: " << *parallelLevel
<< std::endl);
return false;
}
this->CTest->SetParallelLevel(plevel);
}
}
if (this->GetOption("StopOnFailure")) {

View File

@@ -179,7 +179,7 @@ struct cmCTest::Private
int MaxTestNameWidth = 30;
int ParallelLevel = 1;
cm::optional<size_t> ParallelLevel = 1;
bool ParallelLevelSetInCli = false;
unsigned long TestLoad = 0;
@@ -380,14 +380,14 @@ cmCTest::cmCTest()
cmCTest::~cmCTest() = default;
int cmCTest::GetParallelLevel() const
cm::optional<size_t> cmCTest::GetParallelLevel() const
{
return this->Impl->ParallelLevel;
}
void cmCTest::SetParallelLevel(int level)
void cmCTest::SetParallelLevel(cm::optional<size_t> level)
{
this->Impl->ParallelLevel = level < 1 ? 1 : level;
this->Impl->ParallelLevel = level;
}
unsigned long cmCTest::GetTestLoad() const
@@ -1892,14 +1892,31 @@ bool cmCTest::HandleCommandLineArguments(size_t& i,
std::string arg = args[i];
if (this->CheckArgument(arg, "-F"_s)) {
this->Impl->Failover = true;
} else if (this->CheckArgument(arg, "-j"_s, "--parallel") &&
i < args.size() - 1) {
i++;
int plevel = atoi(args[i].c_str());
this->SetParallelLevel(plevel);
} else if (this->CheckArgument(arg, "-j"_s, "--parallel")) {
cm::optional<size_t> parallelLevel;
// No value or an empty value tells ctest to choose a default.
if (i + 1 < args.size() && !cmHasLiteralPrefix(args[i + 1], "-")) {
++i;
if (!args[i].empty()) {
// A non-empty value must be a non-negative integer.
unsigned long plevel = 0;
if (!cmStrToULong(args[i], &plevel)) {
errormsg =
cmStrCat("'", arg, "' given invalid value '", args[i], "'");
return false;
}
parallelLevel = plevel;
}
}
this->SetParallelLevel(parallelLevel);
this->Impl->ParallelLevelSetInCli = true;
} else if (cmHasPrefix(arg, "-j")) {
int plevel = atoi(arg.substr(2).c_str());
// The value must be a non-negative integer.
unsigned long plevel = 0;
if (!cmStrToULong(arg.substr(2), &plevel)) {
errormsg = cmStrCat("'", arg, "' given invalid value '", args[i], "'");
return false;
}
this->SetParallelLevel(plevel);
this->Impl->ParallelLevelSetInCli = true;
}
@@ -2799,10 +2816,20 @@ int cmCTest::Run(std::vector<std::string>& args, std::string* output)
// handle CTEST_PARALLEL_LEVEL environment variable
if (!this->Impl->ParallelLevelSetInCli) {
std::string parallel;
if (cmSystemTools::GetEnv("CTEST_PARALLEL_LEVEL", parallel)) {
int plevel = atoi(parallel.c_str());
this->SetParallelLevel(plevel);
if (cm::optional<std::string> parallelEnv =
cmSystemTools::GetEnvVar("CTEST_PARALLEL_LEVEL")) {
if (parallelEnv->empty() ||
parallelEnv->find_first_not_of(" \t") == std::string::npos) {
// An empty value tells ctest to choose a default.
this->SetParallelLevel(cm::nullopt);
} else {
// A non-empty value must be a non-negative integer.
// Otherwise, ignore it.
unsigned long plevel = 0;
if (cmStrToULong(*parallelEnv, &plevel)) {
this->SetParallelLevel(plevel);
}
}
}
}

View File

@@ -12,6 +12,7 @@
#include <string>
#include <vector>
#include <cm/optional>
#include <cm/string_view>
#include "cmDuration.h"
@@ -116,8 +117,8 @@ public:
cmDuration GetGlobalTimeout() const;
/** how many test to run at the same time */
int GetParallelLevel() const;
void SetParallelLevel(int);
cm::optional<size_t> GetParallelLevel() const;
void SetParallelLevel(cm::optional<size_t> level);
unsigned long GetTestLoad() const;
void SetTestLoad(unsigned long);

View File

@@ -48,9 +48,9 @@ const cmDocumentationEntry cmDocumentationOptions[] = {
"Truncate 'tail' (default), 'middle' or 'head' of test output once "
"maximum output size is reached" },
{ "-F", "Enable failover." },
{ "-j <jobs>, --parallel <jobs>",
"Run the tests in parallel using the "
"given number of jobs." },
{ "-j [<level>], --parallel [<level>]",
"Run tests in parallel, "
"optionally limited to a given level of parallelism." },
{ "-Q,--quiet", "Make ctest quiet." },
{ "-O <file>, --output-log <file>", "Output to log file" },
{ "--output-junit <file>", "Output test results to JUnit XML file." },

View File

@@ -0,0 +1,9 @@
Test project [^
]*/Tests/RunCMake/CTestCommandLine/Parallel-0
Start [0-9]+: test[0-9]+
Start [0-9]+: test[0-9]+
Start [0-9]+: test[0-9]+
Start [0-9]+: test[0-9]+
Start [0-9]+: test[0-9]+
Start [0-9]+: test[0-9]+
1/6 Test #[0-9]+: test[0-9]+ ............................ Passed +[0-9.]+ sec

View File

@@ -0,0 +1,7 @@
Test project [^
]*/Tests/RunCMake/CTestCommandLine/Parallel-4
Start [0-9]+: test[0-9]+
Start [0-9]+: test[0-9]+
Start [0-9]+: test[0-9]+
Start [0-9]+: test[0-9]+
1/6 Test #[0-9]+: test[0-9]+ ............................ Passed +[0-9.]+ sec

View File

@@ -0,0 +1,10 @@
Test project [^
]*/Tests/RunCMake/CTestCommandLine/Parallel-N
Test #1: test1
Test #2: test2
Test #3: test3
Test #4: test4
Test #5: test5
Test #6: test6
Total Tests: 6

View File

@@ -0,0 +1 @@
1

View File

@@ -0,0 +1 @@
^CMake Error: '--parallel' given invalid value 'bad'$

View File

@@ -0,0 +1,5 @@
Test project [^
]*/Tests/RunCMake/CTestCommandLine/Parallel-empty
Start [0-9]+: test[0-9]+
Start [0-9]+: test[0-9]+
1/6 Test #[0-9]+: test[0-9]+ ............................ Passed +[0-9.]+ sec

View File

@@ -0,0 +1,9 @@
Test project [^
]*/Tests/RunCMake/CTestCommandLine/Parallel-env-0
Start [0-9]+: test[0-9]+
Start [0-9]+: test[0-9]+
Start [0-9]+: test[0-9]+
Start [0-9]+: test[0-9]+
Start [0-9]+: test[0-9]+
Start [0-9]+: test[0-9]+
1/6 Test #[0-9]+: test[0-9]+ ............................ Passed +[0-9.]+ sec

View File

@@ -0,0 +1,6 @@
Test project [^
]*/Tests/RunCMake/CTestCommandLine/Parallel-env-3
Start [0-9]+: test[0-9]+
Start [0-9]+: test[0-9]+
Start [0-9]+: test[0-9]+
1/6 Test #[0-9]+: test[0-9]+ ............................ Passed +[0-9.]+ sec

View File

@@ -0,0 +1,4 @@
Test project [^
]*/Tests/RunCMake/CTestCommandLine/Parallel-env-bad
Start [0-9]+: test[0-9]+
1/6 Test #[0-9]+: test[0-9]+ ............................ Passed +[0-9.]+ sec

View File

@@ -0,0 +1,5 @@
Test project [^
]*/Tests/RunCMake/CTestCommandLine/Parallel-env-empty
Start [0-9]+: test[0-9]+
Start [0-9]+: test[0-9]+
1/6 Test #[0-9]+: test[0-9]+ ............................ Passed +[0-9.]+ sec

View File

@@ -0,0 +1 @@
1

View File

@@ -0,0 +1 @@
^CMake Error: '-j' given invalid value 'bad'$

View File

@@ -0,0 +1,5 @@
Test project [^
]*/Tests/RunCMake/CTestCommandLine/Parallel-j
Start [0-9]+: test[0-9]+
Start [0-9]+: test[0-9]+
1/6 Test #[0-9]+: test[0-9]+ ............................ Passed +[0-9.]+ sec

View File

@@ -240,6 +240,44 @@ add_test(Echo \"${CMAKE_COMMAND}\" -E echo \"EchoTest\")
endfunction()
run_SerialFailed()
function(run_Parallel case)
set(RunCMake_TEST_BINARY_DIR ${RunCMake_BINARY_DIR}/Parallel-${case})
set(RunCMake_TEST_NO_CLEAN 1)
file(REMOVE_RECURSE "${RunCMake_TEST_BINARY_DIR}")
file(MAKE_DIRECTORY "${RunCMake_TEST_BINARY_DIR}")
file(WRITE "${RunCMake_TEST_BINARY_DIR}/CTestTestfile.cmake" "
foreach(i RANGE 1 6)
add_test(test\${i} \"${CMAKE_COMMAND}\" -E true)
endforeach()
")
run_cmake_command(Parallel-${case} ${CMAKE_CTEST_COMMAND} ${ARGN})
endfunction()
# Spoof a number of processors to make these tests predictable.
set(ENV{__CTEST_FAKE_PROCESSOR_COUNT_FOR_TESTING} 1)
run_Parallel(bad --parallel bad)
run_Parallel(j-bad -j bad)
set(RunCMake_TEST_RAW_ARGS [[--parallel ""]])
run_Parallel(empty) # With 1 processor, defaults to 2.
unset(RunCMake_TEST_RAW_ARGS)
run_Parallel(j -j) # With 1 processor, defaults to 2.
run_Parallel(0 -j0)
run_Parallel(4 --parallel 4)
run_Parallel(N --parallel -N)
set(ENV{CTEST_PARALLEL_LEVEL} bad)
run_Parallel(env-bad)
if(CMAKE_HOST_WIN32)
set(ENV{CTEST_PARALLEL_LEVEL} " ")
else()
set(ENV{CTEST_PARALLEL_LEVEL} "")
endif()
run_Parallel(env-empty) # With 1 processor, defaults to 2.
set(ENV{CTEST_PARALLEL_LEVEL} 0)
run_Parallel(env-0)
set(ENV{CTEST_PARALLEL_LEVEL} 3)
run_Parallel(env-3)
unset(ENV{CTEST_PARALLEL_LEVEL})
unset(ENV{__CTEST_FAKE_PROCESSOR_COUNT_FOR_TESTING)
function(run_TestLoad name load)
set(RunCMake_TEST_BINARY_DIR ${RunCMake_BINARY_DIR}/TestLoad)
set(RunCMake_TEST_NO_CLEAN 1)

View File

@@ -1,11 +1,11 @@
NoPipe:
env MAKEFLAGS= $(CMAKE_CTEST_COMMAND) -j6
env MAKEFLAGS= $(CMAKE_CTEST_COMMAND) -j0
.PHONY: NoPipe
NoTests:
+$(CMAKE_CTEST_COMMAND) -j6 -R NoTests
+$(CMAKE_CTEST_COMMAND) -j -R NoTests
.PHONY: NoTests
Tests:
+$(CMAKE_CTEST_COMMAND) -j6
+$(CMAKE_CTEST_COMMAND) -j
.PHONY: Tests

View File

@@ -90,10 +90,13 @@ function(run_CTestJobServer)
set(RunCMake_TEST_BINARY_DIR ${RunCMake_BINARY_DIR}/CTestJobServer-build)
run_cmake(CTestJobServer)
set(RunCMake_TEST_NO_CLEAN 1)
# Spoof a number of processors to make sure jobserver integration is unbounded.
set(ENV{__CTEST_FAKE_PROCESSOR_COUNT_FOR_TESTING} 1)
run_make_rule(CTestJobServer NoPipe 2)
run_make_rule(CTestJobServer NoTests 2)
run_make_rule(CTestJobServer Tests 2)
run_make_rule(CTestJobServer Tests 3)
unset(ENV{__CTEST_FAKE_PROCESSOR_COUNT_FOR_TESTING)
endfunction()
# Jobservers are currently only supported by GNU makes, except MSYS2 make

View File

@@ -0,0 +1,9 @@
Test project [^
]*/Tests/RunCMake/ctest_test/Parallel0-build
Start [0-9]+: test[0-9]+
Start [0-9]+: test[0-9]+
Start [0-9]+: test[0-9]+
Start [0-9]+: test[0-9]+
Start [0-9]+: test[0-9]+
Start [0-9]+: test[0-9]+
1/6 Test #[0-9]+: test[0-9]+ ............................ Passed +[0-9.]+ sec

View File

@@ -0,0 +1,7 @@
Test project [^
]*/Tests/RunCMake/ctest_test/Parallel4-build
Start [0-9]+: test[0-9]+
Start [0-9]+: test[0-9]+
Start [0-9]+: test[0-9]+
Start [0-9]+: test[0-9]+
1/6 Test #[0-9]+: test[0-9]+ ............................ Passed +[0-9.]+ sec

View File

@@ -0,0 +1 @@
(-1|255)

View File

@@ -0,0 +1 @@
^ParallelLevel invalid value: bad$

View File

@@ -0,0 +1,5 @@
Test project [^
]*/Tests/RunCMake/ctest_test/ParallelEmpty-build
Start [0-9]+: test[0-9]+
Start [0-9]+: test[0-9]+
1/6 Test #[0-9]+: test[0-9]+ ............................ Passed +[0-9.]+ sec

View File

@@ -0,0 +1,9 @@
Test project [^
]*/Tests/RunCMake/ctest_test/ParallelEnv0-build
Start [0-9]+: test[0-9]+
Start [0-9]+: test[0-9]+
Start [0-9]+: test[0-9]+
Start [0-9]+: test[0-9]+
Start [0-9]+: test[0-9]+
Start [0-9]+: test[0-9]+
1/6 Test #[0-9]+: test[0-9]+ ............................ Passed +[0-9.]+ sec

View File

@@ -0,0 +1,6 @@
Test project [^
]*/Tests/RunCMake/ctest_test/ParallelEnv3-build
Start [0-9]+: test[0-9]+
Start [0-9]+: test[0-9]+
Start [0-9]+: test[0-9]+
1/6 Test #[0-9]+: test[0-9]+ ............................ Passed +[0-9.]+ sec

View File

@@ -0,0 +1,4 @@
Test project [^
]*/Tests/RunCMake/ctest_test/ParallelEnvBad-build
Start [0-9]+: test[0-9]+
1/6 Test #[0-9]+: test[0-9]+ ............................ Passed +[0-9.]+ sec

View File

@@ -0,0 +1,5 @@
Test project [^
]*/Tests/RunCMake/ctest_test/ParallelEnvEmpty-build
Start [0-9]+: test[0-9]+
Start [0-9]+: test[0-9]+
1/6 Test #[0-9]+: test[0-9]+ ............................ Passed +[0-9.]+ sec

View File

@@ -0,0 +1,5 @@
Test project [^
]*/Tests/RunCMake/ctest_test/ParallelOmit-build
Start [0-9]+: test[0-9]+
Start [0-9]+: test[0-9]+
1/6 Test #[0-9]+: test[0-9]+ ............................ Passed +[0-9.]+ sec

View File

@@ -1,8 +1,10 @@
include(RunCTest)
set(RunCMake_TEST_TIMEOUT 60)
set(CASE_CTEST_TEST_ARGS "")
set(CASE_CTEST_TEST_LOAD "")
set(CASE_CTEST_TEST_RAW_ARGS "")
function(run_ctest_test CASE_NAME)
set(CASE_CTEST_TEST_ARGS "${ARGN}")
@@ -11,6 +13,7 @@ endfunction()
run_ctest_test(TestQuiet QUIET)
set(ENV{__CTEST_FAKE_PROCESSOR_COUNT_FOR_TESTING} 4)
set(CASE_CMAKELISTS_SUFFIX_CODE [[
foreach(i RANGE 1 6)
add_test(NAME test${i} COMMAND ${CMAKE_COMMAND} -E true)
@@ -25,6 +28,7 @@ set_property(TEST test6 PROPERTY DEPENDS test1)
]])
run_ctest_test(SerialOrder INCLUDE test)
unset(CASE_CMAKELISTS_SUFFIX_CODE)
unset(ENV{__CTEST_FAKE_PROCESSOR_COUNT_FOR_TESTING)
set(CASE_CMAKELISTS_SUFFIX_CODE [[
add_test(NAME skip COMMAND ${CMAKE_COMMAND} -E true)
@@ -33,6 +37,36 @@ set_property(TEST skip PROPERTY SKIP_RETURN_CODE 0)
run_ctest_test(SkipReturnCode)
unset(CASE_CMAKELISTS_SUFFIX_CODE)
# Spoof a number of processors to make these tests predictable.
set(ENV{__CTEST_FAKE_PROCESSOR_COUNT_FOR_TESTING} 1)
set(CASE_CMAKELISTS_SUFFIX_CODE [[
foreach(i RANGE 1 6)
add_test(NAME test${i} COMMAND ${CMAKE_COMMAND} -E true)
endforeach()
]])
run_ctest_test(ParallelBad INCLUDE test PARALLEL_LEVEL bad)
set(CASE_CTEST_TEST_RAW_ARGS "PARALLEL_LEVEL \"\"")
run_ctest_test(ParallelEmpty INCLUDE test) # With 1 processor, defaults to 2.
unset(CASE_CTEST_TEST_RAW_ARGS)
run_ctest_test(ParallelOmit INCLUDE test PARALLEL_LEVEL) # With 1 processor, defaults to 2.
run_ctest_test(Parallel0 INCLUDE test PARALLEL_LEVEL 0)
run_ctest_test(Parallel4 INCLUDE test PARALLEL_LEVEL 4)
set(ENV{CTEST_PARALLEL_LEVEL} bad)
run_ctest_test(ParallelEnvBad INCLUDE test)
if(CMAKE_HOST_WIN32)
set(ENV{CTEST_PARALLEL_LEVEL} " ")
else()
set(ENV{CTEST_PARALLEL_LEVEL} "")
endif()
run_ctest_test(ParallelEnvEmpty INCLUDE test) # With 1 processor, defaults to 2.
set(ENV{CTEST_PARALLEL_LEVEL} 0)
run_ctest_test(ParallelEnv0 INCLUDE test)
set(ENV{CTEST_PARALLEL_LEVEL} 3)
run_ctest_test(ParallelEnv3 INCLUDE test)
unset(ENV{CTEST_PARALLEL_LEVEL})
unset(CASE_CMAKELISTS_SUFFIX_CODE)
unset(ENV{__CTEST_FAKE_PROCESSOR_COUNT_FOR_TESTING)
# Tests for the 'Test Load' feature of ctest
#
# Spoof a load average value to make these tests more reliable.

View File

@@ -19,5 +19,5 @@ if("@CASE_NAME@" STREQUAL "TestChangingLabels")
ctest_test(${ctest_test_args} INCLUDE_LABEL "^a$")
ctest_test(${ctest_test_args} INCLUDE_LABEL "^b$")
else()
ctest_test(${ctest_test_args})
ctest_test(${ctest_test_args} @CASE_CTEST_TEST_RAW_ARGS@)
endif()