Merge branch 'mh/maint-ceil-absolute'
authorJunio C Hamano <gitster@pobox.com>
Wed, 27 Feb 2013 17:47:27 +0000 (09:47 -0800)
committerJunio C Hamano <gitster@pobox.com>
Wed, 27 Feb 2013 17:47:28 +0000 (09:47 -0800)
An earlier workaround designed to help people who list logical
directories that will not match what getcwd(3) returns in the
GIT_CEILING_DIRECTORIES had an adverse effect when it is slow to
stat and readlink a directory component of an element listed on it.

* mh/maint-ceil-absolute:
Provide a mechanism to turn off symlink resolution in ceiling paths

1  2 
Documentation/git.txt
setup.c
diff --combined Documentation/git.txt
index 0847cdcc6803f0af19d203b613222ac7e22d0cdd,6bc18014cd946e0330f9ed68aceda8ff3e659b94..79aa8cd1491993f3651488b63bd7c5ab67897d1e
@@@ -27,11 -27,11 +27,11 @@@ commands.  The link:user-manual.html[Gi
  in-depth introduction.
  
  After you mastered the basic concepts, you can come back to this
 -page to learn what commands git offers.  You can learn more about
 -individual git commands with "git help command".  linkgit:gitcli[7]
 +page to learn what commands Git offers.  You can learn more about
 +individual Git commands with "git help command".  linkgit:gitcli[7]
  manual page gives you an overview of the command line command syntax.
  
 -Formatted and hyperlinked version of the latest git documentation
 +Formatted and hyperlinked version of the latest Git documentation
  can be viewed at `http://git-htmldocs.googlecode.com/git/git.html`.
  
  ifdef::stalenotes[]
  ============
  
  You are reading the documentation for the latest (possibly
 -unreleased) version of git, that is available from 'master'
 +unreleased) version of Git, that is available from 'master'
  branch of the `git.git` repository.
  Documentation for older releases are available here:
  
 -* link:v1.7.12.1/git.html[documentation for release 1.7.12.1]
 +* link:v1.8.1.4/git.html[documentation for release 1.8.1.4]
  
  * release notes for
 +  link:RelNotes/1.8.1.4.txt[1.8.1.4],
 +  link:RelNotes/1.8.1.3.txt[1.8.1.3],
 +  link:RelNotes/1.8.1.2.txt[1.8.1.2],
 +  link:RelNotes/1.8.1.1.txt[1.8.1.1],
 +  link:RelNotes/1.8.1.txt[1.8.1].
 +
 +* link:v1.8.0.3/git.html[documentation for release 1.8.0.3]
 +
 +* release notes for
 +  link:RelNotes/1.8.0.3.txt[1.8.0.3],
 +  link:RelNotes/1.8.0.2.txt[1.8.0.2],
 +  link:RelNotes/1.8.0.1.txt[1.8.0.1],
 +  link:RelNotes/1.8.0.txt[1.8.0].
 +
 +* link:v1.7.12.4/git.html[documentation for release 1.7.12.4]
 +
 +* release notes for
 +  link:RelNotes/1.7.12.4.txt[1.7.12.4],
 +  link:RelNotes/1.7.12.3.txt[1.7.12.3],
 +  link:RelNotes/1.7.12.2.txt[1.7.12.2],
    link:RelNotes/1.7.12.1.txt[1.7.12.1],
    link:RelNotes/1.7.12.txt[1.7.12].
  
@@@ -358,12 -338,12 +358,12 @@@ endif::stalenotes[
  OPTIONS
  -------
  --version::
 -      Prints the git suite version that the 'git' program came from.
 +      Prints the Git suite version that the 'git' program came from.
  
  --help::
        Prints the synopsis and a list of the most commonly used
        commands. If the option '--all' or '-a' is given then all
 -      available commands are printed. If a git command is named this
 +      available commands are printed. If a Git command is named this
        option will bring up the manual page for that command.
  +
  Other options are available to control how the manual page is
@@@ -378,22 -358,22 +378,22 @@@ help ...`
        'git config' (subkeys separated by dots).
  
  --exec-path[=<path>]::
 -      Path to wherever your core git programs are installed.
 +      Path to wherever your core Git programs are installed.
        This can also be controlled by setting the GIT_EXEC_PATH
        environment variable. If no path is given, 'git' will print
        the current setting and then exit.
  
  --html-path::
 -      Print the path, without trailing slash, where git's HTML
 +      Print the path, without trailing slash, where Git's HTML
        documentation is installed and exit.
  
  --man-path::
        Print the manpath (see `man(1)`) for the man pages for
 -      this version of git and exit.
 +      this version of Git and exit.
  
  --info-path::
        Print the path where the Info files documenting this
 -      version of git are installed and exit.
 +      version of Git are installed and exit.
  
  -p::
  --paginate::
        below).
  
  --no-pager::
 -      Do not pipe git output into a pager.
 +      Do not pipe Git output into a pager.
  
  --git-dir=<path>::
        Set the path to the repository. This can also be controlled by
        more detailed discussion).
  
  --namespace=<path>::
 -      Set the git namespace.  See linkgit:gitnamespaces[7] for more
 +      Set the Git namespace.  See linkgit:gitnamespaces[7] for more
        details.  Equivalent to setting the `GIT_NAMESPACE` environment
        variable.
  
        directory.
  
  --no-replace-objects::
 -      Do not use replacement refs to replace git objects. See
 +      Do not use replacement refs to replace Git objects. See
        linkgit:git-replace[1] for more information.
  
 +--literal-pathspecs::
 +      Treat pathspecs literally, rather than as glob patterns. This is
 +      equivalent to setting the `GIT_LITERAL_PATHSPECS` environment
 +      variable to `1`.
 +
  
  GIT COMMANDS
  ------------
  
 -We divide git into high level ("porcelain") commands and low level
 +We divide Git into high level ("porcelain") commands and low level
  ("plumbing") commands.
  
  High-level commands (porcelain)
@@@ -478,7 -453,7 +478,7 @@@ include::cmds-foreignscminterface.txt[
  Low-level commands (plumbing)
  -----------------------------
  
 -Although git includes its
 +Although Git includes its
  own porcelain layer, its low-level commands are sufficient to support
  development of alternative porcelains.  Developers of such porcelains
  might start by reading about linkgit:git-update-index[1] and
@@@ -536,9 -511,10 +536,9 @@@ include::cmds-purehelpers.txt[
  Configuration Mechanism
  -----------------------
  
 -Starting from 0.99.9 (actually mid 0.99.8.GIT), `.git/config` file
 -is used to hold per-repository configuration options.  It is a
 -simple text file modeled after `.ini` format familiar to some
 -people.  Here is an example:
 +Git uses a simple text format to store customizations that are per
 +repository and are per user.  Such a configuration file may look
 +like this:
  
  ------------
  #
  ; user identity
  [user]
        name = "Junio C Hamano"
 -      email = "junkio@twinsun.com"
 +      email = "gitster@pobox.com"
  
  ------------
  
  Various commands read from the configuration file and adjust
  their operation accordingly.  See linkgit:git-config[1] for a
 -list.
 +list and more details about the configuration mechanism.
  
  
  Identifier Terminology
  
  Symbolic Identifiers
  --------------------
 -Any git command accepting any <object> can also use the following
 +Any Git command accepting any <object> can also use the following
  symbolic notation:
  
  HEAD::
@@@ -634,13 -610,13 +634,13 @@@ Please see linkgit:gitglossary[7]
  
  Environment Variables
  ---------------------
 -Various git commands use the following environment variables:
 +Various Git commands use the following environment variables:
  
 -The git Repository
 +The Git Repository
  ~~~~~~~~~~~~~~~~~~
 -These environment variables apply to 'all' core git commands. Nb: it
 +These environment variables apply to 'all' core Git commands. Nb: it
  is worth noting that they may be used/overridden by SCMS sitting above
 -git so take care if using Cogito etc.
 +Git so take care if using Cogito etc.
  
  'GIT_INDEX_FILE'::
        This environment allows the specification of an alternate
        directory is used.
  
  'GIT_ALTERNATE_OBJECT_DIRECTORIES'::
 -      Due to the immutable nature of git objects, old objects can be
 +      Due to the immutable nature of Git objects, old objects can be
        archived into shared, read-only directories. This variable
        specifies a ":" separated (on Windows ";" separated) list
 -      of git object directories which can be used to search for git
 +      of Git object directories which can be used to search for Git
        objects. New objects will not be written to these directories.
  
  'GIT_DIR'::
        If the 'GIT_DIR' environment variable is set then it
        specifies a path to use instead of the default `.git`
        for the base of the repository.
 +      The '--git-dir' command-line option also sets this value.
  
  'GIT_WORK_TREE'::
        Set the path to the working tree.  The value will not be
        option and the core.worktree configuration variable.
  
  'GIT_NAMESPACE'::
 -      Set the git namespace; see linkgit:gitnamespaces[7] for details.
 +      Set the Git namespace; see linkgit:gitnamespaces[7] for details.
        The '--namespace' command-line option also sets this value.
  
  'GIT_CEILING_DIRECTORIES'::
-       This should be a colon-separated list of absolute paths.
-       If set, it is a list of directories that Git should not chdir
-       up into while looking for a repository directory.
-       It will not exclude the current working directory or
-       a GIT_DIR set on the command line or in the environment.
-       (Useful for excluding slow-loading network directories.)
+       This should be a colon-separated list of absolute paths.  If
 -      set, it is a list of directories that git should not chdir up
++      set, it is a list of directories that Git should not chdir up
+       into while looking for a repository directory (useful for
+       excluding slow-loading network directories).  It will not
+       exclude the current working directory or a GIT_DIR set on the
+       command line or in the environment.  Normally, Git has to read
+       the entries in this list and resolve any symlink that
+       might be present in order to compare them with the current
+       directory.  However, if even this access is slow, you
+       can add an empty entry to the list to tell Git that the
+       subsequent entries are not symlinks and needn't be resolved;
+       e.g.,
+       'GIT_CEILING_DIRECTORIES=/maybe/symlink::/very/slow/non/symlink'.
  
  'GIT_DISCOVERY_ACROSS_FILESYSTEM'::
        When run in a directory that does not have ".git" repository
 -      directory, git tries to find such a directory in the parent
 +      directory, Git tries to find such a directory in the parent
        directories to find the top of the working tree, but by default it
        does not cross filesystem boundaries.  This environment variable
 -      can be set to true to tell git not to stop at filesystem
 +      can be set to true to tell Git not to stop at filesystem
        boundaries.  Like 'GIT_CEILING_DIRECTORIES', this will not affect
        an explicit repository directory set via 'GIT_DIR' or on the
        command line.
  
 -git Commits
 +Git Commits
  ~~~~~~~~~~~
  'GIT_AUTHOR_NAME'::
  'GIT_AUTHOR_EMAIL'::
  'EMAIL'::
        see linkgit:git-commit-tree[1]
  
 -git Diffs
 +Git Diffs
  ~~~~~~~~~
  'GIT_DIFF_OPTS'::
        Only valid setting is "--unified=??" or "-u??" to set the
        number of context lines shown when a unified diff is created.
        This takes precedence over any "-U" or "--unified" option
 -      value passed on the git diff command line.
 +      value passed on the Git diff command line.
  
  'GIT_EXTERNAL_DIFF'::
        When the environment variable 'GIT_EXTERNAL_DIFF' is set, the
@@@ -747,13 -729,13 +754,13 @@@ othe
  
  'GIT_PAGER'::
        This environment variable overrides `$PAGER`. If it is set
 -      to an empty string or to the value "cat", git will not launch
 +      to an empty string or to the value "cat", Git will not launch
        a pager.  See also the `core.pager` option in
        linkgit:git-config[1].
  
  'GIT_EDITOR'::
        This environment variable overrides `$EDITOR` and `$VISUAL`.
 -      It is used by several git commands when, on interactive mode,
 +      It is used by several Git commands when, on interactive mode,
        an editor is to be launched. See also linkgit:git-var[1]
        and the `core.editor` option in linkgit:git-config[1].
  
@@@ -774,20 -756,12 +781,20 @@@ personal `.ssh/config` file.  Please co
  for further details.
  
  'GIT_ASKPASS'::
 -      If this environment variable is set, then git commands which need to
 +      If this environment variable is set, then Git commands which need to
        acquire passwords or passphrases (e.g. for HTTP or IMAP authentication)
        will call this program with a suitable prompt as command line argument
        and read the password from its STDOUT. See also the 'core.askpass'
        option in linkgit:git-config[1].
  
 +'GIT_CONFIG_NOSYSTEM'::
 +      Whether to skip reading settings from the system-wide
 +      `$(prefix)/etc/gitconfig` file.  This environment variable can
 +      be used along with `$HOME` and `$XDG_CONFIG_HOME` to create a
 +      predictable environment for a picky script, or you can set it
 +      temporarily to avoid using a buggy `/etc/gitconfig` file while
 +      waiting for someone with sufficient permissions to fix it.
 +
  'GIT_FLUSH'::
        If this environment variable is set to "1", then commands such
        as 'git blame' (in incremental mode), 'git rev-list', 'git log',
        after each commit-oriented record have been flushed.   If this
        variable is set to "0", the output of these commands will be done
        using completely buffered I/O.   If this environment variable is
 -      not set, git will choose buffered or record-oriented flushing
 +      not set, Git will choose buffered or record-oriented flushing
        based on whether stdout appears to be redirected to a file or not.
  
  'GIT_TRACE'::
        If this variable is set to "1", "2" or "true" (comparison
 -      is case insensitive), git will print `trace:` messages on
 +      is case insensitive), Git will print `trace:` messages on
        stderr telling about alias expansion, built-in command
        execution and external command execution.
        If this variable is set to an integer value greater than 1
 -      and lower than 10 (strictly) then git will interpret this
 +      and lower than 10 (strictly) then Git will interpret this
        value as an open file descriptor and will try to write the
        trace messages into this file descriptor.
        Alternatively, if this variable is set to an absolute path
 -      (starting with a '/' character), git will interpret this
 +      (starting with a '/' character), Git will interpret this
        as a file path and will try to write the trace messages
        into it.
  
 +GIT_LITERAL_PATHSPECS::
 +      Setting this variable to `1` will cause Git to treat all
 +      pathspecs literally, rather than as glob patterns. For example,
 +      running `GIT_LITERAL_PATHSPECS=1 git log -- '*.c'` will search
 +      for commits that touch the path `*.c`, not any paths that the
 +      glob `*.c` matches. You might want this if you are feeding
 +      literal paths to Git (e.g., paths previously given to you by
 +      `git ls-tree`, `--raw` diff output, etc).
 +
 +
  Discussion[[Discussion]]
  ------------------------
  
  More detail on the following is available from the
 -link:user-manual.html#git-concepts[git concepts chapter of the
 +link:user-manual.html#git-concepts[Git concepts chapter of the
  user-manual] and linkgit:gitcore-tutorial[7].
  
 -A git project normally consists of a working directory with a ".git"
 +A Git project normally consists of a working directory with a ".git"
  subdirectory at the top level.  The .git directory contains, among other
  things, a compressed object database representing the complete history
  of the project, an "index" file which links that history to the current
@@@ -879,12 -843,12 +886,12 @@@ FURTHER DOCUMENTATIO
  ---------------------
  
  See the references in the "description" section to get started
 -using git.  The following is probably more detail than necessary
 +using Git.  The following is probably more detail than necessary
  for a first-time user.
  
 -The link:user-manual.html#git-concepts[git concepts chapter of the
 +The link:user-manual.html#git-concepts[Git concepts chapter of the
  user-manual] and linkgit:gitcore-tutorial[7] both provide
 -introductions to the underlying git architecture.
 +introductions to the underlying Git architecture.
  
  See linkgit:gitworkflows[7] for an overview of recommended workflows.
  
@@@ -892,7 -856,7 +899,7 @@@ See also the link:howto-index.html[howt
  examples.
  
  The internals are documented in the
 -link:technical/api-index.html[GIT API documentation].
 +link:technical/api-index.html[Git API documentation].
  
  Users migrating from CVS may also want to
  read linkgit:gitcvs-migration[7].
  Authors
  -------
  Git was started by Linus Torvalds, and is currently maintained by Junio
 -C Hamano. Numerous contributions have come from the git mailing list
 -<git@vger.kernel.org>. For a more complete list of contributors, see
 -http://git-scm.com/about. If you have a clone of git.git itself, the
 +C Hamano. Numerous contributions have come from the Git mailing list
 +<git@vger.kernel.org>.  http://www.ohloh.net/p/git/contributors/summary
 +gives you a more complete list of contributors.
 +
 +If you have a clone of git.git itself, the
  output of linkgit:git-shortlog[1] and linkgit:git-blame[1] can show you
  the authors for specific parts of the project.
  
diff --combined setup.c
index 2e1521b09e5e6a0de585225293805b1d30a03cd4,1b120175f2184b782f389d2fc99f3e37710f35d3..1dee47e0850369c87408c540ce2b2251db17be3d
+++ b/setup.c
@@@ -66,14 -66,7 +66,14 @@@ int check_filename(const char *prefix, 
        const char *name;
        struct stat st;
  
 -      name = prefix ? prefix_filename(prefix, strlen(prefix), arg) : arg;
 +      if (!prefixcmp(arg, ":/")) {
 +              if (arg[2] == '\0') /* ":/" is root dir, always exists */
 +                      return 1;
 +              name = arg + 2;
 +      } else if (prefix)
 +              name = prefix_filename(prefix, strlen(prefix), arg);
 +      else
 +              name = arg;
        if (!lstat(name, &st))
                return 1; /* file exists */
        if (errno == ENOENT || errno == ENOTDIR)
@@@ -253,25 -246,6 +253,25 @@@ static const char *prefix_pathspec(cons
                return prefix_path(prefix, prefixlen, copyfrom);
  }
  
 +/*
 + * N.B. get_pathspec() is deprecated in favor of the "struct pathspec"
 + * based interface - see pathspec_magic above.
 + *
 + * Arguments:
 + *  - prefix - a path relative to the root of the working tree
 + *  - pathspec - a list of paths underneath the prefix path
 + *
 + * Iterates over pathspec, prepending each path with prefix,
 + * and return the resulting list.
 + *
 + * If pathspec is empty, return a singleton list containing prefix.
 + *
 + * If pathspec and prefix are both empty, return an empty list.
 + *
 + * This is typically used by built-in commands such as add.c, in order
 + * to normalize argv arguments provided to the built-in into a list of
 + * paths to process, all relative to the root of the working tree.
 + */
  const char **get_pathspec(const char *prefix, const char **pathspec)
  {
        const char *entry = *pathspec;
@@@ -650,22 -624,32 +650,32 @@@ static dev_t get_device_or_die(const ch
  /*
   * A "string_list_each_func_t" function that canonicalizes an entry
   * from GIT_CEILING_DIRECTORIES using real_path_if_valid(), or
-  * discards it if unusable.
+  * discards it if unusable.  The presence of an empty entry in
+  * GIT_CEILING_DIRECTORIES turns off canonicalization for all
+  * subsequent entries.
   */
  static int canonicalize_ceiling_entry(struct string_list_item *item,
-                                     void *unused)
+                                     void *cb_data)
  {
+       int *empty_entry_found = cb_data;
        char *ceil = item->string;
-       const char *real_path;
  
-       if (!*ceil || !is_absolute_path(ceil))
+       if (!*ceil) {
+               *empty_entry_found = 1;
                return 0;
-       real_path = real_path_if_valid(ceil);
-       if (!real_path)
+       } else if (!is_absolute_path(ceil)) {
                return 0;
-       free(item->string);
-       item->string = xstrdup(real_path);
-       return 1;
+       } else if (*empty_entry_found) {
+               /* Keep entry but do not canonicalize it */
+               return 1;
+       } else {
+               const char *real_path = real_path_if_valid(ceil);
+               if (!real_path)
+                       return 0;
+               free(item->string);
+               item->string = xstrdup(real_path);
+               return 1;
+       }
  }
  
  /*
@@@ -705,9 -689,11 +715,11 @@@ static const char *setup_git_directory_
                return setup_explicit_git_dir(gitdirenv, cwd, len, nongit_ok);
  
        if (env_ceiling_dirs) {
+               int empty_entry_found = 0;
                string_list_split(&ceiling_dirs, env_ceiling_dirs, PATH_SEP, -1);
                filter_string_list(&ceiling_dirs, 0,
-                                  canonicalize_ceiling_entry, NULL);
+                                  canonicalize_ceiling_entry, &empty_entry_found);
                ceil_offset = longest_ancestor_length(cwd, &ceiling_dirs);
                string_list_clear(&ceiling_dirs, 0);
        }