From c3ddb484d73c9418a4a41070666dc27b7111c9d0 Mon Sep 17 00:00:00 2001 From: Craig Scott Date: Sat, 9 Jan 2021 12:13:30 +1100 Subject: Help: Restructure and clarify cmake_path() docs Relates: #21385 --- Help/command/cmake_path.rst | 1057 +++++++++++++++++++++---------------------- 1 file changed, 523 insertions(+), 534 deletions(-) diff --git a/Help/command/cmake_path.rst b/Help/command/cmake_path.rst index 3c9653e..5d2d7eb 100644 --- a/Help/command/cmake_path.rst +++ b/Help/command/cmake_path.rst @@ -3,109 +3,32 @@ cmake_path .. versionadded:: 3.20 -Filesystem path manipulation command. - -This command is dedicated to the manipulation of objects of type path which -represent paths on a filesystem. Only syntactic aspects of paths are handled: -the pathname may represent a non-existing path or even one that is not allowed -to exist on the current file system or OS. - -For operations involving the filesystem, have a look at the :command:`file` +This command is for the manipulation of paths. Only syntactic aspects of +paths are handled, there is no interaction of any kind with any underlying +file system. The path may represent a non-existing path or even one that +is not allowed to exist on the current file system or platform. +For operations that do interact with the filesystem, see the :command:`file` command. -The path name has the following syntax: - -1. ``root-name`` (optional): identifies the root on a filesystem with multiple - roots (such as ``"C:"`` or ``"//myserver"``). - -2. ``root-directory`` (optional): a directory separator that, if present, marks - this path as absolute. If it is missing (and the first element other than - the ``root-name`` is a ``item-name``), then the path is relative. - -Zero or more of the following: - -3. ``item-name``: sequence of characters that aren't directory separators. This - name may identify a file, a hard link, a symbolic link, or a directory. Two - special ``item-names`` are recognized: - - * ``dot``: the item name consisting of a single dot character ``.`` is a - directory name that refers to the current directory. - - * ``dot-dot``: the item name consisting of two dot characters ``..`` is a - directory name that refers to the parent directory. - -4. ``directory-separator``: the forward slash character ``/``. If this - character is repeated, it is treated as a single directory separator: - ``/usr///////lib`` is the same as ``/usr/lib``. - -.. _FILENAME_DEF: - -A path has a filename if it does not ends with a ``directory-separator``. The -filename is the last ``item-name`` of the path. - -.. _EXTENSION_DEF: - -A :ref:`filename ` can have an extension. By default, the -extension is defined as the sub-string beginning at the leftmost period -(including the period) and until the end of the pathname. When the option -``LAST_ONLY`` is specified, the extension is the sub-string beginning at the -rightmost period. - -The following exceptions apply: - - * If the first character in the :ref:`filename ` is a period, - that period is ignored (a filename like ``".profile"`` is not treated as an - extension). - - * If the pathname is either ``.`` or ``..``. - .. note:: - ``cmake_path`` command handles paths in the format of the build system, not - the target system. So this is not generally applicable to the target system - in cross-compiling environment. - -For all commands, ```` placeholder expect a variable name. An error -will be raised if the variable does not exist, except for `SET`_ and `APPEND`_ -sub-commands. ```` placeholder expect a string literal. -``[...]`` placeholder expect zero or more arguments. ```` -placeholder expect a variable name. - -.. note:: - - ``cmake_path`` command does not support list of paths. The ```` - placeholder must store only one path name. - -To initialize a path variable, three possibilities can be used: - -1. :command:`set` command. -2. :ref:`cmake_path(SET) ` command. Mainly used to build a - path variable from a native path. -3. :ref:`cmake_path(APPEND) ` command. Can be used to build a path from - already available path fragments. - - .. code-block:: cmake - - # To build the path "${CMAKE_CURRENT_SOURCE_DIR}/data" - - set (path1 "${CMAKE_CURRENT_SOURCE_DIR}/data") - - cmake_path(SET path2 "${CMAKE_CURRENT_SOURCE_DIR}/data") - - cmake_path(APPEND path3 "${CMAKE_CURRENT_SOURCE_DIR}" "data") - -`Modification`_ and `Generation`_ sub-commands store the result in-place or in -the variable specified by ``OUTPUT_VARIABLE`` option. All other sub-commands -store the result in the required ```` variable. - -Sub-commands supporting ``NORMALIZE`` option will :ref:`normalize ` -the path. + The ``cmake_path`` command handles paths in the format of the build system + (i.e. the host platform), not the target system. When cross-compiling, + if the path contains elements that are not representable on the host + platform (e.g. a drive letter when the host is not Windows), the results + will be unpredictable. Synopsis ^^^^^^^^ .. parsed-literal:: + `Conventions`_ + + `Path Structure And Terminology`_ + + `Normalization`_ + `Decomposition`_ cmake_path(`GET`_ :ref:`ROOT_NAME ` ) cmake_path(`GET`_ :ref:`ROOT_DIRECTORY ` ) @@ -116,32 +39,6 @@ Synopsis cmake_path(`GET`_ :ref:`RELATIVE_PATH ` ) cmake_path(`GET`_ :ref:`PARENT_PATH ` ) - `Modification`_ - cmake_path(`SET`_ [NORMALIZE] ) - cmake_path(`APPEND`_ [...] [OUTPUT_VARIABLE ]) - cmake_path(`APPEND_STRING`_ [...] [OUTPUT_VARIABLE ]) - cmake_path(`REMOVE_FILENAME`_ [OUTPUT_VARIABLE ]) - cmake_path(`REPLACE_FILENAME`_ [OUTPUT_VARIABLE ]) - cmake_path(`REMOVE_EXTENSION`_ [LAST_ONLY] - [OUTPUT_VARIABLE ]) - cmake_path(`REPLACE_EXTENSION`_ [LAST_ONLY] - [OUTPUT_VARIABLE ]) - - `Generation`_ - cmake_path(`NORMAL_PATH`_ [OUTPUT_VARIABLE ]) - cmake_path(`RELATIVE_PATH`_ [BASE_DIRECTORY ] - [OUTPUT_VARIABLE ]) - cmake_path(`ABSOLUTE_PATH`_ [BASE_DIRECTORY ] [NORMALIZE] - [OUTPUT_VARIABLE ]) - - `Conversion`_ - cmake_path(`NATIVE_PATH`_ [NORMALIZE] ) - cmake_path(`CONVERT`_ `TO_CMAKE_PATH_LIST`_ ) - cmake_path(`CONVERT`_ `TO_NATIVE_PATH_LIST`_ ) - - `Comparison`_ - cmake_path(`COMPARE`_ ) - `Query`_ cmake_path(`HAS_ROOT_NAME`_ ) cmake_path(`HAS_ROOT_DIRECTORY`_ ) @@ -154,663 +51,755 @@ Synopsis cmake_path(`IS_ABSOLUTE`_ ) cmake_path(`IS_RELATIVE`_ ) cmake_path(`IS_PREFIX`_ [NORMALIZE] ) + cmake_path(`COMPARE`_ ) - `Hashing`_ - cmake_path(`HASH`_ ) - -Decomposition -^^^^^^^^^^^^^ - -.. _GET: -.. _GET_ROOT_NAME: - -.. code-block:: cmake - - cmake_path(GET ROOT_NAME ) - -Returns the root name of the path. If the path does not include a root name, -returns an empty path. - -.. note:: - - Only ``Windows`` system has the concept of ``root-name``, so on all other - systems, it is always an empty path. - -For example: - - .. code-block:: cmake - - set (path "c:/a") - cmake_path (GET path ROOT_NAME output) - message ("Root name is \"${output}\"") + `Modification`_ + cmake_path(:ref:`SET ` [NORMALIZE] ) + cmake_path(`APPEND`_ [...] [OUTPUT_VARIABLE ]) + cmake_path(`APPEND_STRING`_ [...] [OUTPUT_VARIABLE ]) + cmake_path(`REMOVE_FILENAME`_ [OUTPUT_VARIABLE ]) + cmake_path(`REPLACE_FILENAME`_ [OUTPUT_VARIABLE ]) + cmake_path(`REMOVE_EXTENSION`_ [LAST_ONLY] [OUTPUT_VARIABLE ]) + cmake_path(`REPLACE_EXTENSION`_ [LAST_ONLY] [OUTPUT_VARIABLE ]) - Will display:: + `Generation`_ + cmake_path(`NORMAL_PATH`_ [OUTPUT_VARIABLE ]) + cmake_path(`RELATIVE_PATH`_ [BASE_DIRECTORY ] [OUTPUT_VARIABLE ]) + cmake_path(`ABSOLUTE_PATH`_ [BASE_DIRECTORY ] [NORMALIZE] [OUTPUT_VARIABLE ]) - Root name is "c:" + `Native Conversion`_ + cmake_path(`NATIVE_PATH`_ [NORMALIZE] ) + cmake_path(`CONVERT`_ `TO_CMAKE_PATH_LIST`_ ) + cmake_path(`CONVERT`_ `TO_NATIVE_PATH_LIST`_ ) -.. _GET_ROOT_DIRECTORY: + `Hashing`_ + cmake_path(`HASH`_ ) -.. code-block:: cmake +Conventions +^^^^^^^^^^^ - cmake_path(GET ROOT_DIRECTORY ) +The following conventions are used in this command's documentation: -Returns the root directory of the path. If the path does not include a root -directory, returns an empty path. +```` + Always the name of a variable. For commands that expect a ```` + as input, the variable must exist and it is expected to hold a single path. -For example: +```` + A string literal which may contain a path, path fragment, or multiple paths + with a special separator depending on the command. See the description of + each command to see how this is interpreted. - .. code-block:: cmake +``...`` + Zero or more string literal arguments. - set (path "c:/a") - cmake_path (GET path ROOT_DIRECTORY output) - message ("Root directory is \"${output}\"") +```` + The name of a variable into which the result of a command will be written. - Will display:: - Root directory is "/" +Path Structure And Terminology +^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ -.. _GET_ROOT_PATH: +A path has the following structure (all components are optional, with some +constraints): -.. code-block:: cmake +:: - cmake_path(GET ROOT_PATH ) + root-name root-directory-separator (item-name directory-separator)* filename -Returns the root path of the path. If the path does not include a root path, -returns an empty path. +``root-name`` + Identifies the root on a filesystem with multiple roots (such as ``"C:"`` + or ``"//myserver"``). It is optional. -Effectively, returns the following: ``root-name root-directory``. +``root-directory-separator`` + A directory separator that, if present, indicates that this path is + absolute. If it is missing and the first element other than the + ``root-name`` is an ``item-name``, then the path is relative. -For example: +``item-name`` + A sequence of characters that aren't directory separators. This name may + identify a file, a hard link, a symbolic link, or a directory. Two special + cases are recognized: - .. code-block:: cmake + * The item name consisting of a single dot character ``.`` is a + directory name that refers to the current directory. - set (path "c:/a") - cmake_path (GET path ROOT_PATH output) - message ("Root path is \"${output}\"") + * The item name consisting of two dot characters ``..`` is a + directory name that refers to the parent directory. - Will display:: + The ``(...)*`` pattern shown above is to indicate that there can be zero + or more item names, with multiple items separated by a + ``directory-separator``. The ``()*`` characters are not part of the path. - Root path is "c:/" +``directory-separator`` + The only recognized directory separator is a forward slash character ``/``. + If this character is repeated, it is treated as a single directory + separator. In other words, ``/usr///////lib`` is the same as ``/usr/lib``. -.. _GET_FILENAME: +.. _FILENAME_DEF: +.. _EXTENSION_DEF: +.. _STEM_DEF: -.. code-block:: cmake +``filename`` + A path has a ``filename`` if it does not end with a ``directory-separator``. + The ``filename`` is effectively the last ``item-name`` of the path, so it + can also be a hard link, symbolic link or a directory. - cmake_path(GET FILENAME ) + A ``filename`` can have an *extension*. By default, the extension is + defined as the sub-string beginning at the left-most period (including + the period) and until the end of the ``filename``. In commands that + accept a ``LAST_ONLY`` keyword, ``LAST_ONLY`` changes the interpretation + to the sub-string beginning at the right-most period. -Returns the :ref:`filename ` component of the path. If the path -ends with a ``directory-separator``, there is no filename, so returns an empty -path. + The following exceptions apply to the above interpretation: -For example: + * If the first character in the ``filename`` is a period, that period is + ignored (i.e. a ``filename`` like ``".profile"`` is treated as having + no extension). - .. code-block:: cmake + * If the ``filename`` is either ``.`` or ``..``, it has no extension. - set (path "/a") - cmake_path (GET path FILENAME output) - message ("First filename is \"${output}\"") + The *stem* is the part of the ``filename`` before the extension. - set (path "/a/") - cmake_path (GET path FILENAME output) - message ("Second filename is \"${output}\"") +Some commands refer to a ``root-path``. This is the concatenation of +``root-name`` and ``root-directory``, either or both of which can be empty. - Will display:: - First filename is "a" - Second filename is "" +Creating A Path Variable +^^^^^^^^^^^^^^^^^^^^^^^^ -.. _GET_EXTENSION: +While a path can be created with care using an ordinary :command:`set` +command, it is recommended to use :ref:`cmake_path(SET) ` +instead, as it automatically converts the path to the required form where +required. The :ref:`cmake_path(APPEND) ` subcommand may +be another suitable alternative where a path needs to be constructed by +joining fragments. The following example compares the three methods for +constructing the same path: .. code-block:: cmake - cmake_path(GET EXTENSION [LAST_ONLY] ) - -Returns the :ref:`extension ` of the filename component. - -If the :ref:`filename ` component of the path contains a period -(``.``), and is not one of the special filesystem elements ``dot`` or -``dot-dot``, then the :ref:`extension ` is returned. + set(path1 "${CMAKE_CURRENT_SOURCE_DIR}/data") -For example: + cmake_path(SET path2 "${CMAKE_CURRENT_SOURCE_DIR}/data") - .. code-block:: cmake + cmake_path(APPEND path3 "${CMAKE_CURRENT_SOURCE_DIR}" "data") - set (path "name.ext1.ext2") - cmake_path (GET path EXTENSION result) - message ("Full extension is \"${result}\"") - cmake_path (GET path EXTENSION LAST_ONLY result) - message ("Last extension is \"${result}\"") +`Modification`_ and `Generation`_ sub-commands can either store the result +in-place, or in a separate variable named after an ``OUTPUT_VARIABLE`` +keyword. All other sub-commands store the result in a mandatory ```` +variable. - Will display:: +.. _Normalization: - Full extension is ".ext1.ext2" - Last extension is ".ext2" +Normalization +^^^^^^^^^^^^^ -The following exceptions apply: +Some sub-commands support *normalizing* a path. The algorithm used to +normalize a path is as follows: + +1. If the path is empty, stop (the normalized form of an empty path is + also an empty path). +2. Replace each ``directory-separator``, which may consist of multiple + separators, with a single ``/`` (``/a///b --> /a/b``). +3. Remove each solitary period (``.``) and any immediately following + ``directory-separator`` (``/a/./b/. --> /a/b``). +4. Remove each ``item-name`` (other than ``..``) that is immediately + followed by a ``directory-separator`` and a ``..``, along with any + immediately following ``directory-separator`` (``/a/b/../c --> a/c``). +5. If there is a ``root-directory``, remove any ``..`` and any + ``directory-separators`` immediately following them. The parent of the + root directory is treated as still the root directory (``/../a --> /a``). +6. If the last ``item-name`` is ``..``, remove any trailing + ``directory-separator`` (``../ --> ..``). +7. If the path is empty by this stage, add a ``dot`` (normal form of ``./`` + is ``.``). - * If the first character in the filename is a period, that period is ignored - (a filename like ``".profile"`` is not treated as an extension). - * If the pathname is either ``.`` or ``..``, or if - :ref:`filename ` component does not contain the ``.`` - character, then an empty path is returned. +Decomposition +^^^^^^^^^^^^^ +.. _GET: +.. _GET_ROOT_NAME: +.. _GET_ROOT_DIRECTORY: +.. _GET_ROOT_PATH: +.. _GET_FILENAME: +.. _GET_EXTENSION: .. _GET_STEM: -.. code-block:: cmake +The following forms of the ``GET`` subcommand each retrieve a different +component or group of components from a path. +`Path Structure And Terminology`_ defines the meaning of each path component. + +:: + cmake_path(GET ROOT_NAME ) + cmake_path(GET ROOT_DIRECTORY ) + cmake_path(GET ROOT_PATH ) + cmake_path(GET FILENAME ) + cmake_path(GET EXTENSION [LAST_ONLY] ) cmake_path(GET STEM [LAST_ONLY] ) -Returns the :ref:`filename ` component of the path stripped of -its :ref:`extension `. +If a requested component is not present in the path, an empty string will be +stored in ````. For example, only Windows systems have the concept +of a ``root-name``, so when the host machine is non-Windows, the ``ROOT_NAME`` +subcommand will always return an empty string. -For Example: - .. code-block:: cmake +Root examples +""""""""""""" - set (path "name.ext1.ext2") - cmake_path (GET path STEM result) - message ("Filename without the extension is \"${result}\"") - cmake_path (GET path STEM LAST_ONLY result) - message ("Filename whiteout the last extension is \"${result}\"") +.. code-block:: cmake - Will display:: + set(path "c:/a") - Filename without the extension is "name" - Filename without the last extension is "name.ext1" + cmake_path(GET path ROOT_NAME rootName) + cmake_path(GET path ROOT_DIRECTORY rootDir) + cmake_path(GET path ROOT_PATH rootPath) -The following exceptions apply: + message("Root name is \"${rootName}\"") + message("Root directory is \"${rootDir}\"") + message("Root path is \"${rootPath}\"") - * If the first character in the filename is a period, that period is ignored - (a filename like ``".profile"`` is not treated as an extension). +:: - * If the filename is one of the special filesystem components ``dot`` or - ``dot-dot``, or if it has no periods, the function returns the entire - :ref:`filename ` component. + Root name is "c:" + Root directory is "/" + Root path is "c:/" -.. _GET_RELATIVE_PATH: +Filename examples +""""""""""""""""" .. code-block:: cmake - cmake_path(GET RELATIVE_PATH ) - -Returns path relative to ``root-path``, that is, a pathname composed of -every component of ```` after ``root-path``. If ```` is -an empty path, returns an empty path. + set(path "/a/b") + cmake_path(GET path FILENAME filename) + message("First filename is \"${filename}\"") -For Example: + # Trailing slash means filename is empty + set(path "/a/b/") + cmake_path(GET path FILENAME filename) + message("Second filename is \"${filename}\"") - .. code-block:: cmake +:: - set (path "/a/b") - cmake_path (GET path RELATIVE_PATH result) - message ("Relative path is \"${result}\"") + First filename is "b" + Second filename is "" - set (path "/") - cmake_path (GET path RELATIVE_PATH result) - message ("Relative path is \"${result}\"") +Extension and stem examples +""""""""""""""""""""""""""" - Will display:: +.. code-block:: cmake - Relative path is "a/b" - Relative path is "" + set(path "name.ext1.ext2") -.. _GET_PARENT_PATH: + cmake_path(GET path EXTENSION fullExt) + cmake_path(GET path STEM fullStem) + message("Full extension is \"${fullExt}\"") + message("Full stem is \"${fullStem}\"") -.. code-block:: cmake + # Effect of LAST_ONLY + cmake_path(GET path EXTENSION LAST_ONLY lastExt) + cmake_path(GET path STEM LAST_ONLY lastStem) + message("Last extension is \"${lastExt}\"") + message("Last stem is \"${lastStem}\"") - cmake_path(GET PARENT_PATH ) + # Special cases + set(dotPath "/a/.") + set(dotDotPath "/a/..") + set(someMorePath "/a/.some.more") + cmake_path(GET dotPath EXTENSION dotExt) + cmake_path(GET dotPath STEM dotStem) + cmake_path(GET dotDotPath EXTENSION dotDotExt) + cmake_path(GET dotDotPath STEM dotDotStem) + cmake_path(GET dotMorePath EXTENSION someMoreExt) + cmake_path(GET dotMorePath STEM someMoreStem) + message("Dot extension is \"${dotExt}\"") + message("Dot stem is \"${dotStem}\"") + message("Dot-dot extension is \"${dotDotExt}\"") + message("Dot-dot stem is \"${dotDotStem}\"") + message(".some.more extension is \"${someMoreExt}\"") + message(".some.more stem is \"${someMoreStem}\"") -Returns the path to the parent directory. +:: -If `HAS_RELATIVE_PATH`_ sub-command returns false, the result is a copy of -````. Otherwise, the result is ```` with one fewer element. + Full extension is ".ext1.ext2" + Full stem is "name" + Last extension is ".ext2" + Last stem is "name.ext1" + Dot extension is "" + Dot stem is "." + Dot-dot extension is "" + Dot-dot stem is ".." + .some.more extension is ".more" + .some.more stem is ".some" -For Example: - .. code-block:: cmake +Relative paths +"""""""""""""" - set (path "c:/a/b") - cmake_path (GET path PARENT_PATH result) - message ("Parent path is \"${result}\"") +Two other forms of the ``GET`` subcommand interpret a path and return +another path derived from it. - set (path "c:/") - cmake_path (GET path PARENT_PATH result) - message ("Parent path is \"${result}\"") +.. _GET_RELATIVE_PATH: - Will display:: +:: - Parent path is "c:/a" - Relative path is "c:/" + cmake_path(GET RELATIVE_PATH ) -Modification -^^^^^^^^^^^^ +Returns the path with any ``root-name`` and ``root-directory-separator`` +removed. This leaves just the part of the path relative to the root +directory (or put another way, every component of ```` after +``root-path``). If ```` is an empty path, it returns an empty +path. -.. _cmake_path-SET: -.. _SET: +For example: .. code-block:: cmake - cmake_path(SET [NORMALIZE] ) - -Assign the ```` path to ````. Moreover, if ```` is a -native path, it is converted into cmake-style path with forward-slashes -(``/``). On Windows, the long filename marker is taken into account. + set(path "/a/b") + cmake_path(GET path RELATIVE_PATH result) + message("Relative path is \"${result}\"") -When ``NORMALIZE`` option is specified, the path is :ref:`normalized -` before the conversion. + set(path "/") + cmake_path(GET path RELATIVE_PATH result) + message("Relative path is \"${result}\"") -For Example: +Output:: - .. code-block:: cmake + Relative path is "a/b" + Relative path is "" - set (native_path "c:\\a\\b/..\\c") - cmake_path (SET path "${native_path}") - message ("CMake path is \"${path}\"") +.. _GET_PARENT_PATH: - cmake_path (SET path NORMALIZE "${native_path}") - message ("Normalized CMake path is \"${path}\"") +The other form returns the parent of the path:: - Will display:: + cmake_path(GET PARENT_PATH ) - CMake path is "c:/a/b/../c" - Normalized CMake path is "c:/a/c" +If the `HAS_RELATIVE_PATH`_ sub-command returns false, the result is a +copy of ````. Otherwise, the result is ```` with +one fewer element. -.. _APPEND: +For example: .. code-block:: cmake - cmake_path(APPEND [...] [OUTPUT_VARIABLE ]) + set(path "c:/a/b") + cmake_path(GET path PARENT_PATH result) + message("Parent path is \"${result}\"") -Append all the ```` arguments to the ```` using ``/`` as -``directory-separator``. + set(path "c:/") + cmake_path(GET path PARENT_PATH result) + message("Parent path is \"${result}\"") -For each ```` argument, the following algorithm (pseudo-code) applies: +Output:: - .. code-block:: cmake + Parent path is "c:/a" + Relative path is "c:/" - # is the contents of - - IF (.is_absolute() OR - (.has_root_name() AND - NOT .root_name() STREQUAL .root_name())) - replaces with - RETURN() - ENDIF() +Query +^^^^^ - IF (.has_root_directory()) - remove any root-directory and the entire relative path from - ELSEIF (.has_filename() OR - (NOT .has_root_directory() OR .is_absolute())) - appends directory-separator to - ENDIF() +Most of the ``GET`` subcommands also have corresponding ``HAS_...`` +subcommands which can be used to discover whether a particular path +component is present. `Path Structure And Terminology`_ defines the +meaning of each path component. - appends omitting any root-name to +.. _HAS_ROOT_NAME: +.. _HAS_ROOT_DIRECTORY: +.. _HAS_ROOT_PATH: +.. _HAS_FILENAME: +.. _HAS_EXTENSION: +.. _HAS_STEM: -.. _APPEND_STRING: +:: -.. code-block:: cmake + cmake_path(HAS_ROOT_NAME ) + cmake_path(HAS_ROOT_DIRECTORY ) + cmake_path(HAS_ROOT_PATH ) + cmake_path(HAS_FILENAME ) + cmake_path(HAS_EXTENSION ) + cmake_path(HAS_STEM ) - cmake_path(APPEND_STRING [...] [OUTPUT_VARIABLE ]) +Each of the above follows the predictable pattern of setting ```` +to true if the path has the associated component, or false otherwise. +In the case of ``HAS_ROOT_PATH``, a true result will only be returned if +at least one of ``root-name`` or ``root-directory`` is non-empty. -Append all the ```` arguments to the ```` without -``directory-separator``. +.. _HAS_RELATIVE_PATH: -.. _REMOVE_FILENAME: +:: -.. code-block:: cmake + cmake_path(HAS_RELATIVE_PATH ) - cmake_path(REMOVE_FILENAME [OUTPUT_VARIABLE ]) +A relative path in this context means everything after the ``root-path``, +if present. This command sets ```` to true if there is at least +one ``item-name`` or ``filename`` in the path. -Removes the :ref:`filename ` component (as returned by -:ref:`GET ... FILENAME `) from ````. +.. _HAS_PARENT_PATH: -After this function returns, if change is done in-place, `HAS_FILENAME`_ -returns false for ````. +:: -For Example: + cmake_path(HAS_PARENT_PATH ) - .. code-block:: cmake +This command sets ```` to true if ```` has parent path. +Note that the root directory is also considered to have a parent, which +will be itself. The result is true except if the path consists of just a +:ref:`filename `. - set (path "/a/b") - cmake_path (REMOVE_FILENAME path) - message ("First path is \"${path}\"") +.. _IS_ABSOLUTE: - cmake_path (REMOVE_FILENAME path) - message ("Second path is \"${result}\"") +:: - Will display:: + cmake_path(IS_ABSOLUTE ) - First path is "/a/" - Second path is "/a/" +Sets ```` to true if ```` is absolute. An absolute path +is a path that unambiguously identifies the location of a file without +reference to an additional starting location. On Windows, this means the +path must have both a ``root-name`` and a ``root-directory-separator`` to be +considered absolute. On other platforms, just a ``root-directory-separator`` +is sufficient. Note that this means on Windows, ``IS_ABSOLUTE`` can be +false while ``HAS_ROOT_DIRECTORY`` can be true. -.. _REPLACE_FILENAME: +.. _IS_RELATIVE: -.. code-block:: cmake +:: - cmake_path(REPLACE_FILENAME [OUTPUT_VARIABLE ]) + cmake_path(IS_RELATIVE ) -Replaces the :ref:`filename ` component from ```` with -````. +This will store the opposite of ``IS_ABSOLUTE`` in ````. -If ```` has no filename component (`HAS_FILENAME`_ returns false), -the path is unchanged. +.. _IS_PREFIX: -Equivalent to the following: +:: - .. code-block:: cmake + cmake_path(IS_PREFIX [NORMALIZE] ) - cmake_path(HAS_FILENAME path has_filename) - if (has_filename) - cmake_path(REMOVE_FILENAME path) - cmake_path(APPEND path "replacement"); - endif() +Checks if ```` is the prefix of ````. -.. _REMOVE_EXTENSION: +When the ``NORMALIZE`` option is specified, ```` and ```` +are :ref:`normalized ` before the check. .. code-block:: cmake - cmake_path(REMOVE_EXTENSION [LAST_ONLY] - [OUTPUT_VARIABLE ]) + set(path "/a/b/c/d") + cmake_path(IS_PREFIX path "/a/b" result) # result = true + cmake_path(IS_PREFIX path "/x/y/z" result) # result = false -Removes the :ref:`extension `, if any, from ````. + set(path "/a/b") + cmake_path(IS_PREFIX path "/a/c/../b" NORMALIZE result) # result = true -.. _REPLACE_EXTENSION: +.. _COMPARE: -.. code-block:: cmake +:: - cmake_path(REPLACE_EXTENSION [LAST_ONLY] - [OUTPUT_VARIABLE ]) + cmake_path(COMPARE EQUAL ) + cmake_path(COMPARE NOT_EQUAL ) -Replaces the :ref:`extension ` with ````. +Compares the lexical representations of two paths provided as string literals. +No normalization is performed on either path. Equality is determined +according to the following pseudo-code logic: - 1. If ```` has an :ref:`extension ` - (`HAS_EXTENSION`_ is true), it is removed. - 2. A ``dot`` character is appended to ````, if ```` is not - empty or does not begin with a ``dot`` character. - 3. ```` is appended as if `APPEND_STRING`_ was used. +:: + if(NOT .root_name() STREQUAL .root_name()) + return FALSE -Equivalent to the following: + if(.has_root_directory() XOR .has_root_directory()) + return FALSE - .. code-block:: cmake + Return FALSE if a relative portion of is not lexicographically + equal to the relative portion of . This comparison is performed path + component-wise. If all of the components compare equal, then return TRUE. - cmake_path(REMOVE_EXTENSION path) - if (NOT "input" MATCHES "^\\.") - cmake_path(APPEND_STRING path ".") - endif() - cmake_path(APPEND_STRING path "input"); +.. note:: + Unlike most other ``cmake_path()`` subcommands, the ``COMPARE`` subcommand + takes literal strings as input, not the names of variables. -Generation -^^^^^^^^^^ -.. _NORMAL_PATH: +Modification +^^^^^^^^^^^^ -.. code-block:: cmake +.. _cmake_path-SET: - cmake_path(NORMAL_PATH [OUTPUT_VARIABLE ]) +:: -Normalize ````. + cmake_path(SET [NORMALIZE] ) -A path can be normalized by following this algorithm: +Assign the ```` path to ````. If ```` is a native +path, it is converted into a cmake-style path with forward-slashes +(``/``). On Windows, the long filename marker is taken into account. - 1. If the path is empty, stop (normal form of an empty path is an empty - path). - 2. Replace each ``directory-separator`` (which may consist of multiple - separators) with a single ``/``. - 3. Replace each ``directory-separator`` character in the ``root-name`` with - ``/``. - 4. Remove each ``dot`` and any immediately following ``directory-separator``. - 5. Remove each non-dot-dot filename immediately followed by a - ``directory-separator`` and a ``dot-dot``, along with any immediately - following ``directory-separator``. - 6. If there is ``root-directory``, remove all ``dot-dots`` and any - ``directory-separators`` immediately following them. - 7. If the last filename is ``dot-dot``, remove any trailing - ``directory-separator``. - 8. If the path is empty, add a ``dot`` (normal form of ``./`` is ``.``). +When the ``NORMALIZE`` option is specified, the path is :ref:`normalized +` before the conversion. -.. _cmake_path-RELATIVE_PATH: -.. _RELATIVE_PATH: +For example: .. code-block:: cmake - cmake_path(RELATIVE_PATH [BASE_DIRECTORY ] - [OUTPUT_VARIABLE ]) - -Returns ```` made relative to ``BASE_DIRECTORY`` argument. If -``BASE_DIRECTORY`` is not specified, the default base directory will be -:variable:`CMAKE_CURRENT_SOURCE_DIR`. - -For reference, the algorithm used to compute the relative path is described -`here `_. - -.. _ABSOLUTE_PATH: + set(native_path "c:\\a\\b/..\\c") + cmake_path(SET path "${native_path}") + message("CMake path is \"${path}\"") -.. code-block:: cmake + cmake_path(SET path NORMALIZE "${native_path}") + message("Normalized CMake path is \"${path}\"") - cmake_path(ABSOLUTE_PATH [BASE_DIRECTORY ] [NORMALIZE] - [OUTPUT_VARIABLE ]) +Output:: -If ```` is a relative path (`IS_RELATIVE`_ is true), it is evaluated -relative to the given base directory specified by ``BASE_DIRECTORY`` option. + CMake path is "c:/a/b/../c" + Normalized CMake path is "c:/a/c" -If ``BASE_DIRECTORY`` is not specifired, the default base directory will be -:variable:`CMAKE_CURRENT_SOURCE_DIR`. +.. _APPEND: -When ``NORMALIZE`` option is specified, the path is :ref:`normalized -` after the path computation. +:: -Because ``cmake_path`` does not access to the filesystem, symbolic links are -not resolved. To compute a real path, use :command:`file(REAL_PATH)` -command. + cmake_path(APPEND [...] [OUTPUT_VARIABLE ]) -Conversion -^^^^^^^^^^ +Append all the ```` arguments to the ```` using ``/`` as +the ``directory-separator``. Depending on the ````, the previous +contents of ```` may be discarded. For each ```` argument, +the following algorithm (pseudo-code) applies: -.. _cmake_path-NATIVE_PATH: -.. _NATIVE_PATH: +:: -.. code-block:: cmake + # is the contents of - cmake_path(NATIVE_PATH [NORMALIZE] ) + if(.is_absolute() OR + (.has_root_name() AND + NOT .root_name() STREQUAL .root_name())) + replace with + return() + endif() -Converts a cmake-style ```` into a native -path with platform-specific slashes (``\`` on Windows and ``/`` elsewhere). + if(.has_root_directory()) + remove any root-directory and the entire relative path from + elseif(.has_filename() OR + (NOT .has_root_directory() OR .is_absolute())) + append directory-separator to + endif() -When ``NORMALIZE`` option is specified, the path is :ref:`normalized -` before the conversion. + append omitting any root-name to -.. _CONVERT: -.. _cmake_path-TO_CMAKE_PATH_LIST: -.. _TO_CMAKE_PATH_LIST: +.. _APPEND_STRING: -.. code-block:: cmake +:: - cmake_path(CONVERT TO_CMAKE_PATH_LIST [NORMALIZE]) + cmake_path(APPEND_STRING [...] [OUTPUT_VARIABLE ]) -Converts a native ```` path into cmake-style path with forward-slashes -(``/``). On Windows, the long filename marker is taken into account. The input -can be a single path or a system search path like ``$ENV{PATH}``. A search -path will be converted to a cmake-style list separated by ``;`` characters. The -result of the conversion is stored in the ```` variable. +Append all the ```` arguments to the ```` without adding any +``directory-separator``. -When ``NORMALIZE`` option is specified, the path is :ref:`normalized -` before the conversion. +.. _REMOVE_FILENAME: -.. _cmake_path-TO_NATIVE_PATH_LIST: -.. _TO_NATIVE_PATH_LIST: +:: -.. code-block:: cmake + cmake_path(REMOVE_FILENAME [OUTPUT_VARIABLE ]) - cmake_path(CONVERT TO_NATIVE_PATH_LIST [NORMALIZE]) +Removes the :ref:`filename ` component (as returned by +:ref:`GET ... FILENAME `) from ````. After removal, +any trailing ``directory-separator`` is left alone, if present. -Converts a cmake-style ```` path into a native path with -platform-specific slashes (``\`` on Windows and ``/`` elsewhere). The input can -be a single path or a cmake-style list. A list will be converted into a native -search path. The result of the conversion is stored in the ```` -variable. +If ``OUTPUT_VARIABLE`` is not given, then after this function returns, +`HAS_FILENAME`_ returns false for ````. -When ``NORMALIZE`` option is specified, the path is :ref:`normalized -` before the conversion. +For example: -For Example: +.. code-block:: cmake - .. code-block:: cmake + set(path "/a/b") + cmake_path(REMOVE_FILENAME path) + message("First path is \"${path}\"") - set (paths "/a/b/c" "/x/y/z") - cmake_path (CONVERT "${paths}" TO_NATIVE_PATH_LIST native_paths) - message ("Native path list is \"${native_paths}\"") + # filename is now already empty, the following removes nothing + cmake_path(REMOVE_FILENAME path) + message("Second path is \"${result}\"") - Will display, on Windows:: +Output:: - Native path list is "\a\b\c;\x\y\z" + First path is "/a/" + Second path is "/a/" - And on the all other systems:: +.. _REPLACE_FILENAME: - Native path list is "/a/b/c:/x/y/z" +:: -Comparison -^^^^^^^^^^ + cmake_path(REPLACE_FILENAME [OUTPUT_VARIABLE ]) -.. _COMPARE: +Replaces the :ref:`filename ` component from ```` +with ````. If ```` has no filename component (i.e. +`HAS_FILENAME`_ returns false), the path is unchanged. The operation is +equivalent to the following: .. code-block:: cmake - cmake_path(COMPARE EQUAL ) - cmake_path(COMPARE NOT_EQUAL ) + cmake_path(HAS_FILENAME path has_filename) + if(has_filename) + cmake_path(REMOVE_FILENAME path) + cmake_path(APPEND path input); + endif() + +.. _REMOVE_EXTENSION: -Compares the lexical representations of the path and another path. +:: -For testing equality, the following algorithm (pseudo-code) apply: + cmake_path(REMOVE_EXTENSION [LAST_ONLY] + [OUTPUT_VARIABLE ]) - .. code-block:: cmake +Removes the :ref:`extension `, if any, from ````. - IF (NOT .root_name() STREQUAL .root_name()) - returns FALSE - ELSEIF (.has_root_directory() XOR .has_root_directory()) - returns FALSE - ENDIF() +.. _REPLACE_EXTENSION: - returns TRUE or FALSE if the relative portion of is - lexicographically equal or not to the relative portion of . - Comparison is performed path component-wise +:: -Query -^^^^^ + cmake_path(REPLACE_EXTENSION [LAST_ONLY] + [OUTPUT_VARIABLE ]) -.. _HAS_ROOT_NAME: +Replaces the :ref:`extension ` with ````. Its effect +is equivalent to the following: .. code-block:: cmake - cmake_path(HAS_ROOT_NAME ) + cmake_path(REMOVE_EXTENSION path) + if(NOT "input" MATCHES "^\\.") + cmake_path(APPEND_STRING path ".") + endif() + cmake_path(APPEND_STRING path "input") -Checks if ```` has ``root-name``. -.. _HAS_ROOT_DIRECTORY: - -.. code-block:: cmake +Generation +^^^^^^^^^^ - cmake_path(HAS_ROOT_DIRECTORY ) +.. _NORMAL_PATH: -Checks if ```` has ``root-directory``. +:: -.. _HAS_ROOT_PATH: + cmake_path(NORMAL_PATH [OUTPUT_VARIABLE ]) -.. code-block:: cmake +Normalize ```` according the steps described in :ref:`Normalization`. - cmake_path(HAS_ROOT_PATH ) +.. _cmake_path-RELATIVE_PATH: +.. _RELATIVE_PATH: -Checks if ```` has root path. +:: -Effectively, checks if ```` has ``root-name`` and ``root-directory``. + cmake_path(RELATIVE_PATH [BASE_DIRECTORY ] + [OUTPUT_VARIABLE ]) -.. _HAS_FILENAME: +Modifies ```` to make it relative to the ``BASE_DIRECTORY`` argument. +If ``BASE_DIRECTORY`` is not specified, the default base directory will be +:variable:`CMAKE_CURRENT_SOURCE_DIR`. -.. code-block:: cmake +For reference, the algorithm used to compute the relative path is the same +as that used by C++ +`std::filesystem::path::lexically_relative +`_. - cmake_path(HAS_FILENAME ) +.. _ABSOLUTE_PATH: -Checks if ```` has a :ref:`filename `. +:: -.. _HAS_EXTENSION: + cmake_path(ABSOLUTE_PATH [BASE_DIRECTORY ] [NORMALIZE] + [OUTPUT_VARIABLE ]) -.. code-block:: cmake +If ```` is a relative path (`IS_RELATIVE`_ is true), it is evaluated +relative to the given base directory specified by ``BASE_DIRECTORY`` option. +If ``BASE_DIRECTORY`` is not specified, the default base directory will be +:variable:`CMAKE_CURRENT_SOURCE_DIR`. - cmake_path(HAS_EXTENSION ) +When the ``NORMALIZE`` option is specified, the path is :ref:`normalized +` after the path computation. -Checks if ```` has an :ref:`extension `. If the first -character in the filename is a period, it is not treated as an extension (for -example ".profile"). +Because ``cmake_path()`` does not access the filesystem, symbolic links are +not resolved. To compute a real path with symbolic links resolved, use the +:command:`file(REAL_PATH)` command instead. -.. _HAS_STEM: +Native Conversion +^^^^^^^^^^^^^^^^^ -.. code-block:: cmake +For commands in this section, *native* refers to the host platform, not the +target platform when cross-compiling. - cmake_path(HAS_STEM ) +.. _cmake_path-NATIVE_PATH: +.. _NATIVE_PATH: -Checks if ```` has stem (:ref:`GET ... STEM ` returns a non -empty path). +:: -.. _HAS_RELATIVE_PATH: + cmake_path(NATIVE_PATH [NORMALIZE] ) -.. code-block:: cmake +Converts a cmake-style ```` into a native path with +platform-specific slashes (``\`` on Windows hosts and ``/`` elsewhere). - cmake_path(HAS_RELATIVE_PATH ) +When the ``NORMALIZE`` option is specified, the path is :ref:`normalized +` before the conversion. -Checks if ```` has relative path (`GET_RELATIVE_PATH`_ returns a -non-empty path). +.. _CONVERT: +.. _cmake_path-TO_CMAKE_PATH_LIST: +.. _TO_CMAKE_PATH_LIST: -.. _HAS_PARENT_PATH: +:: -.. code-block:: cmake + cmake_path(CONVERT TO_CMAKE_PATH_LIST [NORMALIZE]) - cmake_path(HAS_PARENT_PATH ) +Converts a native ```` path into a cmake-style path with forward +slashes (``/``). On Windows hosts, the long filename marker is taken into +account. The input can be a single path or a system search path like +``$ENV{PATH}``. A search path will be converted to a cmake-style list +separated by ``;`` characters (on non-Windows platforms, this essentially +means ``:`` separators are replaced with ``;``). The result of the +conversion is stored in the ```` variable. -Checks if ```` has parent path. The result is true except if the path -is only composed of a :ref:`filename `. +When the ``NORMALIZE`` option is specified, the path is :ref:`normalized +` before the conversion. -.. _IS_ABSOLUTE: +.. note:: + Unlike most other ``cmake_path()`` subcommands, the ``CONVERT`` subcommand + takes a literal string as input, not the name of a variable. -.. code-block:: cmake +.. _cmake_path-TO_NATIVE_PATH_LIST: +.. _TO_NATIVE_PATH_LIST: - cmake_path(IS_ABSOLUTE ) +:: -Checks if ```` is absolute. + cmake_path(CONVERT TO_NATIVE_PATH_LIST [NORMALIZE]) -An absolute path is a path that unambiguously identifies the location of a file -without reference to an additional starting location. +Converts a cmake-style ```` path into a native path with +platform-specific slashes (``\`` on Windows hosts and ``/`` elsewhere). +The input can be a single path or a cmake-style list. A list will be +converted into a native search path (``;``-separated on Windows, +``:``-separated on other platforms). The result of the conversion is +stored in the ```` variable. -.. _IS_RELATIVE: +When the ``NORMALIZE`` option is specified, the path is :ref:`normalized +` before the conversion. -.. code-block:: cmake +.. note:: + Unlike most other ``cmake_path()`` subcommands, the ``CONVERT`` subcommand + takes a literal string as input, not the name of a variable. - cmake_path(IS_RELATIVE ) +For example: -Checks if path is relative (i.e. not :ref:`absolute `). +.. code-block:: cmake -.. _IS_PREFIX: + set(paths "/a/b/c" "/x/y/z") + cmake_path(CONVERT "${paths}" TO_NATIVE_PATH_LIST native_paths) + message("Native path list is \"${native_paths}\"") -.. code-block:: cmake +Output on Windows:: - cmake_path(IS_PREFIX [NORMALIZE] ) + Native path list is "\a\b\c;\x\y\z" -Checks if ```` is the prefix of ````. +Output on all other platforms:: -When ``NORMALIZE`` option is specified, the paths are :ref:`normalized -` before the check. + Native path list is "/a/b/c:/x/y/z" Hashing ^^^^^^^ .. _HASH: -.. code-block:: cmake +:: cmake_path(HASH ) -Compute hash value of ```` such that if for two paths (``p1`` and -``p2``) are equal (:ref:`COMPARE ... EQUAL `) then hash value of p1 is -equal to hash value of p2. - -Path is always :ref:`normalized ` before the hash is computed. +Compute a hash value of ```` such that for two paths ``p1`` and +``p2`` that compare equal (:ref:`COMPARE ... EQUAL `), the hash +value of ``p1`` is equal to the hash value of ``p2``. The path is always +:ref:`normalized ` before the hash is computed. -- cgit v0.12