Merge branch 'mh/packed-refs-various'
authorJunio C Hamano <gitster@pobox.com>
Wed, 29 May 2013 21:23:49 +0000 (14:23 -0700)
committerJunio C Hamano <gitster@pobox.com>
Wed, 29 May 2013 21:23:49 +0000 (14:23 -0700)
Update reading and updating packed-refs file, correcting corner case
bugs.

* mh/packed-refs-various: (33 commits)
refs: handle the main ref_cache specially
refs: change do_for_each_*() functions to take ref_cache arguments
pack_one_ref(): do some cheap tests before a more expensive one
pack_one_ref(): use write_packed_entry() to do the writing
pack_one_ref(): use function peel_entry()
refs: inline function do_not_prune()
pack_refs(): change to use do_for_each_entry()
refs: use same lock_file object for both ref-packing functions
pack_one_ref(): rename "path" parameter to "refname"
pack-refs: merge code from pack-refs.{c,h} into refs.{c,h}
pack-refs: rename handle_one_ref() to pack_one_ref()
refs: extract a function write_packed_entry()
repack_without_ref(): write peeled refs in the rewritten file
t3211: demonstrate loss of peeled refs if a packed ref is deleted
refs: change how packed refs are deleted
search_ref_dir(): return an index rather than a pointer
repack_without_ref(): silence errors for dangling packed refs
t3210: test for spurious error messages for dangling packed refs
refs: change the internal reference-iteration API
refs: extract a function peel_entry()
...

1  2 
Makefile
builtin/clone.c
refs.c
refs.h
diff --combined Makefile
index 5f424a7e19fee0df2ef90dd5e8dea0c7f42cf0b1,d2e93779b1ca8fca5372637023c0d947cd333148..77f3f117c4ae8bf24cbd40758874b5c4d03141d5
+++ b/Makefile
@@@ -43,9 -43,6 +43,9 @@@ all:
  # Define EXPATDIR=/foo/bar if your expat header and library files are in
  # /foo/bar/include and /foo/bar/lib directories.
  #
 +# Define EXPAT_NEEDS_XMLPARSE_H if you have an old version of expat (e.g.,
 +# 1.1 or 1.2) that provides xmlparse.h instead of expat.h.
 +#
  # Define NO_GETTEXT if you don't want Git output to be translated.
  # A translated Git requires GNU libintl or another gettext implementation,
  # plus libintl-perl at runtime.
  # Define NO_D_TYPE_IN_DIRENT if your platform defines DT_UNKNOWN but lacks
  # d_type in struct dirent (Cygwin 1.5, fixed in Cygwin 1.7).
  #
 +# Define HAVE_STRINGS_H if you have strings.h and need it for strcasecmp.
 +#
  # Define NO_STRCASESTR if you don't have strcasestr.
  #
  # Define NO_MEMMEM if you don't have memmem.
  #
 +# Define NO_GETPAGESIZE if you don't have getpagesize.
 +#
  # Define NO_STRLCPY if you don't have strlcpy.
  #
  # Define NO_STRTOUMAX if you don't have both strtoimax and strtoumax in the
  #
  # Define NO_MKSTEMPS if you don't have mkstemps in the C library.
  #
 -# Define NO_STRTOK_R if you don't have strtok_r in the C library.
 -#
  # Define NO_FNMATCH if you don't have fnmatch in the C library.
  #
  # Define NO_FNMATCH_CASEFOLD if your fnmatch function doesn't have the
  # FNM_CASEFOLD GNU extension.
  #
 +# Define USE_WILDMATCH if you want to use Git's wildmatch
 +# implementation as fnmatch
 +#
  # Define NO_GECOS_IN_PWENT if you don't have pw_gecos in struct passwd
  # in the C library.
  #
  # Define NO_POLL if you do not have or don't want to use poll().
  # This also implies NO_SYS_POLL_H.
  #
 +# Define NEEDS_SYS_PARAM_H if you need to include sys/param.h to compile,
 +# *PLEASE* REPORT to git@vger.kernel.org if your platform needs this;
 +# we want to know more about the issue.
 +#
  # Define NO_PTHREADS if you do not have or do not want to use Pthreads.
  #
  # Define NO_PREAD if you have a problem with pread() system call (e.g.
  # apostrophes to be ASCII so that cut&pasting examples to the shell
  # will work.
  #
 +# Define PERL_PATH to the path of your Perl binary (usually /usr/bin/perl).
 +#
  # Define NO_PERL_MAKEMAKER if you cannot use Makefiles generated by perl's
  # MakeMaker (e.g. using ActiveState under Cygwin).
  #
  # Define NO_PERL if you do not want Perl scripts or libraries at all.
  #
 +# Define PYTHON_PATH to the path of your Python binary (often /usr/bin/python
 +# but /usr/bin/python2.7 on some platforms).
 +#
  # Define NO_PYTHON if you do not want Python scripts or libraries at all.
  #
  # Define NO_TCLTK if you do not want Tcl/Tk GUI.
@@@ -347,6 -330,19 +347,6 @@@ GIT-VERSION-FILE: FORC
        @$(SHELL_PATH) ./GIT-VERSION-GEN
  -include GIT-VERSION-FILE
  
 -uname_S := $(shell sh -c 'uname -s 2>/dev/null || echo not')
 -uname_M := $(shell sh -c 'uname -m 2>/dev/null || echo not')
 -uname_O := $(shell sh -c 'uname -o 2>/dev/null || echo not')
 -uname_R := $(shell sh -c 'uname -r 2>/dev/null || echo not')
 -uname_P := $(shell sh -c 'uname -p 2>/dev/null || echo not')
 -uname_V := $(shell sh -c 'uname -v 2>/dev/null || echo not')
 -
 -ifdef MSVC
 -      # avoid the MingW and Cygwin configuration sections
 -      uname_S := Windows
 -      uname_O := Windows
 -endif
 -
  # CFLAGS and LDFLAGS are for the users to override from the command line.
  
  CFLAGS = -g -O2 -Wall
@@@ -358,39 -354,33 +358,39 @@@ STRIP ?= stri
  # Among the variables below, these:
  #   gitexecdir
  #   template_dir
 -#   mandir
 -#   infodir
 -#   htmldir
  #   sysconfdir
  # can be specified as a relative path some/where/else;
  # this is interpreted as relative to $(prefix) and "git" at
  # runtime figures out where they are based on the path to the executable.
 +# Additionally, the following will be treated as relative by "git" if they
 +# begin with "$(prefix)/":
 +#   mandir
 +#   infodir
 +#   htmldir
  # This can help installing the suite in a relocatable way.
  
  prefix = $(HOME)
  bindir_relative = bin
  bindir = $(prefix)/$(bindir_relative)
 -mandir = share/man
 -infodir = share/info
 +mandir = $(prefix)/share/man
 +infodir = $(prefix)/share/info
  gitexecdir = libexec/git-core
  mergetoolsdir = $(gitexecdir)/mergetools
  sharedir = $(prefix)/share
  gitwebdir = $(sharedir)/gitweb
  localedir = $(sharedir)/locale
  template_dir = share/git-core/templates
 -htmldir = share/doc/git-doc
 +htmldir = $(prefix)/share/doc/git-doc
  ETC_GITCONFIG = $(sysconfdir)/gitconfig
  ETC_GITATTRIBUTES = $(sysconfdir)/gitattributes
  lib = lib
  # DESTDIR =
  pathsep = :
  
 +mandir_relative = $(patsubst $(prefix)/%,%,$(mandir))
 +infodir_relative = $(patsubst $(prefix)/%,%,$(infodir))
 +htmldir_relative = $(patsubst $(prefix)/%,%,$(htmldir))
 +
  export prefix bindir sharedir sysconfdir gitwebdir localedir
  
  CC = cc
@@@ -460,7 -450,6 +460,7 @@@ SCRIPT_SH += git-mergetool.s
  SCRIPT_SH += git-pull.sh
  SCRIPT_SH += git-quiltimport.sh
  SCRIPT_SH += git-rebase.sh
 +SCRIPT_SH += git-remote-testgit.sh
  SCRIPT_SH += git-repack.sh
  SCRIPT_SH += git-request-pull.sh
  SCRIPT_SH += git-stash.sh
@@@ -485,41 -474,12 +485,41 @@@ SCRIPT_PERL += git-relink.per
  SCRIPT_PERL += git-send-email.perl
  SCRIPT_PERL += git-svn.perl
  
 -SCRIPT_PYTHON += git-remote-testgit.py
 +SCRIPT_PYTHON += git-remote-testpy.py
  SCRIPT_PYTHON += git-p4.py
  
 -SCRIPTS = $(patsubst %.sh,%,$(SCRIPT_SH)) \
 -        $(patsubst %.perl,%,$(SCRIPT_PERL)) \
 -        $(patsubst %.py,%,$(SCRIPT_PYTHON)) \
 +# Generated files for scripts
 +SCRIPT_SH_GEN = $(patsubst %.sh,%,$(SCRIPT_SH))
 +SCRIPT_PERL_GEN = $(patsubst %.perl,%,$(SCRIPT_PERL))
 +SCRIPT_PYTHON_GEN = $(patsubst %.py,%,$(SCRIPT_PYTHON))
 +
 +# Individual rules to allow e.g.
 +# "make -C ../.. SCRIPT_PERL=contrib/foo/bar.perl build-perl-script"
 +# from subdirectories like contrib/*/
 +.PHONY: build-perl-script build-sh-script build-python-script
 +build-perl-script: $(SCRIPT_PERL_GEN)
 +build-sh-script: $(SCRIPT_SH_GEN)
 +build-python-script: $(SCRIPT_PYTHON_GEN)
 +
 +.PHONY: install-perl-script install-sh-script install-python-script
 +install-sh-script: $(SCRIPT_SH_GEN)
 +      $(INSTALL) $(SCRIPT_SH_GEN) '$(DESTDIR_SQ)$(gitexec_instdir_SQ)'
 +install-perl-script: $(SCRIPT_PERL_GEN)
 +      $(INSTALL) $(SCRIPT_PERL_GEN) '$(DESTDIR_SQ)$(gitexec_instdir_SQ)'
 +install-python-script: $(SCRIPT_PYTHON_GEN)
 +      $(INSTALL) $(SCRIPT_PYTHON_GEN) '$(DESTDIR_SQ)$(gitexec_instdir_SQ)'
 +
 +.PHONY: clean-perl-script clean-sh-script clean-python-script
 +clean-sh-script:
 +      $(RM) $(SCRIPT_SH_GEN)
 +clean-perl-script:
 +      $(RM) $(SCRIPT_PERL_GEN)
 +clean-python-script:
 +      $(RM) $(SCRIPT_PYTHON_GEN)
 +
 +SCRIPTS = $(SCRIPT_SH_GEN) \
 +        $(SCRIPT_PERL_GEN) \
 +        $(SCRIPT_PYTHON_GEN) \
          git-instaweb
  
  ETAGS_TARGET = TAGS
@@@ -568,7 -528,6 +568,7 @@@ TEST_PROGRAMS_NEED_X += test-sigchai
  TEST_PROGRAMS_NEED_X += test-string-list
  TEST_PROGRAMS_NEED_X += test-subprocess
  TEST_PROGRAMS_NEED_X += test-svn-fe
 +TEST_PROGRAMS_NEED_X += test-wildmatch
  
  TEST_PROGRAMS = $(patsubst %,%$X,$(TEST_PROGRAMS_NEED_X))
  
@@@ -685,12 -644,10 +685,11 @@@ LIB_H += notes-cache.
  LIB_H += notes-merge.h
  LIB_H += notes.h
  LIB_H += object.h
- LIB_H += pack-refs.h
  LIB_H += pack-revindex.h
  LIB_H += pack.h
  LIB_H += parse-options.h
  LIB_H += patch-ids.h
 +LIB_H += pathspec.h
  LIB_H += pkt-line.h
  LIB_H += progress.h
  LIB_H += prompt.h
@@@ -732,7 -689,6 +731,7 @@@ LIB_H += vcs-svn/sliding_window.
  LIB_H += vcs-svn/svndiff.h
  LIB_H += vcs-svn/svndump.h
  LIB_H += walker.h
 +LIB_H += wildmatch.h
  LIB_H += wt-status.h
  LIB_H += xdiff-interface.h
  LIB_H += xdiff/xdiff.h
@@@ -818,7 -774,6 +817,6 @@@ LIB_OBJS += notes-cache.
  LIB_OBJS += notes-merge.o
  LIB_OBJS += object.o
  LIB_OBJS += pack-check.o
- LIB_OBJS += pack-refs.o
  LIB_OBJS += pack-revindex.o
  LIB_OBJS += pack-write.o
  LIB_OBJS += pager.o
@@@ -827,7 -782,6 +825,7 @@@ LIB_OBJS += parse-options-cb.
  LIB_OBJS += patch-delta.o
  LIB_OBJS += patch-ids.o
  LIB_OBJS += path.o
 +LIB_OBJS += pathspec.o
  LIB_OBJS += pkt-line.o
  LIB_OBJS += preload-index.o
  LIB_OBJS += pretty.o
@@@ -875,7 -829,6 +873,7 @@@ LIB_OBJS += utf8.
  LIB_OBJS += varint.o
  LIB_OBJS += version.o
  LIB_OBJS += walker.o
 +LIB_OBJS += wildmatch.o
  LIB_OBJS += wrapper.o
  LIB_OBJS += write_or_die.o
  LIB_OBJS += ws.o
@@@ -893,7 -846,6 +891,7 @@@ BUILTIN_OBJS += builtin/branch.
  BUILTIN_OBJS += builtin/bundle.o
  BUILTIN_OBJS += builtin/cat-file.o
  BUILTIN_OBJS += builtin/check-attr.o
 +BUILTIN_OBJS += builtin/check-ignore.o
  BUILTIN_OBJS += builtin/check-ref-format.o
  BUILTIN_OBJS += builtin/checkout-index.o
  BUILTIN_OBJS += builtin/checkout.o
@@@ -984,7 -936,518 +982,7 @@@ EXTLIBS 
  
  GIT_USER_AGENT = git/$(GIT_VERSION)
  
 -#
 -# Platform specific tweaks
 -#
 -
 -# We choose to avoid "if .. else if .. else .. endif endif"
 -# because maintaining the nesting to match is a pain.  If
 -# we had "elif" things would have been much nicer...
 -
 -ifeq ($(uname_M),x86_64)
 -      XDL_FAST_HASH = YesPlease
 -endif
 -ifeq ($(uname_S),OSF1)
 -      # Need this for u_short definitions et al
 -      BASIC_CFLAGS += -D_OSF_SOURCE
 -      SOCKLEN_T = int
 -      NO_STRTOULL = YesPlease
 -      NO_NSEC = YesPlease
 -endif
 -ifeq ($(uname_S),Linux)
 -      NO_STRLCPY = YesPlease
 -      NO_MKSTEMPS = YesPlease
 -      HAVE_PATHS_H = YesPlease
 -      LIBC_CONTAINS_LIBINTL = YesPlease
 -      HAVE_DEV_TTY = YesPlease
 -endif
 -ifeq ($(uname_S),GNU/kFreeBSD)
 -      NO_STRLCPY = YesPlease
 -      NO_MKSTEMPS = YesPlease
 -      HAVE_PATHS_H = YesPlease
 -      DIR_HAS_BSD_GROUP_SEMANTICS = YesPlease
 -      LIBC_CONTAINS_LIBINTL = YesPlease
 -endif
 -ifeq ($(uname_S),UnixWare)
 -      CC = cc
 -      NEEDS_SOCKET = YesPlease
 -      NEEDS_NSL = YesPlease
 -      NEEDS_SSL_WITH_CRYPTO = YesPlease
 -      NEEDS_LIBICONV = YesPlease
 -      SHELL_PATH = /usr/local/bin/bash
 -      NO_IPV6 = YesPlease
 -      NO_HSTRERROR = YesPlease
 -      NO_MKSTEMPS = YesPlease
 -      BASIC_CFLAGS += -Kthread
 -      BASIC_CFLAGS += -I/usr/local/include
 -      BASIC_LDFLAGS += -L/usr/local/lib
 -      INSTALL = ginstall
 -      TAR = gtar
 -      NO_STRCASESTR = YesPlease
 -      NO_MEMMEM = YesPlease
 -endif
 -ifeq ($(uname_S),SCO_SV)
 -      ifeq ($(uname_R),3.2)
 -              CFLAGS = -O2
 -      endif
 -      ifeq ($(uname_R),5)
 -              CC = cc
 -              BASIC_CFLAGS += -Kthread
 -      endif
 -      NEEDS_SOCKET = YesPlease
 -      NEEDS_NSL = YesPlease
 -      NEEDS_SSL_WITH_CRYPTO = YesPlease
 -      NEEDS_LIBICONV = YesPlease
 -      SHELL_PATH = /usr/bin/bash
 -      NO_IPV6 = YesPlease
 -      NO_HSTRERROR = YesPlease
 -      NO_MKSTEMPS = YesPlease
 -      BASIC_CFLAGS += -I/usr/local/include
 -      BASIC_LDFLAGS += -L/usr/local/lib
 -      NO_STRCASESTR = YesPlease
 -      NO_MEMMEM = YesPlease
 -      INSTALL = ginstall
 -      TAR = gtar
 -endif
 -ifeq ($(uname_S),Darwin)
 -      NEEDS_CRYPTO_WITH_SSL = YesPlease
 -      NEEDS_SSL_WITH_CRYPTO = YesPlease
 -      NEEDS_LIBICONV = YesPlease
 -      ifeq ($(shell expr "$(uname_R)" : '[15678]\.'),2)
 -              OLD_ICONV = UnfortunatelyYes
 -      endif
 -      ifeq ($(shell expr "$(uname_R)" : '[15]\.'),2)
 -              NO_STRLCPY = YesPlease
 -      endif
 -      NO_MEMMEM = YesPlease
 -      USE_ST_TIMESPEC = YesPlease
 -      HAVE_DEV_TTY = YesPlease
 -      COMPAT_OBJS += compat/precompose_utf8.o
 -      BASIC_CFLAGS += -DPRECOMPOSE_UNICODE
 -endif
 -ifeq ($(uname_S),SunOS)
 -      NEEDS_SOCKET = YesPlease
 -      NEEDS_NSL = YesPlease
 -      SHELL_PATH = /bin/bash
 -      SANE_TOOL_PATH = /usr/xpg6/bin:/usr/xpg4/bin
 -      NO_STRCASESTR = YesPlease
 -      NO_MEMMEM = YesPlease
 -      NO_MKDTEMP = YesPlease
 -      NO_MKSTEMPS = YesPlease
 -      NO_REGEX = YesPlease
 -      NO_FNMATCH_CASEFOLD = YesPlease
 -      NO_MSGFMT_EXTENDED_OPTIONS = YesPlease
 -      HAVE_DEV_TTY = YesPlease
 -      ifeq ($(uname_R),5.6)
 -              SOCKLEN_T = int
 -              NO_HSTRERROR = YesPlease
 -              NO_IPV6 = YesPlease
 -              NO_SOCKADDR_STORAGE = YesPlease
 -              NO_UNSETENV = YesPlease
 -              NO_SETENV = YesPlease
 -              NO_STRLCPY = YesPlease
 -              NO_STRTOUMAX = YesPlease
 -              GIT_TEST_CMP = cmp
 -      endif
 -      ifeq ($(uname_R),5.7)
 -              NEEDS_RESOLV = YesPlease
 -              NO_IPV6 = YesPlease
 -              NO_SOCKADDR_STORAGE = YesPlease
 -              NO_UNSETENV = YesPlease
 -              NO_SETENV = YesPlease
 -              NO_STRLCPY = YesPlease
 -              NO_STRTOUMAX = YesPlease
 -              GIT_TEST_CMP = cmp
 -      endif
 -      ifeq ($(uname_R),5.8)
 -              NO_UNSETENV = YesPlease
 -              NO_SETENV = YesPlease
 -              NO_STRTOUMAX = YesPlease
 -              GIT_TEST_CMP = cmp
 -      endif
 -      ifeq ($(uname_R),5.9)
 -              NO_UNSETENV = YesPlease
 -              NO_SETENV = YesPlease
 -              NO_STRTOUMAX = YesPlease
 -              GIT_TEST_CMP = cmp
 -      endif
 -      INSTALL = /usr/ucb/install
 -      TAR = gtar
 -      BASIC_CFLAGS += -D__EXTENSIONS__ -D__sun__ -DHAVE_ALLOCA_H
 -endif
 -ifeq ($(uname_O),Cygwin)
 -      ifeq ($(shell expr "$(uname_R)" : '1\.[1-6]\.'),4)
 -              NO_D_TYPE_IN_DIRENT = YesPlease
 -              NO_D_INO_IN_DIRENT = YesPlease
 -              NO_STRCASESTR = YesPlease
 -              NO_MEMMEM = YesPlease
 -              NO_MKSTEMPS = YesPlease
 -              NO_SYMLINK_HEAD = YesPlease
 -              NO_IPV6 = YesPlease
 -              OLD_ICONV = UnfortunatelyYes
 -              CYGWIN_V15_WIN32API = YesPlease
 -      endif
 -      NO_THREAD_SAFE_PREAD = YesPlease
 -      NEEDS_LIBICONV = YesPlease
 -      NO_FAST_WORKING_DIRECTORY = UnfortunatelyYes
 -      NO_TRUSTABLE_FILEMODE = UnfortunatelyYes
 -      NO_ST_BLOCKS_IN_STRUCT_STAT = YesPlease
 -      # There are conflicting reports about this.
 -      # On some boxes NO_MMAP is needed, and not so elsewhere.
 -      # Try commenting this out if you suspect MMAP is more efficient
 -      NO_MMAP = YesPlease
 -      X = .exe
 -      COMPAT_OBJS += compat/cygwin.o
 -      UNRELIABLE_FSTAT = UnfortunatelyYes
 -      SPARSE_FLAGS = -isystem /usr/include/w32api -Wno-one-bit-signed-bitfield
 -endif
 -ifeq ($(uname_S),FreeBSD)
 -      NEEDS_LIBICONV = YesPlease
 -      OLD_ICONV = YesPlease
 -      NO_MEMMEM = YesPlease
 -      BASIC_CFLAGS += -I/usr/local/include
 -      BASIC_LDFLAGS += -L/usr/local/lib
 -      DIR_HAS_BSD_GROUP_SEMANTICS = YesPlease
 -      USE_ST_TIMESPEC = YesPlease
 -      ifeq ($(shell expr "$(uname_R)" : '4\.'),2)
 -              PTHREAD_LIBS = -pthread
 -              NO_UINTMAX_T = YesPlease
 -              NO_STRTOUMAX = YesPlease
 -      endif
 -      PYTHON_PATH = /usr/local/bin/python
 -      HAVE_PATHS_H = YesPlease
 -endif
 -ifeq ($(uname_S),OpenBSD)
 -      NO_STRCASESTR = YesPlease
 -      NO_MEMMEM = YesPlease
 -      USE_ST_TIMESPEC = YesPlease
 -      NEEDS_LIBICONV = YesPlease
 -      BASIC_CFLAGS += -I/usr/local/include
 -      BASIC_LDFLAGS += -L/usr/local/lib
 -      HAVE_PATHS_H = YesPlease
 -endif
 -ifeq ($(uname_S),NetBSD)
 -      ifeq ($(shell expr "$(uname_R)" : '[01]\.'),2)
 -              NEEDS_LIBICONV = YesPlease
 -      endif
 -      BASIC_CFLAGS += -I/usr/pkg/include
 -      BASIC_LDFLAGS += -L/usr/pkg/lib $(CC_LD_DYNPATH)/usr/pkg/lib
 -      USE_ST_TIMESPEC = YesPlease
 -      NO_MKSTEMPS = YesPlease
 -      HAVE_PATHS_H = YesPlease
 -endif
 -ifeq ($(uname_S),AIX)
 -      DEFAULT_PAGER = more
 -      NO_STRCASESTR = YesPlease
 -      NO_MEMMEM = YesPlease
 -      NO_MKDTEMP = YesPlease
 -      NO_MKSTEMPS = YesPlease
 -      NO_STRLCPY = YesPlease
 -      NO_NSEC = YesPlease
 -      FREAD_READS_DIRECTORIES = UnfortunatelyYes
 -      INTERNAL_QSORT = UnfortunatelyYes
 -      NEEDS_LIBICONV = YesPlease
 -      BASIC_CFLAGS += -D_LARGE_FILES
 -      ifeq ($(shell expr "$(uname_V)" : '[1234]'),1)
 -              NO_PTHREADS = YesPlease
 -      else
 -              PTHREAD_LIBS = -lpthread
 -      endif
 -      ifeq ($(shell expr "$(uname_V).$(uname_R)" : '5\.1'),3)
 -              INLINE = ''
 -      endif
 -      GIT_TEST_CMP = cmp
 -endif
 -ifeq ($(uname_S),GNU)
 -      # GNU/Hurd
 -      NO_STRLCPY = YesPlease
 -      NO_MKSTEMPS = YesPlease
 -      HAVE_PATHS_H = YesPlease
 -      LIBC_CONTAINS_LIBINTL = YesPlease
 -endif
 -ifeq ($(uname_S),IRIX)
 -      NO_SETENV = YesPlease
 -      NO_UNSETENV = YesPlease
 -      NO_STRCASESTR = YesPlease
 -      NO_MEMMEM = YesPlease
 -      NO_MKSTEMPS = YesPlease
 -      NO_MKDTEMP = YesPlease
 -      # When compiled with the MIPSpro 7.4.4m compiler, and without pthreads
 -      # (i.e. NO_PTHREADS is set), and _with_ MMAP (i.e. NO_MMAP is not set),
 -      # git dies with a segmentation fault when trying to access the first
 -      # entry of a reflog.  The conservative choice is made to always set
 -      # NO_MMAP.  If you suspect that your compiler is not affected by this
 -      # issue, comment out the NO_MMAP statement.
 -      NO_MMAP = YesPlease
 -      NO_REGEX = YesPlease
 -      NO_FNMATCH_CASEFOLD = YesPlease
 -      SNPRINTF_RETURNS_BOGUS = YesPlease
 -      SHELL_PATH = /usr/gnu/bin/bash
 -      NEEDS_LIBGEN = YesPlease
 -endif
 -ifeq ($(uname_S),IRIX64)
 -      NO_SETENV = YesPlease
 -      NO_UNSETENV = YesPlease
 -      NO_STRCASESTR = YesPlease
 -      NO_MEMMEM = YesPlease
 -      NO_MKSTEMPS = YesPlease
 -      NO_MKDTEMP = YesPlease
 -      # When compiled with the MIPSpro 7.4.4m compiler, and without pthreads
 -      # (i.e. NO_PTHREADS is set), and _with_ MMAP (i.e. NO_MMAP is not set),
 -      # git dies with a segmentation fault when trying to access the first
 -      # entry of a reflog.  The conservative choice is made to always set
 -      # NO_MMAP.  If you suspect that your compiler is not affected by this
 -      # issue, comment out the NO_MMAP statement.
 -      NO_MMAP = YesPlease
 -      NO_REGEX = YesPlease
 -      NO_FNMATCH_CASEFOLD = YesPlease
 -      SNPRINTF_RETURNS_BOGUS = YesPlease
 -      SHELL_PATH = /usr/gnu/bin/bash
 -      NEEDS_LIBGEN = YesPlease
 -endif
 -ifeq ($(uname_S),HP-UX)
 -      INLINE = __inline
 -      NO_IPV6 = YesPlease
 -      NO_SETENV = YesPlease
 -      NO_STRCASESTR = YesPlease
 -      NO_MEMMEM = YesPlease
 -      NO_MKSTEMPS = YesPlease
 -      NO_STRLCPY = YesPlease
 -      NO_MKDTEMP = YesPlease
 -      NO_UNSETENV = YesPlease
 -      NO_HSTRERROR = YesPlease
 -      NO_SYS_SELECT_H = YesPlease
 -      NO_FNMATCH_CASEFOLD = YesPlease
 -      SNPRINTF_RETURNS_BOGUS = YesPlease
 -      NO_NSEC = YesPlease
 -      ifeq ($(uname_R),B.11.00)
 -              NO_INET_NTOP = YesPlease
 -              NO_INET_PTON = YesPlease
 -      endif
 -      ifeq ($(uname_R),B.10.20)
 -              # Override HP-UX 11.x setting:
 -              INLINE =
 -              SOCKLEN_T = size_t
 -              NO_PREAD = YesPlease
 -              NO_INET_NTOP = YesPlease
 -              NO_INET_PTON = YesPlease
 -      endif
 -      GIT_TEST_CMP = cmp
 -endif
 -ifeq ($(uname_S),Windows)
 -      GIT_VERSION := $(GIT_VERSION).MSVC
 -      pathsep = ;
 -      NO_PREAD = YesPlease
 -      NEEDS_CRYPTO_WITH_SSL = YesPlease
 -      NO_LIBGEN_H = YesPlease
 -      NO_POLL = YesPlease
 -      NO_SYMLINK_HEAD = YesPlease
 -      NO_IPV6 = YesPlease
 -      NO_UNIX_SOCKETS = YesPlease
 -      NO_SETENV = YesPlease
 -      NO_UNSETENV = YesPlease
 -      NO_STRCASESTR = YesPlease
 -      NO_STRLCPY = YesPlease
 -      NO_STRTOK_R = YesPlease
 -      NO_FNMATCH = YesPlease
 -      NO_MEMMEM = YesPlease
 -      # NEEDS_LIBICONV = YesPlease
 -      NO_ICONV = YesPlease
 -      NO_STRTOUMAX = YesPlease
 -      NO_STRTOULL = YesPlease
 -      NO_MKDTEMP = YesPlease
 -      NO_MKSTEMPS = YesPlease
 -      SNPRINTF_RETURNS_BOGUS = YesPlease
 -      NO_SVN_TESTS = YesPlease
 -      NO_PERL_MAKEMAKER = YesPlease
 -      RUNTIME_PREFIX = YesPlease
 -      NO_ST_BLOCKS_IN_STRUCT_STAT = YesPlease
 -      NO_NSEC = YesPlease
 -      USE_WIN32_MMAP = YesPlease
 -      # USE_NED_ALLOCATOR = YesPlease
 -      UNRELIABLE_FSTAT = UnfortunatelyYes
 -      OBJECT_CREATION_USES_RENAMES = UnfortunatelyNeedsTo
 -      NO_REGEX = YesPlease
 -      NO_CURL = YesPlease
 -      NO_PYTHON = YesPlease
 -      BLK_SHA1 = YesPlease
 -      NO_POSIX_GOODIES = UnfortunatelyYes
 -      NATIVE_CRLF = YesPlease
 -      DEFAULT_HELP_FORMAT = html
 -
 -      CC = compat/vcbuild/scripts/clink.pl
 -      AR = compat/vcbuild/scripts/lib.pl
 -      CFLAGS =
 -      BASIC_CFLAGS = -nologo -I. -I../zlib -Icompat/vcbuild -Icompat/vcbuild/include -DWIN32 -D_CONSOLE -DHAVE_STRING_H -D_CRT_SECURE_NO_WARNINGS -D_CRT_NONSTDC_NO_DEPRECATE
 -      COMPAT_OBJS = compat/msvc.o compat/winansi.o \
 -              compat/win32/pthread.o compat/win32/syslog.o \
 -              compat/win32/dirent.o
 -      COMPAT_CFLAGS = -D__USE_MINGW_ACCESS -DNOGDI -DHAVE_STRING_H -DHAVE_ALLOCA_H -Icompat -Icompat/regex -Icompat/win32 -DSTRIP_EXTENSION=\".exe\"
 -      BASIC_LDFLAGS = -IGNORE:4217 -IGNORE:4049 -NOLOGO -SUBSYSTEM:CONSOLE -NODEFAULTLIB:MSVCRT.lib
 -      EXTLIBS = user32.lib advapi32.lib shell32.lib wininet.lib ws2_32.lib
 -      PTHREAD_LIBS =
 -      lib =
 -ifndef DEBUG
 -      BASIC_CFLAGS += -GL -Os -MT
 -      BASIC_LDFLAGS += -LTCG
 -      AR += -LTCG
 -else
 -      BASIC_CFLAGS += -Zi -MTd
 -endif
 -      X = .exe
 -endif
 -ifeq ($(uname_S),Interix)
 -      NO_INITGROUPS = YesPlease
 -      NO_IPV6 = YesPlease
 -      NO_MEMMEM = YesPlease
 -      NO_MKDTEMP = YesPlease
 -      NO_STRTOUMAX = YesPlease
 -      NO_NSEC = YesPlease
 -      NO_MKSTEMPS = YesPlease
 -      ifeq ($(uname_R),3.5)
 -              NO_INET_NTOP = YesPlease
 -              NO_INET_PTON = YesPlease
 -              NO_SOCKADDR_STORAGE = YesPlease
 -              NO_FNMATCH_CASEFOLD = YesPlease
 -      endif
 -      ifeq ($(uname_R),5.2)
 -              NO_INET_NTOP = YesPlease
 -              NO_INET_PTON = YesPlease
 -              NO_SOCKADDR_STORAGE = YesPlease
 -              NO_FNMATCH_CASEFOLD = YesPlease
 -      endif
 -endif
 -ifeq ($(uname_S),Minix)
 -      NO_IPV6 = YesPlease
 -      NO_ST_BLOCKS_IN_STRUCT_STAT = YesPlease
 -      NO_NSEC = YesPlease
 -      NEEDS_LIBGEN =
 -      NEEDS_CRYPTO_WITH_SSL = YesPlease
 -      NEEDS_IDN_WITH_CURL = YesPlease
 -      NEEDS_SSL_WITH_CURL = YesPlease
 -      NEEDS_RESOLV =
 -      NO_HSTRERROR = YesPlease
 -      NO_MMAP = YesPlease
 -      NO_CURL =
 -      NO_EXPAT =
 -endif
 -ifeq ($(uname_S),NONSTOP_KERNEL)
 -      # Needs some C99 features, "inline" is just one of them.
 -      # INLINE='' would just replace one set of warnings with another and
 -      # still not compile in c89 mode, due to non-const array initializations.
 -      CC = cc -c99
 -      # Disable all optimization, seems to result in bad code, with -O or -O2
 -      # or even -O1 (default), /usr/local/libexec/git-core/git-pack-objects
 -      # abends on "git push". Needs more investigation.
 -      CFLAGS = -g -O0
 -      # We'd want it to be here.
 -      prefix = /usr/local
 -      # Our's are in ${prefix}/bin (perl might also be in /usr/bin/perl).
 -      PERL_PATH = ${prefix}/bin/perl
 -      PYTHON_PATH = ${prefix}/bin/python
 -
 -      # As detected by './configure'.
 -      # Missdetected, hence commented out, see below.
 -      #NO_CURL = YesPlease
 -      # Added manually, see above.
 -      NEEDS_SSL_WITH_CURL = YesPlease
 -      HAVE_LIBCHARSET_H = YesPlease
 -      NEEDS_LIBICONV = YesPlease
 -      NEEDS_LIBINTL_BEFORE_LIBICONV = YesPlease
 -      NO_SYS_SELECT_H = UnfortunatelyYes
 -      NO_D_TYPE_IN_DIRENT = YesPlease
 -      NO_HSTRERROR = YesPlease
 -      NO_STRCASESTR = YesPlease
 -      NO_FNMATCH_CASEFOLD = YesPlease
 -      NO_MEMMEM = YesPlease
 -      NO_STRLCPY = YesPlease
 -      NO_SETENV = YesPlease
 -      NO_UNSETENV = YesPlease
 -      NO_MKDTEMP = YesPlease
 -      NO_MKSTEMPS = YesPlease
 -      # Currently libiconv-1.9.1.
 -      OLD_ICONV = UnfortunatelyYes
 -      NO_REGEX = YesPlease
 -      NO_PTHREADS = UnfortunatelyYes
 -
 -      # Not detected (nor checked for) by './configure'.
 -      # We don't have SA_RESTART on NonStop, unfortunalety.
 -      COMPAT_CFLAGS += -DSA_RESTART=0
 -      # Apparently needed in compat/fnmatch/fnmatch.c.
 -      COMPAT_CFLAGS += -DHAVE_STRING_H=1
 -      NO_ST_BLOCKS_IN_STRUCT_STAT = YesPlease
 -      NO_NSEC = YesPlease
 -      NO_PREAD = YesPlease
 -      NO_MMAP = YesPlease
 -      NO_POLL = YesPlease
 -      NO_INTPTR_T = UnfortunatelyYes
 -      # Bug report 10-120822-4477 submitted to HP NonStop development.
 -      MKDIR_WO_TRAILING_SLASH = YesPlease
 -      # RFE 10-120912-4693 submitted to HP NonStop development.
 -      NO_SETITIMER = UnfortunatelyYes
 -      SANE_TOOL_PATH = /usr/coreutils/bin:/usr/local/bin
 -      SHELL_PATH = /usr/local/bin/bash
 -      # as of H06.25/J06.14, we might better use this
 -      #SHELL_PATH = /usr/coreutils/bin/bash
 -endif
 -ifneq (,$(findstring MINGW,$(uname_S)))
 -      pathsep = ;
 -      NO_PREAD = YesPlease
 -      NEEDS_CRYPTO_WITH_SSL = YesPlease
 -      NO_LIBGEN_H = YesPlease
 -      NO_POLL = YesPlease
 -      NO_SYMLINK_HEAD = YesPlease
 -      NO_UNIX_SOCKETS = YesPlease
 -      NO_SETENV = YesPlease
 -      NO_UNSETENV = YesPlease
 -      NO_STRCASESTR = YesPlease
 -      NO_STRLCPY = YesPlease
 -      NO_STRTOK_R = YesPlease
 -      NO_FNMATCH = YesPlease
 -      NO_MEMMEM = YesPlease
 -      NEEDS_LIBICONV = YesPlease
 -      OLD_ICONV = YesPlease
 -      NO_STRTOUMAX = YesPlease
 -      NO_MKDTEMP = YesPlease
 -      NO_MKSTEMPS = YesPlease
 -      NO_SVN_TESTS = YesPlease
 -      NO_PERL_MAKEMAKER = YesPlease
 -      RUNTIME_PREFIX = YesPlease
 -      NO_ST_BLOCKS_IN_STRUCT_STAT = YesPlease
 -      NO_NSEC = YesPlease
 -      USE_WIN32_MMAP = YesPlease
 -      USE_NED_ALLOCATOR = YesPlease
 -      UNRELIABLE_FSTAT = UnfortunatelyYes
 -      OBJECT_CREATION_USES_RENAMES = UnfortunatelyNeedsTo
 -      NO_REGEX = YesPlease
 -      NO_PYTHON = YesPlease
 -      BLK_SHA1 = YesPlease
 -      ETAGS_TARGET = ETAGS
 -      NO_INET_PTON = YesPlease
 -      NO_INET_NTOP = YesPlease
 -      NO_POSIX_GOODIES = UnfortunatelyYes
 -      COMPAT_CFLAGS += -D__USE_MINGW_ACCESS -DNOGDI -Icompat -Icompat/win32
 -      COMPAT_CFLAGS += -DSTRIP_EXTENSION=\".exe\"
 -      COMPAT_OBJS += compat/mingw.o compat/winansi.o \
 -              compat/win32/pthread.o compat/win32/syslog.o \
 -              compat/win32/dirent.o
 -      EXTLIBS += -lws2_32
 -      PTHREAD_LIBS =
 -      X = .exe
 -      SPARSE_FLAGS = -Wno-one-bit-signed-bitfield
 -ifneq (,$(wildcard ../THIS_IS_MSYSGIT))
 -      htmldir = doc/git/html/
 -      prefix =
 -      INSTALL = /bin/install
 -      EXTLIBS += /mingw/lib/libz.a
 -      NO_R_TO_GCC_LINKER = YesPlease
 -      INTERNAL_QSORT = YesPlease
 -      HAVE_LIBCHARSET_H = YesPlease
 -else
 -      NO_CURL = YesPlease
 -endif
 -endif
 -
 +include config.mak.uname
  -include config.mak.autogen
  -include config.mak
  
@@@ -1123,9 -1586,6 +1121,9 @@@ els
                else
                        EXPAT_LIBEXPAT = -lexpat
                endif
 +              ifdef EXPAT_NEEDS_XMLPARSE_H
 +                      BASIC_CFLAGS += -DEXPAT_NEEDS_XMLPARSE_H
 +              endif
        endif
  endif
  
@@@ -1195,9 -1655,6 +1193,9 @@@ endi
  ifdef NO_D_INO_IN_DIRENT
        BASIC_CFLAGS += -DNO_D_INO_IN_DIRENT
  endif
 +ifdef NO_GECOS_IN_PWENT
 +      BASIC_CFLAGS += -DNO_GECOS_IN_PWENT
 +endif
  ifdef NO_ST_BLOCKS_IN_STRUCT_STAT
        BASIC_CFLAGS += -DNO_ST_BLOCKS_IN_STRUCT_STAT
  endif
@@@ -1251,6 -1708,10 +1249,6 @@@ endi
  ifdef NO_STRTOULL
        COMPAT_CFLAGS += -DNO_STRTOULL
  endif
 -ifdef NO_STRTOK_R
 -      COMPAT_CFLAGS += -DNO_STRTOK_R
 -      COMPAT_OBJS += compat/strtok_r.o
 -endif
  ifdef NO_FNMATCH
        COMPAT_CFLAGS += -Icompat/fnmatch
        COMPAT_CFLAGS += -DNO_FNMATCH
@@@ -1262,9 -1723,6 +1260,9 @@@ ifdef NO_FNMATCH_CASEFOL
        COMPAT_OBJS += compat/fnmatch/fnmatch.o
  endif
  endif
 +ifdef USE_WILDMATCH
 +      COMPAT_CFLAGS += -DUSE_WILDMATCH
 +endif
  ifdef NO_SETENV
        COMPAT_CFLAGS += -DNO_SETENV
        COMPAT_OBJS += compat/setenv.o
@@@ -1290,9 -1748,6 +1288,9 @@@ endi
  ifdef NO_SYS_POLL_H
        BASIC_CFLAGS += -DNO_SYS_POLL_H
  endif
 +ifdef NEEDS_SYS_PARAM_H
 +      BASIC_CFLAGS += -DNEEDS_SYS_PARAM_H
 +endif
  ifdef NO_INTTYPES_H
        BASIC_CFLAGS += -DNO_INTTYPES_H
  endif
@@@ -1404,9 -1859,6 +1402,9 @@@ ifdef NO_MEMME
        COMPAT_CFLAGS += -DNO_MEMMEM
        COMPAT_OBJS += compat/memmem.o
  endif
 +ifdef NO_GETPAGESIZE
 +      COMPAT_CFLAGS += -DNO_GETPAGESIZE
 +endif
  ifdef INTERNAL_QSORT
        COMPAT_CFLAGS += -DINTERNAL_QSORT
        COMPAT_OBJS += compat/qsort.o
@@@ -1432,10 -1884,6 +1430,10 @@@ ifdef HAVE_LIBCHARSET_
        EXTLIBS += $(CHARSET_LIB)
  endif
  
 +ifdef HAVE_STRINGS_H
 +      BASIC_CFLAGS += -DHAVE_STRINGS_H
 +endif
 +
  ifdef HAVE_DEV_TTY
        BASIC_CFLAGS += -DHAVE_DEV_TTY
  endif
@@@ -1546,12 -1994,12 +1544,12 @@@ ETC_GITATTRIBUTES_SQ = $(subst ','\'',$
  DESTDIR_SQ = $(subst ','\'',$(DESTDIR))
  bindir_SQ = $(subst ','\'',$(bindir))
  bindir_relative_SQ = $(subst ','\'',$(bindir_relative))
 -mandir_SQ = $(subst ','\'',$(mandir))
 -infodir_SQ = $(subst ','\'',$(infodir))
 +mandir_relative_SQ = $(subst ','\'',$(mandir_relative))
 +infodir_relative_SQ = $(subst ','\'',$(infodir_relative))
  localedir_SQ = $(subst ','\'',$(localedir))
  gitexecdir_SQ = $(subst ','\'',$(gitexecdir))
  template_dir_SQ = $(subst ','\'',$(template_dir))
 -htmldir_SQ = $(subst ','\'',$(htmldir))
 +htmldir_relative_SQ = $(subst ','\'',$(htmldir_relative))
  prefix_SQ = $(subst ','\'',$(prefix))
  gitwebdir_SQ = $(subst ','\'',$(gitwebdir))
  
@@@ -1683,9 -2131,9 +1681,9 @@@ strip: $(PROGRAMS) git$
  
  git.sp git.s git.o: GIT-PREFIX
  git.sp git.s git.o: EXTRA_CPPFLAGS = \
 -      '-DGIT_HTML_PATH="$(htmldir_SQ)"' \
 -      '-DGIT_MAN_PATH="$(mandir_SQ)"' \
 -      '-DGIT_INFO_PATH="$(infodir_SQ)"'
 +      '-DGIT_HTML_PATH="$(htmldir_relative_SQ)"' \
 +      '-DGIT_MAN_PATH="$(mandir_relative_SQ)"' \
 +      '-DGIT_INFO_PATH="$(infodir_relative_SQ)"'
  
  git$X: git.o GIT-LDFLAGS $(BUILTIN_OBJS) $(GITLIBS)
        $(QUIET_LINK)$(CC) $(ALL_CFLAGS) -o $@ git.o \
@@@ -1695,9 -2143,9 +1693,9 @@@ help.sp help.s help.o: common-cmds.
  
  builtin/help.sp builtin/help.s builtin/help.o: common-cmds.h GIT-PREFIX
  builtin/help.sp builtin/help.s builtin/help.o: EXTRA_CPPFLAGS = \
 -      '-DGIT_HTML_PATH="$(htmldir_SQ)"' \
 -      '-DGIT_MAN_PATH="$(mandir_SQ)"' \
 -      '-DGIT_INFO_PATH="$(infodir_SQ)"'
 +      '-DGIT_HTML_PATH="$(htmldir_relative_SQ)"' \
 +      '-DGIT_MAN_PATH="$(mandir_relative_SQ)"' \
 +      '-DGIT_INFO_PATH="$(infodir_relative_SQ)"'
  
  version.sp version.s version.o: GIT-VERSION-FILE GIT-USER-AGENT
  version.sp version.s version.o: EXTRA_CPPFLAGS = \
@@@ -1735,7 -2183,7 +1733,7 @@@ ende
  GIT-SCRIPT-DEFINES: FORCE
        @FLAGS='$(SCRIPT_DEFINES)'; \
            if test x"$$FLAGS" != x"`cat $@ 2>/dev/null`" ; then \
 -              echo 1>&2 "    * new script parameters"; \
 +              echo >&2 "    * new script parameters"; \
                echo "$$FLAGS" >$@; \
              fi
  
@@@ -2124,7 -2572,7 +2122,7 @@@ TRACK_PREFIX = $(bindir_SQ):$(gitexecdi
  GIT-PREFIX: FORCE
        @FLAGS='$(TRACK_PREFIX)'; \
        if test x"$$FLAGS" != x"`cat GIT-PREFIX 2>/dev/null`" ; then \
 -              echo 1>&2 "    * new prefix flags"; \
 +              echo >&2 "    * new prefix flags"; \
                echo "$$FLAGS" >GIT-PREFIX; \
        fi
  
@@@ -2133,7 -2581,7 +2131,7 @@@ TRACK_CFLAGS = $(CC):$(subst ','\'',$(A
  GIT-CFLAGS: FORCE
        @FLAGS='$(TRACK_CFLAGS)'; \
            if test x"$$FLAGS" != x"`cat GIT-CFLAGS 2>/dev/null`" ; then \
 -              echo 1>&2 "    * new build flags"; \
 +              echo >&2 "    * new build flags"; \
                echo "$$FLAGS" >GIT-CFLAGS; \
              fi
  
@@@ -2142,7 -2590,7 +2140,7 @@@ TRACK_LDFLAGS = $(subst ','\'',$(ALL_LD
  GIT-LDFLAGS: FORCE
        @FLAGS='$(TRACK_LDFLAGS)'; \
            if test x"$$FLAGS" != x"`cat GIT-LDFLAGS 2>/dev/null`" ; then \
 -              echo 1>&2 "    * new link flags"; \
 +              echo >&2 "    * new link flags"; \
                echo "$$FLAGS" >GIT-LDFLAGS; \
              fi
  
@@@ -2184,6 -2632,18 +2182,6 @@@ ifdef GIT_PERF_MAKE_OPT
        @echo GIT_PERF_MAKE_OPTS=\''$(subst ','\'',$(subst ','\'',$(GIT_PERF_MAKE_OPTS)))'\' >>$@
  endif
  
 -### Detect Tck/Tk interpreter path changes
 -ifndef NO_TCLTK
 -TRACK_VARS = $(subst ','\'',-DTCLTK_PATH='$(TCLTK_PATH_SQ)')
 -
 -GIT-GUI-VARS: FORCE
 -      @VARS='$(TRACK_VARS)'; \
 -          if test x"$$VARS" != x"`cat $@ 2>/dev/null`" ; then \
 -              echo 1>&2 "    * new Tcl/Tk interpreter location"; \
 -              echo "$$VARS" >$@; \
 -            fi
 -endif
 -
  ### Detect Python interpreter path changes
  ifndef NO_PYTHON
  TRACK_PYTHON = $(subst ','\'',-DPYTHON_PATH='$(PYTHON_PATH_SQ)')
  GIT-PYTHON-VARS: FORCE
        @VARS='$(TRACK_PYTHON)'; \
            if test x"$$VARS" != x"`cat $@ 2>/dev/null`" ; then \
 -              echo 1>&2 "    * new Python interpreter location"; \
 +              echo >&2 "    * new Python interpreter location"; \
                echo "$$VARS" >$@; \
              fi
  endif
@@@ -2449,7 -2909,8 +2447,7 @@@ clean: profile-clea
                builtin/*.o $(LIB_FILE) $(XDIFF_LIB) $(VCSSVN_LIB)
        $(RM) $(ALL_PROGRAMS) $(SCRIPT_LIB) $(BUILT_INS) git$X
        $(RM) $(TEST_PROGRAMS)
 -      $(RM) -r bin-wrappers
 -      $(RM) -r $(dep_dirs)
 +      $(RM) -r bin-wrappers $(dep_dirs)
        $(RM) -r po/build/
        $(RM) *.spec *.pyc *.pyo */*.pyc */*.pyo common-cmds.h $(ETAGS_TARGET) tags cscope*
        $(RM) -r $(GIT_TARNAME) .doc-tmp-dir
@@@ -2469,7 -2930,7 +2467,7 @@@ ifndef NO_TCLT
        $(MAKE) -C gitk-git clean
        $(MAKE) -C git-gui clean
  endif
 -      $(RM) GIT-VERSION-FILE GIT-CFLAGS GIT-LDFLAGS GIT-GUI-VARS GIT-BUILD-OPTIONS
 +      $(RM) GIT-VERSION-FILE GIT-CFLAGS GIT-LDFLAGS GIT-BUILD-OPTIONS
        $(RM) GIT-USER-AGENT GIT-PREFIX GIT-SCRIPT-DEFINES GIT-PYTHON-VARS
  
  .PHONY: all install profile-clean clean strip
diff --combined builtin/clone.c
index 035ab649504d69a83f87b854582c8003a2dcb905,a0f5bb9f450dc44745ccded86fe5ad3f8cd8758e..5e70696d8470a1a33066d291beb47dfb5164263a
  #include "transport.h"
  #include "strbuf.h"
  #include "dir.h"
- #include "pack-refs.h"
  #include "sigchain.h"
  #include "branch.h"
  #include "remote.h"
  #include "run-command.h"
 +#include "connected.h"
  
  /*
   * Overall FIXMEs:
@@@ -232,26 -230,16 +231,26 @@@ static void strip_trailing_slashes(cha
  static int add_one_reference(struct string_list_item *item, void *cb_data)
  {
        char *ref_git;
 +      const char *repo;
        struct strbuf alternate = STRBUF_INIT;
  
 -      /* Beware: real_path() and mkpath() return static buffer */
 +      /* Beware: read_gitfile(), real_path() and mkpath() return static buffer */
        ref_git = xstrdup(real_path(item->string));
 -      if (is_directory(mkpath("%s/.git/objects", ref_git))) {
 +
 +      repo = read_gitfile(ref_git);
 +      if (!repo)
 +              repo = read_gitfile(mkpath("%s/.git", ref_git));
 +      if (repo) {
 +              free(ref_git);
 +              ref_git = xstrdup(repo);
 +      }
 +
 +      if (!repo && is_directory(mkpath("%s/.git/objects", ref_git))) {
                char *ref_git_git = mkpathdup("%s/.git", ref_git);
                free(ref_git);
                ref_git = ref_git_git;
        } else if (!is_directory(mkpath("%s/objects", ref_git)))
 -              die(_("reference repository '%s' is not a local directory."),
 +              die(_("reference repository '%s' is not a local repository."),
                    item->string);
  
        strbuf_addf(&alternate, "%s/objects", ref_git);
@@@ -387,32 -375,10 +386,32 @@@ static void clone_local(const char *src
  static const char *junk_work_tree;
  static const char *junk_git_dir;
  static pid_t junk_pid;
 +static enum {
 +      JUNK_LEAVE_NONE,
 +      JUNK_LEAVE_REPO,
 +      JUNK_LEAVE_ALL
 +} junk_mode = JUNK_LEAVE_NONE;
 +
 +static const char junk_leave_repo_msg[] =
 +N_("Clone succeeded, but checkout failed.\n"
 +   "You can inspect what was checked out with 'git status'\n"
 +   "and retry the checkout with 'git checkout -f HEAD'\n");
  
  static void remove_junk(void)
  {
        struct strbuf sb = STRBUF_INIT;
 +
 +      switch (junk_mode) {
 +      case JUNK_LEAVE_REPO:
 +              warning("%s", _(junk_leave_repo_msg));
 +              /* fall-through */
 +      case JUNK_LEAVE_ALL:
 +              return;
 +      default:
 +              /* proceed to removal */
 +              break;
 +      }
 +
        if (getpid() != junk_pid)
                return;
        if (junk_git_dir) {
@@@ -518,37 -484,12 +517,37 @@@ static void write_followtags(const stru
        }
  }
  
 +static int iterate_ref_map(void *cb_data, unsigned char sha1[20])
 +{
 +      struct ref **rm = cb_data;
 +      struct ref *ref = *rm;
 +
 +      /*
 +       * Skip anything missing a peer_ref, which we are not
 +       * actually going to write a ref for.
 +       */
 +      while (ref && !ref->peer_ref)
 +              ref = ref->next;
 +      /* Returning -1 notes "end of list" to the caller. */
 +      if (!ref)
 +              return -1;
 +
 +      hashcpy(sha1, ref->old_sha1);
 +      *rm = ref->next;
 +      return 0;
 +}
 +
  static void update_remote_refs(const struct ref *refs,
                               const struct ref *mapped_refs,
                               const struct ref *remote_head_points_at,
                               const char *branch_top,
                               const char *msg)
  {
 +      const struct ref *rm = mapped_refs;
 +
 +      if (check_everything_connected(iterate_ref_map, 0, &rm))
 +              die(_("remote did not send all necessary objects"));
 +
        if (refs) {
                write_remote_refs(mapped_refs);
                if (option_single_branch)
@@@ -637,8 -578,7 +636,8 @@@ static int checkout(void
        tree = parse_tree_indirect(sha1);
        parse_tree(tree);
        init_tree_desc(&t, tree->buffer, tree->size);
 -      unpack_trees(1, &t, &opts);
 +      if (unpack_trees(1, &t, &opts) < 0)
 +              die(_("unable to checkout working tree"));
  
        if (write_cache(fd, active_cache, active_nr) ||
            commit_locked_index(lock_file))
@@@ -826,6 -766,8 +825,6 @@@ int cmd_clone(int argc, const char **ar
        atexit(remove_junk);
        sigchain_push_common(remove_junk_on_signal);
  
 -      setenv(CONFIG_ENVIRONMENT, mkpath("%s/config", git_dir), 1);
 -
        if (safe_create_leading_directories_const(git_dir) < 0)
                die(_("could not create leading directories of '%s'"), git_dir);
  
        init_db(option_template, INIT_DB_QUIET);
        write_config(&option_config);
  
 -      /*
 -       * At this point, the config exists, so we do not need the
 -       * environment variable.  We actually need to unset it, too, to
 -       * re-enable parsing of the global configs.
 -       */
 -      unsetenv(CONFIG_ENVIRONMENT);
 -
        git_config(git_default_config, NULL);
  
        if (option_bare) {
        transport_unlock_pack(transport);
        transport_disconnect(transport);
  
 +      junk_mode = JUNK_LEAVE_REPO;
        err = checkout();
  
        strbuf_release(&reflog_msg);
        strbuf_release(&branch_top);
        strbuf_release(&key);
        strbuf_release(&value);
 -      junk_pid = 0;
 +      junk_mode = JUNK_LEAVE_ALL;
        return err;
  }
diff --combined refs.c
index de2d8eb866062649a0d0f23a77e350bca1a182cd,fa53b39c28508232fe1968209daac713cb9fa029..d17931a8bcd5322ab95f92094a009433f14d250a
--- 1/refs.c
--- 2/refs.c
+++ b/refs.c
@@@ -3,7 -3,6 +3,7 @@@
  #include "object.h"
  #include "tag.h"
  #include "dir.h"
 +#include "string-list.h"
  
  /*
   * Make sure "ref" is something reasonable to have under ".git/refs/";
@@@ -109,7 -108,20 +109,20 @@@ struct ref_entry
   * (ref_entry->flag & REF_DIR) is zero.
   */
  struct ref_value {
+       /*
+        * The name of the object to which this reference resolves
+        * (which may be a tag object).  If REF_ISBROKEN, this is
+        * null.  If REF_ISSYMREF, then this is the name of the object
+        * referred to by the last reference in the symlink chain.
+        */
        unsigned char sha1[20];
+       /*
+        * If REF_KNOWS_PEELED, then this field holds the peeled value
+        * of this reference, or null if the reference is known not to
+        * be peelable.  See the documentation for peel_ref() for an
+        * exact definition of "peelable".
+        */
        unsigned char peeled[20];
  };
  
@@@ -158,7 -170,17 +171,17 @@@ struct ref_dir 
        struct ref_entry **entries;
  };
  
- /* ISSYMREF=0x01, ISPACKED=0x02, and ISBROKEN=0x04 are public interfaces */
+ /*
+  * Bit values for ref_entry::flag.  REF_ISSYMREF=0x01,
+  * REF_ISPACKED=0x02, and REF_ISBROKEN=0x04 are public values; see
+  * refs.h.
+  */
+ /*
+  * The field ref_entry->u.value.peeled of this value entry contains
+  * the correct peeled value for the reference, which might be
+  * null_sha1 if the reference is not a tag or if it is broken.
+  */
  #define REF_KNOWS_PEELED 0x08
  
  /* ref_entry represents a directory of references */
@@@ -334,27 -356,28 +357,26 @@@ struct string_slice 
  
  static int ref_entry_cmp_sslice(const void *key_, const void *ent_)
  {
 -      struct string_slice *key = (struct string_slice *)key_;
 -      struct ref_entry *ent = *(struct ref_entry **)ent_;
 -      int entlen = strlen(ent->name);
 -      int cmplen = key->len < entlen ? key->len : entlen;
 -      int cmp = memcmp(key->str, ent->name, cmplen);
 +      const struct string_slice *key = key_;
 +      const struct ref_entry *ent = *(const struct ref_entry * const *)ent_;
 +      int cmp = strncmp(key->str, ent->name, key->len);
        if (cmp)
                return cmp;
 -      return key->len - entlen;
 +      return '\0' - (unsigned char)ent->name[key->len];
  }
  
  /*
-  * Return the entry with the given refname from the ref_dir
-  * (non-recursively), sorting dir if necessary.  Return NULL if no
-  * such entry is found.  dir must already be complete.
+  * Return the index of the entry with the given refname from the
+  * ref_dir (non-recursively), sorting dir if necessary.  Return -1 if
+  * no such entry is found.  dir must already be complete.
   */
- static struct ref_entry *search_ref_dir(struct ref_dir *dir,
-                                       const char *refname, size_t len)
+ static int search_ref_dir(struct ref_dir *dir, const char *refname, size_t len)
  {
        struct ref_entry **r;
        struct string_slice key;
  
        if (refname == NULL || !dir->nr)
-               return NULL;
+               return -1;
  
        sort_ref_dir(dir);
        key.len = len;
                    ref_entry_cmp_sslice);
  
        if (r == NULL)
-               return NULL;
+               return -1;
  
-       return *r;
+       return r - dir->entries;
  }
  
  /*
@@@ -379,8 -402,9 +401,9 @@@ static struct ref_dir *search_for_subdi
                                         const char *subdirname, size_t len,
                                         int mkdir)
  {
-       struct ref_entry *entry = search_ref_dir(dir, subdirname, len);
-       if (!entry) {
+       int entry_index = search_ref_dir(dir, subdirname, len);
+       struct ref_entry *entry;
+       if (entry_index == -1) {
                if (!mkdir)
                        return NULL;
                /*
                 */
                entry = create_dir_entry(dir->ref_cache, subdirname, len, 0);
                add_entry_to_dir(dir, entry);
+       } else {
+               entry = dir->entries[entry_index];
        }
        return get_ref_dir(entry);
  }
@@@ -429,12 -455,67 +454,67 @@@ static struct ref_dir *find_containing_
   */
  static struct ref_entry *find_ref(struct ref_dir *dir, const char *refname)
  {
+       int entry_index;
        struct ref_entry *entry;
        dir = find_containing_dir(dir, refname, 0);
        if (!dir)
                return NULL;
-       entry = search_ref_dir(dir, refname, strlen(refname));
-       return (entry && !(entry->flag & REF_DIR)) ? entry : NULL;
+       entry_index = search_ref_dir(dir, refname, strlen(refname));
+       if (entry_index == -1)
+               return NULL;
+       entry = dir->entries[entry_index];
+       return (entry->flag & REF_DIR) ? NULL : entry;
+ }
+ /*
+  * Remove the entry with the given name from dir, recursing into
+  * subdirectories as necessary.  If refname is the name of a directory
+  * (i.e., ends with '/'), then remove the directory and its contents.
+  * If the removal was successful, return the number of entries
+  * remaining in the directory entry that contained the deleted entry.
+  * If the name was not found, return -1.  Please note that this
+  * function only deletes the entry from the cache; it does not delete
+  * it from the filesystem or ensure that other cache entries (which
+  * might be symbolic references to the removed entry) are updated.
+  * Nor does it remove any containing dir entries that might be made
+  * empty by the removal.  dir must represent the top-level directory
+  * and must already be complete.
+  */
+ static int remove_entry(struct ref_dir *dir, const char *refname)
+ {
+       int refname_len = strlen(refname);
+       int entry_index;
+       struct ref_entry *entry;
+       int is_dir = refname[refname_len - 1] == '/';
+       if (is_dir) {
+               /*
+                * refname represents a reference directory.  Remove
+                * the trailing slash; otherwise we will get the
+                * directory *representing* refname rather than the
+                * one *containing* it.
+                */
+               char *dirname = xmemdupz(refname, refname_len - 1);
+               dir = find_containing_dir(dir, dirname, 0);
+               free(dirname);
+       } else {
+               dir = find_containing_dir(dir, refname, 0);
+       }
+       if (!dir)
+               return -1;
+       entry_index = search_ref_dir(dir, refname, refname_len);
+       if (entry_index == -1)
+               return -1;
+       entry = dir->entries[entry_index];
+       memmove(&dir->entries[entry_index],
+               &dir->entries[entry_index + 1],
+               (dir->nr - entry_index - 1) * sizeof(*dir->entries)
+               );
+       dir->nr--;
+       if (dir->sorted > entry_index)
+               dir->sorted--;
+       free_ref_entry(entry);
+       return dir->nr;
  }
  
  /*
@@@ -503,27 -584,64 +583,64 @@@ static void sort_ref_dir(struct ref_di
        dir->sorted = dir->nr = i;
  }
  
- #define DO_FOR_EACH_INCLUDE_BROKEN 01
+ /* Include broken references in a do_for_each_ref*() iteration: */
+ #define DO_FOR_EACH_INCLUDE_BROKEN 0x01
+ /*
+  * Return true iff the reference described by entry can be resolved to
+  * an object in the database.  Emit a warning if the referred-to
+  * object does not exist.
+  */
+ static int ref_resolves_to_object(struct ref_entry *entry)
+ {
+       if (entry->flag & REF_ISBROKEN)
+               return 0;
+       if (!has_sha1_file(entry->u.value.sha1)) {
+               error("%s does not point to a valid object!", entry->name);
+               return 0;
+       }
+       return 1;
+ }
  
+ /*
+  * current_ref is a performance hack: when iterating over references
+  * using the for_each_ref*() functions, current_ref is set to the
+  * current reference's entry before calling the callback function.  If
+  * the callback function calls peel_ref(), then peel_ref() first
+  * checks whether the reference to be peeled is the current reference
+  * (it usually is) and if so, returns that reference's peeled version
+  * if it is available.  This avoids a refname lookup in a common case.
+  */
  static struct ref_entry *current_ref;
  
- static int do_one_ref(const char *base, each_ref_fn fn, int trim,
-                     int flags, void *cb_data, struct ref_entry *entry)
+ typedef int each_ref_entry_fn(struct ref_entry *entry, void *cb_data);
+ struct ref_entry_cb {
+       const char *base;
+       int trim;
+       int flags;
+       each_ref_fn *fn;
+       void *cb_data;
+ };
+ /*
+  * Handle one reference in a do_for_each_ref*()-style iteration,
+  * calling an each_ref_fn for each entry.
+  */
+ static int do_one_ref(struct ref_entry *entry, void *cb_data)
  {
+       struct ref_entry_cb *data = cb_data;
        int retval;
-       if (prefixcmp(entry->name, base))
+       if (prefixcmp(entry->name, data->base))
+               return 0;
+       if (!(data->flags & DO_FOR_EACH_INCLUDE_BROKEN) &&
+             !ref_resolves_to_object(entry))
                return 0;
  
-       if (!(flags & DO_FOR_EACH_INCLUDE_BROKEN)) {
-               if (entry->flag & REF_ISBROKEN)
-                       return 0; /* ignore broken refs e.g. dangling symref */
-               if (!has_sha1_file(entry->u.value.sha1)) {
-                       error("%s does not point to a valid object!", entry->name);
-                       return 0;
-               }
-       }
        current_ref = entry;
-       retval = fn(entry->name + trim, entry->u.value.sha1, entry->flag, cb_data);
+       retval = data->fn(entry->name + data->trim, entry->u.value.sha1,
+                         entry->flag, data->cb_data);
        current_ref = NULL;
        return retval;
  }
   * Call fn for each reference in dir that has index in the range
   * offset <= index < dir->nr.  Recurse into subdirectories that are in
   * that index range, sorting them before iterating.  This function
-  * does not sort dir itself; it should be sorted beforehand.
+  * does not sort dir itself; it should be sorted beforehand.  fn is
+  * called for all references, including broken ones.
   */
- static int do_for_each_ref_in_dir(struct ref_dir *dir, int offset,
-                                 const char *base,
-                                 each_ref_fn fn, int trim, int flags, void *cb_data)
+ static int do_for_each_entry_in_dir(struct ref_dir *dir, int offset,
+                                   each_ref_entry_fn fn, void *cb_data)
  {
        int i;
        assert(dir->sorted == dir->nr);
                if (entry->flag & REF_DIR) {
                        struct ref_dir *subdir = get_ref_dir(entry);
                        sort_ref_dir(subdir);
-                       retval = do_for_each_ref_in_dir(subdir, 0,
-                                                       base, fn, trim, flags, cb_data);
+                       retval = do_for_each_entry_in_dir(subdir, 0, fn, cb_data);
                } else {
-                       retval = do_one_ref(base, fn, trim, flags, cb_data, entry);
+                       retval = fn(entry, cb_data);
                }
                if (retval)
                        return retval;
   * by refname.  Recurse into subdirectories.  If a value entry appears
   * in both dir1 and dir2, then only process the version that is in
   * dir2.  The input dirs must already be sorted, but subdirs will be
-  * sorted as needed.
+  * sorted as needed.  fn is called for all references, including
+  * broken ones.
   */
- static int do_for_each_ref_in_dirs(struct ref_dir *dir1,
-                                  struct ref_dir *dir2,
-                                  const char *base, each_ref_fn fn, int trim,
-                                  int flags, void *cb_data)
+ static int do_for_each_entry_in_dirs(struct ref_dir *dir1,
+                                    struct ref_dir *dir2,
+                                    each_ref_entry_fn fn, void *cb_data)
  {
        int retval;
        int i1 = 0, i2 = 0;
                struct ref_entry *e1, *e2;
                int cmp;
                if (i1 == dir1->nr) {
-                       return do_for_each_ref_in_dir(dir2, i2,
-                                                     base, fn, trim, flags, cb_data);
+                       return do_for_each_entry_in_dir(dir2, i2, fn, cb_data);
                }
                if (i2 == dir2->nr) {
-                       return do_for_each_ref_in_dir(dir1, i1,
-                                                     base, fn, trim, flags, cb_data);
+                       return do_for_each_entry_in_dir(dir1, i1, fn, cb_data);
                }
                e1 = dir1->entries[i1];
                e2 = dir2->entries[i2];
                                struct ref_dir *subdir2 = get_ref_dir(e2);
                                sort_ref_dir(subdir1);
                                sort_ref_dir(subdir2);
-                               retval = do_for_each_ref_in_dirs(
-                                               subdir1, subdir2,
-                                               base, fn, trim, flags, cb_data);
+                               retval = do_for_each_entry_in_dirs(
+                                               subdir1, subdir2, fn, cb_data);
                                i1++;
                                i2++;
                        } else if (!(e1->flag & REF_DIR) && !(e2->flag & REF_DIR)) {
                                /* Both are references; ignore the one from dir1. */
-                               retval = do_one_ref(base, fn, trim, flags, cb_data, e2);
+                               retval = fn(e2, cb_data);
                                i1++;
                                i2++;
                        } else {
                        if (e->flag & REF_DIR) {
                                struct ref_dir *subdir = get_ref_dir(e);
                                sort_ref_dir(subdir);
-                               retval = do_for_each_ref_in_dir(
-                                               subdir, 0,
-                                               base, fn, trim, flags, cb_data);
+                               retval = do_for_each_entry_in_dir(
+                                               subdir, 0, fn, cb_data);
                        } else {
-                               retval = do_one_ref(base, fn, trim, flags, cb_data, e);
+                               retval = fn(e, cb_data);
                        }
                }
                if (retval)
                        return retval;
        }
-       if (i1 < dir1->nr)
-               return do_for_each_ref_in_dir(dir1, i1,
-                                             base, fn, trim, flags, cb_data);
-       if (i2 < dir2->nr)
-               return do_for_each_ref_in_dir(dir2, i2,
-                                             base, fn, trim, flags, cb_data);
-       return 0;
  }
  
  /*
@@@ -661,14 -767,13 +766,13 @@@ struct name_conflict_cb 
        const char *conflicting_refname;
  };
  
- static int name_conflict_fn(const char *existingrefname, const unsigned char *sha1,
-                           int flags, void *cb_data)
+ static int name_conflict_fn(struct ref_entry *entry, void *cb_data)
  {
        struct name_conflict_cb *data = (struct name_conflict_cb *)cb_data;
-       if (data->oldrefname && !strcmp(data->oldrefname, existingrefname))
+       if (data->oldrefname && !strcmp(data->oldrefname, entry->name))
                return 0;
-       if (names_conflict(data->refname, existingrefname)) {
-               data->conflicting_refname = existingrefname;
+       if (names_conflict(data->refname, entry->name)) {
+               data->conflicting_refname = entry->name;
                return 1;
        }
        return 0;
  
  /*
   * Return true iff a reference named refname could be created without
-  * conflicting with the name of an existing reference in array.  If
+  * conflicting with the name of an existing reference in dir.  If
   * oldrefname is non-NULL, ignore potential conflicts with oldrefname
   * (e.g., because oldrefname is scheduled for deletion in the same
   * operation).
@@@ -690,9 -795,7 +794,7 @@@ static int is_refname_available(const c
        data.conflicting_refname = NULL;
  
        sort_ref_dir(dir);
-       if (do_for_each_ref_in_dir(dir, 0, "", name_conflict_fn,
-                                  0, DO_FOR_EACH_INCLUDE_BROKEN,
-                                  &data)) {
+       if (do_for_each_entry_in_dir(dir, 0, name_conflict_fn, &data)) {
                error("'%s' exists; cannot create '%s'",
                      data.conflicting_refname, refname);
                return 0;
@@@ -708,9 -811,13 +810,13 @@@ static struct ref_cache 
        struct ref_cache *next;
        struct ref_entry *loose;
        struct ref_entry *packed;
-       /* The submodule name, or "" for the main repo. */
-       char name[FLEX_ARRAY];
- } *ref_cache;
+       /*
+        * The submodule name, or "" for the main repo.  We allocate
+        * length 1 rather than FLEX_ARRAY so that the main ref_cache
+        * is initialized correctly.
+        */
+       char name[1];
+ } ref_cache, *submodule_ref_caches;
  
  static void clear_packed_ref_cache(struct ref_cache *refs)
  {
@@@ -748,18 -855,18 +854,18 @@@ static struct ref_cache *create_ref_cac
   */
  static struct ref_cache *get_ref_cache(const char *submodule)
  {
-       struct ref_cache *refs = ref_cache;
-       if (!submodule)
-               submodule = "";
-       while (refs) {
+       struct ref_cache *refs;
+       if (!submodule || !*submodule)
+               return &ref_cache;
+       for (refs = submodule_ref_caches; refs; refs = refs->next)
                if (!strcmp(submodule, refs->name))
                        return refs;
-               refs = refs->next;
-       }
  
        refs = create_ref_cache(submodule);
-       refs->next = ref_cache;
-       ref_cache = refs;
+       refs->next = submodule_ref_caches;
+       submodule_ref_caches = refs;
        return refs;
  }
  
@@@ -770,6 -877,16 +876,16 @@@ void invalidate_ref_cache(const char *s
        clear_loose_ref_cache(refs);
  }
  
+ /* The length of a peeled reference line in packed-refs, including EOL: */
+ #define PEELED_LINE_LENGTH 42
+ /*
+  * The packed-refs header line that we write out.  Perhaps other
+  * traits will be added later.  The trailing space is required.
+  */
+ static const char PACKED_REFS_HEADER[] =
+       "# pack-refs with: peeled fully-peeled \n";
  /*
   * Parse one line from a packed-refs file.  Write the SHA1 to sha1.
   * Return a pointer to the refname within the line (null-terminated),
@@@ -862,8 -979,8 +978,8 @@@ static void read_packed_refs(FILE *f, s
                }
                if (last &&
                    refline[0] == '^' &&
-                   strlen(refline) == 42 &&
-                   refline[41] == '\n' &&
+                   strlen(refline) == PEELED_LINE_LENGTH &&
+                   refline[PEELED_LINE_LENGTH - 1] == '\n' &&
                    !get_sha1_hex(refline + 1, sha1)) {
                        hashcpy(last->u.value.peeled, sha1);
                        /*
@@@ -898,8 -1015,8 +1014,8 @@@ static struct ref_dir *get_packed_refs(
  
  void add_packed_ref(const char *refname, const unsigned char *sha1)
  {
-       add_ref(get_packed_refs(get_ref_cache(NULL)),
-                       create_ref_entry(refname, sha1, REF_ISPACKED, 1));
+       add_ref(get_packed_refs(&ref_cache),
+               create_ref_entry(refname, sha1, REF_ISPACKED, 1));
  }
  
  /*
@@@ -1069,18 -1186,12 +1185,12 @@@ int resolve_gitlink_ref(const char *pat
  }
  
  /*
-  * Try to read ref from the packed references.  On success, set sha1
-  * and return 0; otherwise, return -1.
+  * Return the ref_entry for the given refname from the packed
+  * references.  If it does not exist, return NULL.
   */
- static int get_packed_ref(const char *refname, unsigned char *sha1)
+ static struct ref_entry *get_packed_ref(const char *refname)
  {
-       struct ref_dir *packed = get_packed_refs(get_ref_cache(NULL));
-       struct ref_entry *entry = find_ref(packed, refname);
-       if (entry) {
-               hashcpy(sha1, entry->u.value.sha1);
-               return 0;
-       }
-       return -1;
+       return find_ref(get_packed_refs(&ref_cache), refname);
  }
  
  const char *resolve_ref_unsafe(const char *refname, unsigned char *sha1, int reading, int *flag)
                git_snpath(path, sizeof(path), "%s", refname);
  
                if (lstat(path, &st) < 0) {
+                       struct ref_entry *entry;
                        if (errno != ENOENT)
                                return NULL;
                        /*
                         * The loose reference file does not exist;
                         * check for a packed reference.
                         */
-                       if (!get_packed_ref(refname, sha1)) {
+                       entry = get_packed_ref(refname);
+                       if (entry) {
+                               hashcpy(sha1, entry->u.value.sha1);
                                if (flag)
                                        *flag |= REF_ISPACKED;
                                return refname;
@@@ -1231,54 -1346,130 +1345,130 @@@ static int filter_refs(const char *refn
        return filter->fn(refname, sha1, flags, filter->cb_data);
  }
  
+ enum peel_status {
+       /* object was peeled successfully: */
+       PEEL_PEELED = 0,
+       /*
+        * object cannot be peeled because the named object (or an
+        * object referred to by a tag in the peel chain), does not
+        * exist.
+        */
+       PEEL_INVALID = -1,
+       /* object cannot be peeled because it is not a tag: */
+       PEEL_NON_TAG = -2,
+       /* ref_entry contains no peeled value because it is a symref: */
+       PEEL_IS_SYMREF = -3,
+       /*
+        * ref_entry cannot be peeled because it is broken (i.e., the
+        * symbolic reference cannot even be resolved to an object
+        * name):
+        */
+       PEEL_BROKEN = -4
+ };
+ /*
+  * Peel the named object; i.e., if the object is a tag, resolve the
+  * tag recursively until a non-tag is found.  If successful, store the
+  * result to sha1 and return PEEL_PEELED.  If the object is not a tag
+  * or is not valid, return PEEL_NON_TAG or PEEL_INVALID, respectively,
+  * and leave sha1 unchanged.
+  */
+ static enum peel_status peel_object(const unsigned char *name, unsigned char *sha1)
+ {
+       struct object *o = lookup_unknown_object(name);
+       if (o->type == OBJ_NONE) {
+               int type = sha1_object_info(name, NULL);
+               if (type < 0)
+                       return PEEL_INVALID;
+               o->type = type;
+       }
+       if (o->type != OBJ_TAG)
+               return PEEL_NON_TAG;
+       o = deref_tag_noverify(o);
+       if (!o)
+               return PEEL_INVALID;
+       hashcpy(sha1, o->sha1);
+       return PEEL_PEELED;
+ }
+ /*
+  * Peel the entry (if possible) and return its new peel_status.  If
+  * repeel is true, re-peel the entry even if there is an old peeled
+  * value that is already stored in it.
+  *
+  * It is OK to call this function with a packed reference entry that
+  * might be stale and might even refer to an object that has since
+  * been garbage-collected.  In such a case, if the entry has
+  * REF_KNOWS_PEELED then leave the status unchanged and return
+  * PEEL_PEELED or PEEL_NON_TAG; otherwise, return PEEL_INVALID.
+  */
+ static enum peel_status peel_entry(struct ref_entry *entry, int repeel)
+ {
+       enum peel_status status;
+       if (entry->flag & REF_KNOWS_PEELED) {
+               if (repeel) {
+                       entry->flag &= ~REF_KNOWS_PEELED;
+                       hashclr(entry->u.value.peeled);
+               } else {
+                       return is_null_sha1(entry->u.value.peeled) ?
+                               PEEL_NON_TAG : PEEL_PEELED;
+               }
+       }
+       if (entry->flag & REF_ISBROKEN)
+               return PEEL_BROKEN;
+       if (entry->flag & REF_ISSYMREF)
+               return PEEL_IS_SYMREF;
+       status = peel_object(entry->u.value.sha1, entry->u.value.peeled);
+       if (status == PEEL_PEELED || status == PEEL_NON_TAG)
+               entry->flag |= REF_KNOWS_PEELED;
+       return status;
+ }
  int peel_ref(const char *refname, unsigned char *sha1)
  {
        int flag;
        unsigned char base[20];
-       struct object *o;
  
        if (current_ref && (current_ref->name == refname
-               || !strcmp(current_ref->name, refname))) {
-               if (current_ref->flag & REF_KNOWS_PEELED) {
-                       if (is_null_sha1(current_ref->u.value.peeled))
-                           return -1;
-                       hashcpy(sha1, current_ref->u.value.peeled);
-                       return 0;
-               }
-               hashcpy(base, current_ref->u.value.sha1);
-               goto fallback;
+                           || !strcmp(current_ref->name, refname))) {
+               if (peel_entry(current_ref, 0))
+                       return -1;
+               hashcpy(sha1, current_ref->u.value.peeled);
+               return 0;
        }
  
        if (read_ref_full(refname, base, 1, &flag))
                return -1;
  
-       if ((flag & REF_ISPACKED)) {
-               struct ref_dir *dir = get_packed_refs(get_ref_cache(NULL));
-               struct ref_entry *r = find_ref(dir, refname);
-               if (r != NULL && r->flag & REF_KNOWS_PEELED) {
+       /*
+        * If the reference is packed, read its ref_entry from the
+        * cache in the hope that we already know its peeled value.
+        * We only try this optimization on packed references because
+        * (a) forcing the filling of the loose reference cache could
+        * be expensive and (b) loose references anyway usually do not
+        * have REF_KNOWS_PEELED.
+        */
+       if (flag & REF_ISPACKED) {
+               struct ref_entry *r = get_packed_ref(refname);
+               if (r) {
+                       if (peel_entry(r, 0))
+                               return -1;
                        hashcpy(sha1, r->u.value.peeled);
                        return 0;
                }
        }
  
- fallback:
-       o = lookup_unknown_object(base);
-       if (o->type == OBJ_NONE) {
-               int type = sha1_object_info(base, NULL);
-               if (type < 0)
-                       return -1;
-               o->type = type;
-       }
-       if (o->type == OBJ_TAG) {
-               o = deref_tag_noverify(o);
-               if (o) {
-                       hashcpy(sha1, o->sha1);
-                       return 0;
-               }
-       }
-       return -1;
+       return peel_object(base, sha1);
  }
  
  struct warn_if_dangling_data {
@@@ -1316,10 -1507,16 +1506,16 @@@ void warn_dangling_symref(FILE *fp, con
        for_each_rawref(warn_if_dangling_symref, &data);
  }
  
- static int do_for_each_ref(const char *submodule, const char *base, each_ref_fn fn,
-                          int trim, int flags, void *cb_data)
+ /*
+  * Call fn for each reference in the specified ref_cache, omitting
+  * references not in the containing_dir of base.  fn is called for all
+  * references, including broken ones.  If fn ever returns a non-zero
+  * value, stop the iteration and return that value; otherwise, return
+  * 0.
+  */
+ static int do_for_each_entry(struct ref_cache *refs, const char *base,
+                            each_ref_entry_fn fn, void *cb_data)
  {
-       struct ref_cache *refs = get_ref_cache(submodule);
        struct ref_dir *packed_dir = get_packed_refs(refs);
        struct ref_dir *loose_dir = get_loose_refs(refs);
        int retval = 0;
        if (packed_dir && loose_dir) {
                sort_ref_dir(packed_dir);
                sort_ref_dir(loose_dir);
-               retval = do_for_each_ref_in_dirs(
-                               packed_dir, loose_dir,
-                               base, fn, trim, flags, cb_data);
+               retval = do_for_each_entry_in_dirs(
+                               packed_dir, loose_dir, fn, cb_data);
        } else if (packed_dir) {
                sort_ref_dir(packed_dir);
-               retval = do_for_each_ref_in_dir(
-                               packed_dir, 0,
-                               base, fn, trim, flags, cb_data);
+               retval = do_for_each_entry_in_dir(
+                               packed_dir, 0, fn, cb_data);
        } else if (loose_dir) {
                sort_ref_dir(loose_dir);
-               retval = do_for_each_ref_in_dir(
-                               loose_dir, 0,
-                               base, fn, trim, flags, cb_data);
+               retval = do_for_each_entry_in_dir(
+                               loose_dir, 0, fn, cb_data);
        }
  
        return retval;
  }
  
+ /*
+  * Call fn for each reference in the specified ref_cache for which the
+  * refname begins with base.  If trim is non-zero, then trim that many
+  * characters off the beginning of each refname before passing the
+  * refname to fn.  flags can be DO_FOR_EACH_INCLUDE_BROKEN to include
+  * broken references in the iteration.  If fn ever returns a non-zero
+  * value, stop the iteration and return that value; otherwise, return
+  * 0.
+  */
+ static int do_for_each_ref(struct ref_cache *refs, const char *base,
+                          each_ref_fn fn, int trim, int flags, void *cb_data)
+ {
+       struct ref_entry_cb data;
+       data.base = base;
+       data.trim = trim;
+       data.flags = flags;
+       data.fn = fn;
+       data.cb_data = cb_data;
+       return do_for_each_entry(refs, base, do_one_ref, &data);
+ }
  static int do_head_ref(const char *submodule, each_ref_fn fn, void *cb_data)
  {
        unsigned char sha1[20];
@@@ -1380,23 -1596,23 +1595,23 @@@ int head_ref_submodule(const char *subm
  
  int for_each_ref(each_ref_fn fn, void *cb_data)
  {
-       return do_for_each_ref(NULL, "", fn, 0, 0, cb_data);
+       return do_for_each_ref(&ref_cache, "", fn, 0, 0, cb_data);
  }
  
  int for_each_ref_submodule(const char *submodule, each_ref_fn fn, void *cb_data)
  {
-       return do_for_each_ref(submodule, "", fn, 0, 0, cb_data);
+       return do_for_each_ref(get_ref_cache(submodule), "", fn, 0, 0, cb_data);
  }
  
  int for_each_ref_in(const char *prefix, each_ref_fn fn, void *cb_data)
  {
-       return do_for_each_ref(NULL, prefix, fn, strlen(prefix), 0, cb_data);
+       return do_for_each_ref(&ref_cache, prefix, fn, strlen(prefix), 0, cb_data);
  }
  
  int for_each_ref_in_submodule(const char *submodule, const char *prefix,
                each_ref_fn fn, void *cb_data)
  {
-       return do_for_each_ref(submodule, prefix, fn, strlen(prefix), 0, cb_data);
+       return do_for_each_ref(get_ref_cache(submodule), prefix, fn, strlen(prefix), 0, cb_data);
  }
  
  int for_each_tag_ref(each_ref_fn fn, void *cb_data)
@@@ -1431,7 -1647,7 +1646,7 @@@ int for_each_remote_ref_submodule(cons
  
  int for_each_replace_ref(each_ref_fn fn, void *cb_data)
  {
-       return do_for_each_ref(NULL, "refs/replace/", fn, 13, 0, cb_data);
+       return do_for_each_ref(&ref_cache, "refs/replace/", fn, 13, 0, cb_data);
  }
  
  int head_ref_namespaced(each_ref_fn fn, void *cb_data)
@@@ -1454,7 -1670,7 +1669,7 @@@ int for_each_namespaced_ref(each_ref_f
        struct strbuf buf = STRBUF_INIT;
        int ret;
        strbuf_addf(&buf, "%srefs/", get_git_namespace());
-       ret = do_for_each_ref(NULL, buf.buf, fn, 0, 0, cb_data);
+       ret = do_for_each_ref(&ref_cache, buf.buf, fn, 0, 0, cb_data);
        strbuf_release(&buf);
        return ret;
  }
@@@ -1496,7 -1712,7 +1711,7 @@@ int for_each_glob_ref(each_ref_fn fn, c
  
  int for_each_rawref(each_ref_fn fn, void *cb_data)
  {
-       return do_for_each_ref(NULL, "", fn, 0,
+       return do_for_each_ref(&ref_cache, "", fn, 0,
                               DO_FOR_EACH_INCLUDE_BROKEN, cb_data);
  }
  
@@@ -1702,7 -1918,7 +1917,7 @@@ static struct ref_lock *lock_ref_sha1_b
         * name is a proper prefix of our refname.
         */
        if (missing &&
-            !is_refname_available(refname, NULL, get_packed_refs(get_ref_cache(NULL)))) {
+            !is_refname_available(refname, NULL, get_packed_refs(&ref_cache))) {
                last_errno = ENOTDIR;
                goto error_return;
        }
@@@ -1754,47 -1970,224 +1969,224 @@@ struct ref_lock *lock_any_ref_for_updat
        return lock_ref_sha1_basic(refname, old_sha1, flags, NULL);
  }
  
- struct repack_without_ref_sb {
-       const char *refname;
-       int fd;
- };
- static int repack_without_ref_fn(const char *refname, const unsigned char *sha1,
-                                int flags, void *cb_data)
+ /*
+  * Write an entry to the packed-refs file for the specified refname.
+  * If peeled is non-NULL, write it as the entry's peeled value.
+  */
+ static void write_packed_entry(int fd, char *refname, unsigned char *sha1,
+                              unsigned char *peeled)
  {
-       struct repack_without_ref_sb *data = cb_data;
        char line[PATH_MAX + 100];
        int len;
  
-       if (!strcmp(data->refname, refname))
-               return 0;
        len = snprintf(line, sizeof(line), "%s %s\n",
                       sha1_to_hex(sha1), refname);
        /* this should not happen but just being defensive */
        if (len > sizeof(line))
                die("too long a refname '%s'", refname);
-       write_or_die(data->fd, line, len);
+       write_or_die(fd, line, len);
+       if (peeled) {
+               if (snprintf(line, sizeof(line), "^%s\n",
+                            sha1_to_hex(peeled)) != PEELED_LINE_LENGTH)
+                       die("internal error");
+               write_or_die(fd, line, PEELED_LINE_LENGTH);
+       }
+ }
+ struct ref_to_prune {
+       struct ref_to_prune *next;
+       unsigned char sha1[20];
+       char name[FLEX_ARRAY];
+ };
+ struct pack_refs_cb_data {
+       unsigned int flags;
+       struct ref_to_prune *ref_to_prune;
+       int fd;
+ };
+ static int pack_one_ref(struct ref_entry *entry, void *cb_data)
+ {
+       struct pack_refs_cb_data *cb = cb_data;
+       enum peel_status peel_status;
+       int is_tag_ref = !prefixcmp(entry->name, "refs/tags/");
+       /* ALWAYS pack refs that were already packed or are tags */
+       if (!(cb->flags & PACK_REFS_ALL) && !is_tag_ref &&
+           !(entry->flag & REF_ISPACKED))
+               return 0;
+       /* Do not pack symbolic or broken refs: */
+       if ((entry->flag & REF_ISSYMREF) || !ref_resolves_to_object(entry))
+               return 0;
+       peel_status = peel_entry(entry, 1);
+       if (peel_status != PEEL_PEELED && peel_status != PEEL_NON_TAG)
+               die("internal error peeling reference %s (%s)",
+                   entry->name, sha1_to_hex(entry->u.value.sha1));
+       write_packed_entry(cb->fd, entry->name, entry->u.value.sha1,
+                          peel_status == PEEL_PEELED ?
+                          entry->u.value.peeled : NULL);
+       /* If the ref was already packed, there is no need to prune it. */
+       if ((cb->flags & PACK_REFS_PRUNE) && !(entry->flag & REF_ISPACKED)) {
+               int namelen = strlen(entry->name) + 1;
+               struct ref_to_prune *n = xcalloc(1, sizeof(*n) + namelen);
+               hashcpy(n->sha1, entry->u.value.sha1);
+               strcpy(n->name, entry->name);
+               n->next = cb->ref_to_prune;
+               cb->ref_to_prune = n;
+       }
        return 0;
  }
  
+ /*
+  * Remove empty parents, but spare refs/ and immediate subdirs.
+  * Note: munges *name.
+  */
+ static void try_remove_empty_parents(char *name)
+ {
+       char *p, *q;
+       int i;
+       p = name;
+       for (i = 0; i < 2; i++) { /* refs/{heads,tags,...}/ */
+               while (*p && *p != '/')
+                       p++;
+               /* tolerate duplicate slashes; see check_refname_format() */
+               while (*p == '/')
+                       p++;
+       }
+       for (q = p; *q; q++)
+               ;
+       while (1) {
+               while (q > p && *q != '/')
+                       q--;
+               while (q > p && *(q-1) == '/')
+                       q--;
+               if (q == p)
+                       break;
+               *q = '\0';
+               if (rmdir(git_path("%s", name)))
+                       break;
+       }
+ }
+ /* make sure nobody touched the ref, and unlink */
+ static void prune_ref(struct ref_to_prune *r)
+ {
+       struct ref_lock *lock = lock_ref_sha1(r->name + 5, r->sha1);
+       if (lock) {
+               unlink_or_warn(git_path("%s", r->name));
+               unlock_ref(lock);
+               try_remove_empty_parents(r->name);
+       }
+ }
+ static void prune_refs(struct ref_to_prune *r)
+ {
+       while (r) {
+               prune_ref(r);
+               r = r->next;
+       }
+ }
  static struct lock_file packlock;
  
- static int repack_without_ref(const char *refname)
+ int pack_refs(unsigned int flags)
+ {
+       struct pack_refs_cb_data cbdata;
+       memset(&cbdata, 0, sizeof(cbdata));
+       cbdata.flags = flags;
+       cbdata.fd = hold_lock_file_for_update(&packlock, git_path("packed-refs"),
+                                             LOCK_DIE_ON_ERROR);
+       write_or_die(cbdata.fd, PACKED_REFS_HEADER, strlen(PACKED_REFS_HEADER));
+       do_for_each_entry(&ref_cache, "", pack_one_ref, &cbdata);
+       if (commit_lock_file(&packlock) < 0)
+               die_errno("unable to overwrite old ref-pack file");
+       prune_refs(cbdata.ref_to_prune);
+       return 0;
+ }
+ static int repack_ref_fn(struct ref_entry *entry, void *cb_data)
  {
-       struct repack_without_ref_sb data;
-       struct ref_cache *refs = get_ref_cache(NULL);
-       struct ref_dir *packed = get_packed_refs(refs);
-       if (find_ref(packed, refname) == NULL)
+       int *fd = cb_data;
+       enum peel_status peel_status;
+       if (entry->flag & REF_ISBROKEN) {
+               /* This shouldn't happen to packed refs. */
+               error("%s is broken!", entry->name);
                return 0;
-       data.refname = refname;
-       data.fd = hold_lock_file_for_update(&packlock, git_path("packed-refs"), 0);
-       if (data.fd < 0) {
+       }
+       if (!has_sha1_file(entry->u.value.sha1)) {
+               unsigned char sha1[20];
+               int flags;
+               if (read_ref_full(entry->name, sha1, 0, &flags))
+                       /* We should at least have found the packed ref. */
+                       die("Internal error");
+               if ((flags & REF_ISSYMREF) || !(flags & REF_ISPACKED))
+                       /*
+                        * This packed reference is overridden by a
+                        * loose reference, so it is OK that its value
+                        * is no longer valid; for example, it might
+                        * refer to an object that has been garbage
+                        * collected.  For this purpose we don't even
+                        * care whether the loose reference itself is
+                        * invalid, broken, symbolic, etc.  Silently
+                        * omit the packed reference from the output.
+                        */
+                       return 0;
+               /*
+                * There is no overriding loose reference, so the fact
+                * that this reference doesn't refer to a valid object
+                * indicates some kind of repository corruption.
+                * Report the problem, then omit the reference from
+                * the output.
+                */
+               error("%s does not point to a valid object!", entry->name);
+               return 0;
+       }
+       peel_status = peel_entry(entry, 0);
+       write_packed_entry(*fd, entry->name, entry->u.value.sha1,
+                          peel_status == PEEL_PEELED ?
+                          entry->u.value.peeled : NULL);
+       return 0;
+ }
+ static int repack_without_ref(const char *refname)
+ {
+       int fd;
+       struct ref_dir *packed;
+       if (!get_packed_ref(refname))
+               return 0; /* refname does not exist in packed refs */
+       fd = hold_lock_file_for_update(&packlock, git_path("packed-refs"), 0);
+       if (fd < 0) {
                unable_to_lock_error(git_path("packed-refs"), errno);
                return error("cannot delete '%s' from packed refs", refname);
        }
-       clear_packed_ref_cache(refs);
-       packed = get_packed_refs(refs);
-       do_for_each_ref_in_dir(packed, 0, "", repack_without_ref_fn, 0, 0, &data);
+       clear_packed_ref_cache(&ref_cache);
+       packed = get_packed_refs(&ref_cache);
+       /* Remove refname from the cache. */
+       if (remove_entry(packed, refname) == -1) {
+               /*
+                * The packed entry disappeared while we were
+                * acquiring the lock.
+                */
+               rollback_lock_file(&packlock);
+               return 0;
+       }
+       write_or_die(fd, PACKED_REFS_HEADER, strlen(PACKED_REFS_HEADER));
+       do_for_each_entry_in_dir(packed, 0, repack_ref_fn, &fd);
        return commit_lock_file(&packlock);
  }
  
@@@ -1823,7 -2216,7 +2215,7 @@@ int delete_ref(const char *refname, con
        ret |= repack_without_ref(lock->ref_name);
  
        unlink_or_warn(git_path("logs/%s", lock->ref_name));
-       invalidate_ref_cache(NULL);
+       clear_loose_ref_cache(&ref_cache);
        unlock_ref(lock);
        return ret;
  }
@@@ -1845,7 -2238,6 +2237,6 @@@ int rename_ref(const char *oldrefname, 
        struct stat loginfo;
        int log = !lstat(git_path("logs/%s", oldrefname), &loginfo);
        const char *symref = NULL;
-       struct ref_cache *refs = get_ref_cache(NULL);
  
        if (log && S_ISLNK(loginfo.st_mode))
                return error("reflog for %s is a symlink", oldrefname);
        if (!symref)
                return error("refname %s not found", oldrefname);
  
-       if (!is_refname_available(newrefname, oldrefname, get_packed_refs(refs)))
+       if (!is_refname_available(newrefname, oldrefname, get_packed_refs(&ref_cache)))
                return 1;
  
-       if (!is_refname_available(newrefname, oldrefname, get_loose_refs(refs)))
+       if (!is_refname_available(newrefname, oldrefname, get_loose_refs(&ref_cache)))
                return 1;
  
        if (log && rename(git_path("logs/%s", oldrefname), git_path(TMP_RENAMED_LOG)))
@@@ -2116,7 -2508,7 +2507,7 @@@ int write_ref_sha1(struct ref_lock *loc
                unlock_ref(lock);
                return -1;
        }
-       clear_loose_ref_cache(get_ref_cache(NULL));
+       clear_loose_ref_cache(&ref_cache);
        if (log_ref_write(lock->ref_name, lock->old_sha1, sha1, logmsg) < 0 ||
            (strcmp(lock->ref_name, lock->orig_ref_name) &&
             log_ref_write(lock->orig_ref_name, lock->old_sha1, sha1, logmsg) < 0)) {
@@@ -2332,117 -2724,59 +2723,117 @@@ int read_ref_at(const char *refname, un
        return 1;
  }
  
 -int for_each_recent_reflog_ent(const char *refname, each_reflog_ent_fn fn, long ofs, void *cb_data)
 +static int show_one_reflog_ent(struct strbuf *sb, each_reflog_ent_fn fn, void *cb_data)
 +{
 +      unsigned char osha1[20], nsha1[20];
 +      char *email_end, *message;
 +      unsigned long timestamp;
 +      int tz;
 +
 +      /* old SP new SP name <email> SP time TAB msg LF */
 +      if (sb->len < 83 || sb->buf[sb->len - 1] != '\n' ||
 +          get_sha1_hex(sb->buf, osha1) || sb->buf[40] != ' ' ||
 +          get_sha1_hex(sb->buf + 41, nsha1) || sb->buf[81] != ' ' ||
 +          !(email_end = strchr(sb->buf + 82, '>')) ||
 +          email_end[1] != ' ' ||
 +          !(timestamp = strtoul(email_end + 2, &message, 10)) ||
 +          !message || message[0] != ' ' ||
 +          (message[1] != '+' && message[1] != '-') ||
 +          !isdigit(message[2]) || !isdigit(message[3]) ||
 +          !isdigit(message[4]) || !isdigit(message[5]))
 +              return 0; /* corrupt? */
 +      email_end[1] = '\0';
 +      tz = strtol(message + 1, NULL, 10);
 +      if (message[6] != '\t')
 +              message += 6;
 +      else
 +              message += 7;
 +      return fn(osha1, nsha1, sb->buf + 82, timestamp, tz, message, cb_data);
 +}
 +
 +static char *find_beginning_of_line(char *bob, char *scan)
 +{
 +      while (bob < scan && *(--scan) != '\n')
 +              ; /* keep scanning backwards */
 +      /*
 +       * Return either beginning of the buffer, or LF at the end of
 +       * the previous line.
 +       */
 +      return scan;
 +}
 +
 +int for_each_reflog_ent_reverse(const char *refname, each_reflog_ent_fn fn, void *cb_data)
  {
 -      const char *logfile;
 -      FILE *logfp;
        struct strbuf sb = STRBUF_INIT;
 -      int ret = 0;
 +      FILE *logfp;
 +      long pos;
 +      int ret = 0, at_tail = 1;
  
 -      logfile = git_path("logs/%s", refname);
 -      logfp = fopen(logfile, "r");
 +      logfp = fopen(git_path("logs/%s", refname), "r");
        if (!logfp)
                return -1;
  
 -      if (ofs) {
 -              struct stat statbuf;
 -              if (fstat(fileno(logfp), &statbuf) ||
 -                  statbuf.st_size < ofs ||
 -                  fseek(logfp, -ofs, SEEK_END) ||
 -                  strbuf_getwholeline(&sb, logfp, '\n')) {
 -                      fclose(logfp);
 -                      strbuf_release(&sb);
 -                      return -1;
 +      /* Jump to the end */
 +      if (fseek(logfp, 0, SEEK_END) < 0)
 +              return error("cannot seek back reflog for %s: %s",
 +                           refname, strerror(errno));
 +      pos = ftell(logfp);
 +      while (!ret && 0 < pos) {
 +              int cnt;
 +              size_t nread;
 +              char buf[BUFSIZ];
 +              char *endp, *scanp;
 +
 +              /* Fill next block from the end */
 +              cnt = (sizeof(buf) < pos) ? sizeof(buf) : pos;
 +              if (fseek(logfp, pos - cnt, SEEK_SET))
 +                      return error("cannot seek back reflog for %s: %s",
 +                                   refname, strerror(errno));
 +              nread = fread(buf, cnt, 1, logfp);
 +              if (nread != 1)
 +                      return error("cannot read %d bytes from reflog for %s: %s",
 +                                   cnt, refname, strerror(errno));
 +              pos -= cnt;
 +
 +              scanp = endp = buf + cnt;
 +              if (at_tail && scanp[-1] == '\n')
 +                      /* Looking at the final LF at the end of the file */
 +                      scanp--;
 +              at_tail = 0;
 +
 +              while (buf < scanp) {
 +                      /*
 +                       * terminating LF of the previous line, or the beginning
 +                       * of the buffer.
 +                       */
 +                      char *bp;
 +
 +                      bp = find_beginning_of_line(buf, scanp);
 +
 +                      if (*bp != '\n') {
 +                              strbuf_splice(&sb, 0, 0, buf, endp - buf);
 +                              if (pos)
 +                                      break; /* need to fill another block */
 +                              scanp = buf - 1; /* leave loop */
 +                      } else {
 +                              /*
 +                               * (bp + 1) thru endp is the beginning of the
 +                               * current line we have in sb
 +                               */
 +                              strbuf_splice(&sb, 0, 0, bp + 1, endp - (bp + 1));
 +                              scanp = bp;
 +                              endp = bp + 1;
 +                      }
 +                      ret = show_one_reflog_ent(&sb, fn, cb_data);
 +                      strbuf_reset(&sb);
 +                      if (ret)
 +                              break;
                }
 -      }
  
 -      while (!strbuf_getwholeline(&sb, logfp, '\n')) {
 -              unsigned char osha1[20], nsha1[20];
 -              char *email_end, *message;
 -              unsigned long timestamp;
 -              int tz;
 -
 -              /* old SP new SP name <email> SP time TAB msg LF */
 -              if (sb.len < 83 || sb.buf[sb.len - 1] != '\n' ||
 -                  get_sha1_hex(sb.buf, osha1) || sb.buf[40] != ' ' ||
 -                  get_sha1_hex(sb.buf + 41, nsha1) || sb.buf[81] != ' ' ||
 -                  !(email_end = strchr(sb.buf + 82, '>')) ||
 -                  email_end[1] != ' ' ||
 -                  !(timestamp = strtoul(email_end + 2, &message, 10)) ||
 -                  !message || message[0] != ' ' ||
 -                  (message[1] != '+' && message[1] != '-') ||
 -                  !isdigit(message[2]) || !isdigit(message[3]) ||
 -                  !isdigit(message[4]) || !isdigit(message[5]))
 -                      continue; /* corrupt? */
 -              email_end[1] = '\0';
 -              tz = strtol(message + 1, NULL, 10);
 -              if (message[6] != '\t')
 -                      message += 6;
 -              else
 -                      message += 7;
 -              ret = fn(osha1, nsha1, sb.buf + 82, timestamp, tz, message,
 -                       cb_data);
 -              if (ret)
 -                      break;
        }
 +      if (!ret && sb.len)
 +              ret = show_one_reflog_ent(&sb, fn, cb_data);
 +
        fclose(logfp);
        strbuf_release(&sb);
        return ret;
  
  int for_each_reflog_ent(const char *refname, each_reflog_ent_fn fn, void *cb_data)
  {
 -      return for_each_recent_reflog_ent(refname, fn, 0, cb_data);
 -}
 +      FILE *logfp;
 +      struct strbuf sb = STRBUF_INIT;
 +      int ret = 0;
  
 +      logfp = fopen(git_path("logs/%s", refname), "r");
 +      if (!logfp)
 +              return -1;
 +
 +      while (!ret && !strbuf_getwholeline(&sb, logfp, '\n'))
 +              ret = show_one_reflog_ent(&sb, fn, cb_data);
 +      fclose(logfp);
 +      strbuf_release(&sb);
 +      return ret;
 +}
  /*
   * Call fn for each reflog in the namespace indicated by name.  name
   * must be empty or end with '/'.  Name will be used as a scratch
@@@ -2663,46 -2986,3 +3054,46 @@@ char *shorten_unambiguous_ref(const cha
        free(short_name);
        return xstrdup(refname);
  }
 +
 +static struct string_list *hide_refs;
 +
 +int parse_hide_refs_config(const char *var, const char *value, const char *section)
 +{
 +      if (!strcmp("transfer.hiderefs", var) ||
 +          /* NEEDSWORK: use parse_config_key() once both are merged */
 +          (!prefixcmp(var, section) && var[strlen(section)] == '.' &&
 +           !strcmp(var + strlen(section), ".hiderefs"))) {
 +              char *ref;
 +              int len;
 +
 +              if (!value)
 +                      return config_error_nonbool(var);
 +              ref = xstrdup(value);
 +              len = strlen(ref);
 +              while (len && ref[len - 1] == '/')
 +                      ref[--len] = '\0';
 +              if (!hide_refs) {
 +                      hide_refs = xcalloc(1, sizeof(*hide_refs));
 +                      hide_refs->strdup_strings = 1;
 +              }
 +              string_list_append(hide_refs, ref);
 +      }
 +      return 0;
 +}
 +
 +int ref_is_hidden(const char *refname)
 +{
 +      struct string_list_item *item;
 +
 +      if (!hide_refs)
 +              return 0;
 +      for_each_string_list_item(item, hide_refs) {
 +              int len;
 +              if (prefixcmp(refname, item->string))
 +                      continue;
 +              len = strlen(item->string);
 +              if (!refname[len] || refname[len] == '/')
 +                      return 1;
 +      }
 +      return 0;
 +}
diff --combined refs.h
index a35eafc4ee15493c1473d71b5c2e83a1f9137c1a,175b7a92d1e7d8938e5bac1ce4db6b8cad34feae..8060ed831308fdde3a39c837f8b0953ea35baaef
--- 1/refs.h
--- 2/refs.h
+++ b/refs.h
@@@ -10,8 -10,21 +10,21 @@@ struct ref_lock 
        int force_write;
  };
  
+ /*
+  * Bit values set in the flags argument passed to each_ref_fn():
+  */
+ /* Reference is a symbolic reference. */
  #define REF_ISSYMREF 0x01
+ /* Reference is a packed reference. */
  #define REF_ISPACKED 0x02
+ /*
+  * Reference cannot be resolved to an object name: dangling symbolic
+  * reference (directly or indirectly), corrupt reference file, or
+  * symbolic reference refers to ill-formatted reference name.
+  */
  #define REF_ISBROKEN 0x04
  
  /*
@@@ -59,8 -72,30 +72,30 @@@ extern void warn_dangling_symref(FILE *
   */
  extern void add_packed_ref(const char *refname, const unsigned char *sha1);
  
+ /*
+  * Flags for controlling behaviour of pack_refs()
+  * PACK_REFS_PRUNE: Prune loose refs after packing
+  * PACK_REFS_ALL:   Pack _all_ refs, not just tags and already packed refs
+  */
+ #define PACK_REFS_PRUNE 0x0001
+ #define PACK_REFS_ALL   0x0002
+ /*
+  * Write a packed-refs file for the current repository.
+  * flags: Combination of the above PACK_REFS_* flags.
+  */
+ int pack_refs(unsigned int flags);
  extern int ref_exists(const char *);
  
+ /*
+  * If refname is a non-symbolic reference that refers to a tag object,
+  * and the tag can be (recursively) dereferenced to a non-tag object,
+  * store the SHA1 of the referred-to object to sha1 and return 0.  If
+  * any of these conditions are not met, return a non-zero value.
+  * Symbolic references are considered unpeelable, even if they
+  * ultimately resolve to a peelable tag.
+  */
  extern int peel_ref(const char *refname, unsigned char *sha1);
  
  /** Locks a "refs/" ref returning the lock on success and NULL on failure. **/
@@@ -103,7 -138,7 +138,7 @@@ extern int read_ref_at(const char *refn
  /* iterate over reflog entries */
  typedef int each_reflog_ent_fn(unsigned char *osha1, unsigned char *nsha1, const char *, unsigned long, int, const char *, void *);
  int for_each_reflog_ent(const char *refname, each_reflog_ent_fn fn, void *cb_data);
 -int for_each_recent_reflog_ent(const char *refname, each_reflog_ent_fn fn, long, void *cb_data);
 +int for_each_reflog_ent_reverse(const char *refname, each_reflog_ent_fn fn, void *cb_data);
  
  /*
   * Calls the specified function for each reflog file until it returns nonzero,
@@@ -147,7 -182,4 +182,7 @@@ int update_ref(const char *action, cons
                const unsigned char *sha1, const unsigned char *oldval,
                int flags, enum action_on_err onerr);
  
 +extern int parse_hide_refs_config(const char *var, const char *value, const char *);
 +extern int ref_is_hidden(const char *);
 +
  #endif /* REFS_H */