From 56ae40cc59b74d7d4cf4c00cf35badec6aea2274 Mon Sep 17 00:00:00 2001 From: Matthew Woehlke Date: Wed, 14 Sep 2022 16:35:50 -0400 Subject: try_compile: Add PROJECT keyword-dispatched signature Introduce a new signature for the project flavor of try_compile (and try_run) which removes the `bindir` argument and adds a required PROJECT tag. This is similar to the SOURCES flavor added by commit aa9220d3 (try_compile: Add keyword-dispatched signature, 2022-09-02). --- Help/command/try_compile.rst | 18 +++- Source/cmCoreTryCompile.cxx | 100 ++++++++++++++------- .../try_compile/OldProjectBinDirEmpty-result.txt | 1 + .../try_compile/OldProjectBinDirEmpty-stderr.txt | 4 + .../try_compile/OldProjectBinDirEmpty.cmake | 1 + .../try_compile/OldProjectSrcDirEmpty-result.txt | 1 + .../try_compile/OldProjectSrcDirEmpty-stderr.txt | 4 + .../try_compile/OldProjectSrcDirEmpty.cmake | 1 + .../try_compile/ProjectBinDirEmpty-result.txt | 1 + .../try_compile/ProjectBinDirEmpty-stderr.txt | 4 + .../RunCMake/try_compile/ProjectBinDirEmpty.cmake | 4 + .../try_compile/ProjectCopyFile-result.txt | 1 + .../try_compile/ProjectCopyFile-stderr.txt | 7 ++ Tests/RunCMake/try_compile/ProjectCopyFile.cmake | 4 + .../try_compile/ProjectSrcDirEmpty-result.txt | 1 + .../try_compile/ProjectSrcDirEmpty-stderr.txt | 4 + .../RunCMake/try_compile/ProjectSrcDirEmpty.cmake | 1 + .../try_compile/ProjectSrcDirMissing-result.txt | 1 + .../try_compile/ProjectSrcDirMissing-stderr.txt | 4 + .../try_compile/ProjectSrcDirMissing.cmake | 1 + Tests/RunCMake/try_compile/RunCMakeTest.cmake | 6 ++ Tests/TryCompile/CMakeLists.txt | 11 ++- 22 files changed, 147 insertions(+), 33 deletions(-) create mode 100644 Tests/RunCMake/try_compile/OldProjectBinDirEmpty-result.txt create mode 100644 Tests/RunCMake/try_compile/OldProjectBinDirEmpty-stderr.txt create mode 100644 Tests/RunCMake/try_compile/OldProjectBinDirEmpty.cmake create mode 100644 Tests/RunCMake/try_compile/OldProjectSrcDirEmpty-result.txt create mode 100644 Tests/RunCMake/try_compile/OldProjectSrcDirEmpty-stderr.txt create mode 100644 Tests/RunCMake/try_compile/OldProjectSrcDirEmpty.cmake create mode 100644 Tests/RunCMake/try_compile/ProjectBinDirEmpty-result.txt create mode 100644 Tests/RunCMake/try_compile/ProjectBinDirEmpty-stderr.txt create mode 100644 Tests/RunCMake/try_compile/ProjectBinDirEmpty.cmake create mode 100644 Tests/RunCMake/try_compile/ProjectCopyFile-result.txt create mode 100644 Tests/RunCMake/try_compile/ProjectCopyFile-stderr.txt create mode 100644 Tests/RunCMake/try_compile/ProjectCopyFile.cmake create mode 100644 Tests/RunCMake/try_compile/ProjectSrcDirEmpty-result.txt create mode 100644 Tests/RunCMake/try_compile/ProjectSrcDirEmpty-stderr.txt create mode 100644 Tests/RunCMake/try_compile/ProjectSrcDirEmpty.cmake create mode 100644 Tests/RunCMake/try_compile/ProjectSrcDirMissing-result.txt create mode 100644 Tests/RunCMake/try_compile/ProjectSrcDirMissing-stderr.txt create mode 100644 Tests/RunCMake/try_compile/ProjectSrcDirMissing.cmake diff --git a/Help/command/try_compile.rst b/Help/command/try_compile.rst index 710fd21..97ad481 100644 --- a/Help/command/try_compile.rst +++ b/Help/command/try_compile.rst @@ -14,10 +14,15 @@ Try Compiling Whole Projects .. code-block:: cmake - try_compile( - [] [CMAKE_FLAGS ...] + try_compile( PROJECT + SOURCE_DIR + [BINARY_DIR ] + [TARGET ] + [CMAKE_FLAGS ...] [OUTPUT_VARIABLE ]) +.. versionadded:: 3.25 + Try building a project. The success or failure of the ``try_compile``, i.e. ``TRUE`` or ``FALSE`` respectively, is returned in ````. @@ -34,6 +39,15 @@ below for the meaning of other options. Previously this was only done by the :ref:`source file ` signature. +This command also supports an alternate signature +which was present in older versions of CMake: + +.. code-block:: cmake + + try_compile( + [] [CMAKE_FLAGS ...] + [OUTPUT_VARIABLE ]) + .. _`Try Compiling Source Files`: Try Compiling Source Files diff --git a/Source/cmCoreTryCompile.cxx b/Source/cmCoreTryCompile.cxx index 75aef16..2f63b3c 100644 --- a/Source/cmCoreTryCompile.cxx +++ b/Source/cmCoreTryCompile.cxx @@ -121,6 +121,26 @@ ArgumentParser::Continue TryCompileCompileDefs(Arguments& args, return ArgumentParser::Continue::Yes; } +cmArgumentParser makeTryCompileParser( + const cmArgumentParser& base) +{ + return cmArgumentParser{ base }.Bind("OUTPUT_VARIABLE"_s, + &Arguments::OutputVariable); +} + +cmArgumentParser makeTryRunParser( + const cmArgumentParser& base) +{ + return cmArgumentParser{ base } + .Bind("COMPILE_OUTPUT_VARIABLE"_s, &Arguments::CompileOutputVariable) + .Bind("RUN_OUTPUT_VARIABLE"_s, &Arguments::RunOutputVariable) + .Bind("RUN_OUTPUT_STDOUT_VARIABLE"_s, &Arguments::RunOutputStdOutVariable) + .Bind("RUN_OUTPUT_STDERR_VARIABLE"_s, &Arguments::RunOutputStdErrVariable) + .Bind("WORKING_DIRECTORY"_s, &Arguments::RunWorkingDirectory) + .Bind("ARGS"_s, &Arguments::RunArgs) + /* keep semicolon on own line */; +} + #define BIND_LANG_PROPS(lang) \ Bind(#lang "_STANDARD"_s, TryCompileLangProp) \ .Bind(#lang "_STANDARD_REQUIRED"_s, TryCompileLangProp) \ @@ -129,13 +149,17 @@ ArgumentParser::Continue TryCompileCompileDefs(Arguments& args, auto const TryCompileBaseArgParser = cmArgumentParser{} .Bind(0, &Arguments::CompileResultVariable) - .Bind("SOURCES"_s, &Arguments::Sources) .Bind("CMAKE_FLAGS"_s, &Arguments::CMakeFlags) + .Bind("__CMAKE_INTERNAL"_s, &Arguments::CMakeInternal) + /* keep semicolon on own line */; + +auto const TryCompileBaseNonProjectArgParser = + cmArgumentParser{ TryCompileBaseArgParser } + .Bind("SOURCES"_s, &Arguments::Sources) .Bind("COMPILE_DEFINITIONS"_s, TryCompileCompileDefs, ArgumentParser::ExpectAtLeast{ 0 }) .Bind("LINK_LIBRARIES"_s, &Arguments::LinkLibraries) .Bind("LINK_OPTIONS"_s, &Arguments::LinkOptions) - .Bind("__CMAKE_INTERNAL"_s, &Arguments::CMakeInternal) .Bind("COPY_FILE"_s, &Arguments::CopyFileTo) .Bind("COPY_FILE_ERROR"_s, &Arguments::CopyFileError) .BIND_LANG_PROPS(C) @@ -146,38 +170,35 @@ auto const TryCompileBaseArgParser = .BIND_LANG_PROPS(OBJCXX) /* keep semicolon on own line */; -auto const TryCompileArgParser = - cmArgumentParser{ TryCompileBaseArgParser }.Bind( - "OUTPUT_VARIABLE"_s, &Arguments::OutputVariable) +auto const TryCompileBaseProjectArgParser = + cmArgumentParser{ TryCompileBaseArgParser } + .Bind("PROJECT"_s, &Arguments::ProjectName) + .Bind("SOURCE_DIR"_s, &Arguments::SourceDirectoryOrFile) + .Bind("BINARY_DIR"_s, &Arguments::BinaryDirectory) + .Bind("TARGET"_s, &Arguments::TargetName) /* keep semicolon on own line */; +auto const TryCompileProjectArgParser = + makeTryCompileParser(TryCompileBaseProjectArgParser); + +auto const TryCompileSourcesArgParser = + makeTryCompileParser(TryCompileBaseNonProjectArgParser); + auto const TryCompileOldArgParser = - cmArgumentParser{ TryCompileArgParser } + makeTryCompileParser(TryCompileBaseNonProjectArgParser) .Bind(1, &Arguments::BinaryDirectory) .Bind(2, &Arguments::SourceDirectoryOrFile) .Bind(3, &Arguments::ProjectName) .Bind(4, &Arguments::TargetName) /* keep semicolon on own line */; -auto const TryRunArgParser = - cmArgumentParser{ TryCompileBaseArgParser } - .Bind("COMPILE_OUTPUT_VARIABLE"_s, &Arguments::CompileOutputVariable) - .Bind("RUN_OUTPUT_VARIABLE"_s, &Arguments::RunOutputVariable) - .Bind("RUN_OUTPUT_STDOUT_VARIABLE"_s, &Arguments::RunOutputStdOutVariable) - .Bind("RUN_OUTPUT_STDERR_VARIABLE"_s, &Arguments::RunOutputStdErrVariable) - .Bind("WORKING_DIRECTORY"_s, &Arguments::RunWorkingDirectory) - .Bind("ARGS"_s, &Arguments::RunArgs) - /* keep semicolon on own line */; +auto const TryRunProjectArgParser = + makeTryRunParser(TryCompileBaseProjectArgParser); -auto const TryRunOldArgParser = - cmArgumentParser{ TryCompileOldArgParser } - .Bind("COMPILE_OUTPUT_VARIABLE"_s, &Arguments::CompileOutputVariable) - .Bind("RUN_OUTPUT_VARIABLE"_s, &Arguments::RunOutputVariable) - .Bind("RUN_OUTPUT_STDOUT_VARIABLE"_s, &Arguments::RunOutputStdOutVariable) - .Bind("RUN_OUTPUT_STDERR_VARIABLE"_s, &Arguments::RunOutputStdErrVariable) - .Bind("WORKING_DIRECTORY"_s, &Arguments::RunWorkingDirectory) - .Bind("ARGS"_s, &Arguments::RunArgs) - /* keep semicolon on own line */; +auto const TryRunSourcesArgParser = + makeTryRunParser(TryCompileBaseNonProjectArgParser); + +auto const TryRunOldArgParser = makeTryRunParser(TryCompileOldArgParser); #undef BIND_LANG_PROPS } @@ -203,11 +224,24 @@ Arguments cmCoreTryCompile::ParseArgs( cmRange::const_iterator> args, bool isTryRun) { std::vector unparsedArguments; - if (cmHasLiteralPrefix(*(++args.begin()), "SOURCE")) { - // New signature. - auto arguments = - this->ParseArgs(args, isTryRun ? TryRunArgParser : TryCompileArgParser, - unparsedArguments); + const auto& second = *(++args.begin()); + + if (second == "PROJECT") { + // New PROJECT signature. + auto arguments = this->ParseArgs( + args, isTryRun ? TryRunProjectArgParser : TryCompileProjectArgParser, + unparsedArguments); + if (!arguments.BinaryDirectory) { + arguments.BinaryDirectory = unique_binary_directory; + } + return arguments; + } + + if (cmHasLiteralPrefix(second, "SOURCE")) { + // New SOURCES signature. + auto arguments = this->ParseArgs( + args, isTryRun ? TryRunSourcesArgParser : TryCompileSourcesArgParser, + unparsedArguments); arguments.BinaryDirectory = unique_binary_directory; return arguments; } @@ -256,8 +290,14 @@ bool cmCoreTryCompile::TryCompileCode(Arguments& arguments, std::string sourceDirectory; std::string projectName; std::string targetName; - if (arguments.SourceDirectoryOrFile && arguments.ProjectName) { + if (arguments.ProjectName) { this->SrcFileSignature = false; + if (!arguments.SourceDirectoryOrFile || + arguments.SourceDirectoryOrFile->empty()) { + this->Makefile->IssueMessage(MessageType::FATAL_ERROR, + "No specified."); + return false; + } sourceDirectory = *arguments.SourceDirectoryOrFile; projectName = *arguments.ProjectName; if (arguments.TargetName) { diff --git a/Tests/RunCMake/try_compile/OldProjectBinDirEmpty-result.txt b/Tests/RunCMake/try_compile/OldProjectBinDirEmpty-result.txt new file mode 100644 index 0000000..d00491f --- /dev/null +++ b/Tests/RunCMake/try_compile/OldProjectBinDirEmpty-result.txt @@ -0,0 +1 @@ +1 diff --git a/Tests/RunCMake/try_compile/OldProjectBinDirEmpty-stderr.txt b/Tests/RunCMake/try_compile/OldProjectBinDirEmpty-stderr.txt new file mode 100644 index 0000000..e9ec450 --- /dev/null +++ b/Tests/RunCMake/try_compile/OldProjectBinDirEmpty-stderr.txt @@ -0,0 +1,4 @@ +^CMake Error at OldProjectBinDirEmpty.cmake:[0-9]+ \(try_compile\): + No specified. +Call Stack \(most recent call first\): + CMakeLists.txt:[0-9]+ \(include\)$ diff --git a/Tests/RunCMake/try_compile/OldProjectBinDirEmpty.cmake b/Tests/RunCMake/try_compile/OldProjectBinDirEmpty.cmake new file mode 100644 index 0000000..fa922d9 --- /dev/null +++ b/Tests/RunCMake/try_compile/OldProjectBinDirEmpty.cmake @@ -0,0 +1 @@ +try_compile(RESULT "" ${CMAKE_CURRENT_SOURCE_DIR}/proj Foo) diff --git a/Tests/RunCMake/try_compile/OldProjectSrcDirEmpty-result.txt b/Tests/RunCMake/try_compile/OldProjectSrcDirEmpty-result.txt new file mode 100644 index 0000000..d00491f --- /dev/null +++ b/Tests/RunCMake/try_compile/OldProjectSrcDirEmpty-result.txt @@ -0,0 +1 @@ +1 diff --git a/Tests/RunCMake/try_compile/OldProjectSrcDirEmpty-stderr.txt b/Tests/RunCMake/try_compile/OldProjectSrcDirEmpty-stderr.txt new file mode 100644 index 0000000..47dd60f --- /dev/null +++ b/Tests/RunCMake/try_compile/OldProjectSrcDirEmpty-stderr.txt @@ -0,0 +1,4 @@ +CMake Error at OldProjectSrcDirEmpty.cmake:[0-9]+ \(try_compile\): + No specified. +Call Stack \(most recent call first\): + CMakeLists.txt:3 \(include\) diff --git a/Tests/RunCMake/try_compile/OldProjectSrcDirEmpty.cmake b/Tests/RunCMake/try_compile/OldProjectSrcDirEmpty.cmake new file mode 100644 index 0000000..dfbfba6 --- /dev/null +++ b/Tests/RunCMake/try_compile/OldProjectSrcDirEmpty.cmake @@ -0,0 +1 @@ +try_compile(RESULT ${CMAKE_CURRENT_BINARY_DIR} "" Foo) diff --git a/Tests/RunCMake/try_compile/ProjectBinDirEmpty-result.txt b/Tests/RunCMake/try_compile/ProjectBinDirEmpty-result.txt new file mode 100644 index 0000000..d00491f --- /dev/null +++ b/Tests/RunCMake/try_compile/ProjectBinDirEmpty-result.txt @@ -0,0 +1 @@ +1 diff --git a/Tests/RunCMake/try_compile/ProjectBinDirEmpty-stderr.txt b/Tests/RunCMake/try_compile/ProjectBinDirEmpty-stderr.txt new file mode 100644 index 0000000..57a2bd0 --- /dev/null +++ b/Tests/RunCMake/try_compile/ProjectBinDirEmpty-stderr.txt @@ -0,0 +1,4 @@ +^CMake Error at ProjectBinDirEmpty.cmake:[0-9]+ \(try_compile\): + No specified. +Call Stack \(most recent call first\): + CMakeLists.txt:[0-9]+ \(include\)$ diff --git a/Tests/RunCMake/try_compile/ProjectBinDirEmpty.cmake b/Tests/RunCMake/try_compile/ProjectBinDirEmpty.cmake new file mode 100644 index 0000000..e867cc6 --- /dev/null +++ b/Tests/RunCMake/try_compile/ProjectBinDirEmpty.cmake @@ -0,0 +1,4 @@ +try_compile(RESULT PROJECT Foo + SOURCE_DIR ${CMAKE_CURRENT_SOURCE_DIR}/proj + BINARY_DIR "" + ) diff --git a/Tests/RunCMake/try_compile/ProjectCopyFile-result.txt b/Tests/RunCMake/try_compile/ProjectCopyFile-result.txt new file mode 100644 index 0000000..573541a --- /dev/null +++ b/Tests/RunCMake/try_compile/ProjectCopyFile-result.txt @@ -0,0 +1 @@ +0 diff --git a/Tests/RunCMake/try_compile/ProjectCopyFile-stderr.txt b/Tests/RunCMake/try_compile/ProjectCopyFile-stderr.txt new file mode 100644 index 0000000..45241c9 --- /dev/null +++ b/Tests/RunCMake/try_compile/ProjectCopyFile-stderr.txt @@ -0,0 +1,7 @@ +CMake Warning \(dev\) at ProjectCopyFile.cmake:[0-9]+ \(try_compile\): + Unknown arguments: + + "COPY_FILE" + "result" +Call Stack \(most recent call first\): + CMakeLists.txt:3 \(include\) diff --git a/Tests/RunCMake/try_compile/ProjectCopyFile.cmake b/Tests/RunCMake/try_compile/ProjectCopyFile.cmake new file mode 100644 index 0000000..6bfec99 --- /dev/null +++ b/Tests/RunCMake/try_compile/ProjectCopyFile.cmake @@ -0,0 +1,4 @@ +try_compile(RESULT + PROJECT TestProject + SOURCE_DIR ${CMAKE_CURRENT_SOURCE_DIR}/proj + COPY_FILE result) diff --git a/Tests/RunCMake/try_compile/ProjectSrcDirEmpty-result.txt b/Tests/RunCMake/try_compile/ProjectSrcDirEmpty-result.txt new file mode 100644 index 0000000..d00491f --- /dev/null +++ b/Tests/RunCMake/try_compile/ProjectSrcDirEmpty-result.txt @@ -0,0 +1 @@ +1 diff --git a/Tests/RunCMake/try_compile/ProjectSrcDirEmpty-stderr.txt b/Tests/RunCMake/try_compile/ProjectSrcDirEmpty-stderr.txt new file mode 100644 index 0000000..dc6f354 --- /dev/null +++ b/Tests/RunCMake/try_compile/ProjectSrcDirEmpty-stderr.txt @@ -0,0 +1,4 @@ +CMake Error at ProjectSrcDirEmpty.cmake:[0-9]+ \(try_compile\): + No specified. +Call Stack \(most recent call first\): + CMakeLists.txt:3 \(include\) diff --git a/Tests/RunCMake/try_compile/ProjectSrcDirEmpty.cmake b/Tests/RunCMake/try_compile/ProjectSrcDirEmpty.cmake new file mode 100644 index 0000000..ac33ed0 --- /dev/null +++ b/Tests/RunCMake/try_compile/ProjectSrcDirEmpty.cmake @@ -0,0 +1 @@ +try_compile(RESULT PROJECT Foo SOURCE_DIR "") diff --git a/Tests/RunCMake/try_compile/ProjectSrcDirMissing-result.txt b/Tests/RunCMake/try_compile/ProjectSrcDirMissing-result.txt new file mode 100644 index 0000000..d00491f --- /dev/null +++ b/Tests/RunCMake/try_compile/ProjectSrcDirMissing-result.txt @@ -0,0 +1 @@ +1 diff --git a/Tests/RunCMake/try_compile/ProjectSrcDirMissing-stderr.txt b/Tests/RunCMake/try_compile/ProjectSrcDirMissing-stderr.txt new file mode 100644 index 0000000..af6edd5 --- /dev/null +++ b/Tests/RunCMake/try_compile/ProjectSrcDirMissing-stderr.txt @@ -0,0 +1,4 @@ +CMake Error at ProjectSrcDirMissing.cmake:[0-9]+ \(try_compile\): + No specified. +Call Stack \(most recent call first\): + CMakeLists.txt:3 \(include\) diff --git a/Tests/RunCMake/try_compile/ProjectSrcDirMissing.cmake b/Tests/RunCMake/try_compile/ProjectSrcDirMissing.cmake new file mode 100644 index 0000000..e32cdf4 --- /dev/null +++ b/Tests/RunCMake/try_compile/ProjectSrcDirMissing.cmake @@ -0,0 +1 @@ +try_compile(RESULT PROJECT Foo) diff --git a/Tests/RunCMake/try_compile/RunCMakeTest.cmake b/Tests/RunCMake/try_compile/RunCMakeTest.cmake index c0fdd9f..bb11a57 100644 --- a/Tests/RunCMake/try_compile/RunCMakeTest.cmake +++ b/Tests/RunCMake/try_compile/RunCMakeTest.cmake @@ -6,6 +6,11 @@ run_cmake(TwoArgs) run_cmake(NoSources) run_cmake(BinDirEmpty) run_cmake(BinDirRelative) +run_cmake(ProjectSrcDirMissing) +run_cmake(ProjectSrcDirEmpty) +run_cmake(ProjectBinDirEmpty) +run_cmake(OldProjectSrcDirEmpty) +run_cmake(OldProjectBinDirEmpty) set(RunCMake_TEST_OPTIONS -Dtry_compile_DEFS=old_signature.cmake) include(${RunCMake_SOURCE_DIR}/old_and_new_signature_tests.cmake) @@ -15,6 +20,7 @@ set(RunCMake_TEST_OPTIONS -Dtry_compile_DEFS=new_signature.cmake) include(${RunCMake_SOURCE_DIR}/old_and_new_signature_tests.cmake) unset(RunCMake_TEST_OPTIONS) +run_cmake(ProjectCopyFile) run_cmake(NonSourceCopyFile) run_cmake(NonSourceCompileDefinitions) diff --git a/Tests/TryCompile/CMakeLists.txt b/Tests/TryCompile/CMakeLists.txt index a53dd93..8ebb00a 100644 --- a/Tests/TryCompile/CMakeLists.txt +++ b/Tests/TryCompile/CMakeLists.txt @@ -65,7 +65,7 @@ set(try_compile_run_output_var RUN_OUTPUT) include(old_and_new_signature_tests.cmake) # try to compile a project (old signature) -message("Testing try_compile project mode") +message("Testing try_compile project mode (old signature)") try_compile(TEST_INNER ${TryCompile_BINARY_DIR}/CMakeFiles/Inner ${TryCompile_SOURCE_DIR}/Inner @@ -73,6 +73,15 @@ try_compile(TEST_INNER OUTPUT_VARIABLE output) TEST_ASSERT(TEST_INNER "try_compile project mode failed:\n${output}") +# try to compile a project (new signature) +message("Testing try_compile project mode (new signature)") +try_compile(TEST_INNER + PROJECT TryCompileInner + SOURCE_DIR ${TryCompile_SOURCE_DIR}/Inner + TARGET innerexe + OUTPUT_VARIABLE output) +TEST_ASSERT(TEST_INNER "try_compile project mode failed:\n${output}") + add_executable(TryCompile pass.c) ####################################################################### -- cgit v0.12