]> git.ipfire.org Git - thirdparty/git.git/commitdiff
Merge branch 'ds/omit-trailing-hash-in-index'
authorJunio C Hamano <gitster@pobox.com>
Mon, 16 Jan 2023 20:07:47 +0000 (12:07 -0800)
committerJunio C Hamano <gitster@pobox.com>
Mon, 16 Jan 2023 20:07:47 +0000 (12:07 -0800)
Introduce an optional configuration to allow the trailing hash that
protects the index file from bit flipping.

* ds/omit-trailing-hash-in-index:
  features: feature.manyFiles implies fast index writes
  test-lib-functions: add helper for trailing hash
  read-cache: add index.skipHash config option
  hashfile: allow skipping the hash function

225 files changed:
.github/workflows/check-whitespace.yml
.github/workflows/l10n.yml
.github/workflows/main.yml
.gitignore
Documentation/RelNotes/2.40.0.txt [new file with mode: 0644]
Documentation/config/format.txt
Documentation/config/transfer.txt
Documentation/git-am.txt
Documentation/git-branch.txt
Documentation/git-cat-file.txt
Documentation/git-merge-tree.txt
Documentation/git-status.txt
Documentation/git-var.txt
Documentation/git.txt
Documentation/gitattributes.txt
Documentation/gitprotocol-v2.txt
GIT-VERSION-GEN
Makefile
RelNotes
add-interactive.c
apply.c
blob.c
blob.h
builtin.h
builtin/add.c
builtin/am.c
builtin/bisect.c [moved from builtin/bisect--helper.c with 92% similarity]
builtin/branch.c
builtin/bugreport.c
builtin/bundle.c
builtin/cat-file.c
builtin/checkout.c
builtin/clone.c
builtin/commit.c
builtin/config.c
builtin/diff.c
builtin/fast-export.c
builtin/fsmonitor--daemon.c
builtin/log.c
builtin/ls-files.c
builtin/merge-tree.c
builtin/merge.c
builtin/pack-objects.c
builtin/range-diff.c
builtin/read-tree.c
builtin/rebase.c
builtin/repack.c
builtin/rerere.c
builtin/reset.c
builtin/rev-parse.c
builtin/revert.c
builtin/rm.c
builtin/stash.c
builtin/submodule--helper.c
builtin/unpack-file.c
builtin/var.c
builtin/worktree.c
bundle-uri.c
bundle-uri.h
cache.h
ci/lib.sh
combine-diff.c
commit.c
compat/fsmonitor/fsm-darwin-gcc.h
compat/fsmonitor/fsm-listen-darwin.c
compat/mingw.c
compat/win32/pthread.h
connect.c
connect.h
connected.c
contrib/completion/git-completion.bash
contrib/git-jump/README
contrib/git-jump/git-jump
diff-lib.c
diff-no-index.c
diff.c
diff.h
dir.c
dir.h
entry.c
entry.h
environment.c
fetch-pack.c
git-bisect.sh [deleted file]
git-compat-util.h
git-submodule.sh
git.c
help.c
http-fetch.c
line-range.c
list-objects-filter-options.c
list-objects-filter-options.h
list-objects-filter.c
list-objects.c
ls-refs.c
object-file.c
object-store.h
object.c
pack-bitmap.c
pack-write.c
parse-options.h
range-diff.c
read-cache.c
ref-filter.c
reflog.c
refs.c
refs.h
refs/packed-backend.c
remote.h
revision.c
run-command.c
sequencer.c
serve.c
strbuf.c
strbuf.h
submodule.c
submodule.h
t/helper/test-bundle-uri.c
t/helper/test-fake-ssh.c
t/lib-bundle-uri-protocol.sh [new file with mode: 0644]
t/t0000-basic.sh
t/t0007-git-var.sh
t/t0021-conversion.sh
t/t0068-for-each-repo.sh
t/t0070-fundamental.sh
t/t1001-read-tree-m-2way.sh
t/t1011-read-tree-sparse-checkout.sh
t/t1022-read-tree-partial-clone.sh
t/t1091-sparse-checkout-builtin.sh
t/t1302-repo-version.sh
t/t1404-update-ref-errors.sh
t/t1409-avoid-packing-refs.sh
t/t1413-reflog-detach.sh
t/t1500-rev-parse.sh
t/t1501-work-tree.sh
t/t1509-root-work-tree.sh
t/t2012-checkout-last.sh
t/t2018-checkout-branch.sh
t/t2025-checkout-no-overlay.sh
t/t3009-ls-files-others-nonsubmodule.sh
t/t3010-ls-files-killed-modified.sh
t/t3050-subprojects-fetch.sh
t/t3060-ls-files-with-tree.sh
t/t3204-branch-name-interpretation.sh
t/t3409-rebase-environ.sh
t/t3413-rebase-hook.sh
t/t3428-rebase-signoff.sh
t/t3429-rebase-edit-todo.sh
t/t3433-rebase-across-mode-change.sh
t/t3920-crlf-messages.sh
t/t4014-format-patch.sh
t/t4015-diff-whitespace.sh
t/t4023-diff-rename-typechange.sh
t/t4044-diff-index-unique-abbrev.sh
t/t4045-diff-relative.sh
t/t4046-diff-unmerged.sh
t/t4052-stat-output.sh
t/t4053-diff-no-index.sh
t/t4067-diff-partial-clone.sh
t/t4111-apply-subdir.sh
t/t4135-apply-weird-filenames.sh
t/t4150-am.sh
t/t4203-mailmap.sh
t/t4205-log-pretty-formats.sh
t/t4211-line-log.sh
t/t4213-log-tabexpand.sh
t/t4301-merge-tree-write-tree.sh
t/t5310-pack-bitmaps.sh
t/t5313-pack-bounds-checks.sh
t/t5314-pack-cycle-detection.sh
t/t5317-pack-objects-filter-objects.sh
t/t5544-pack-objects-hook.sh
t/t5554-noop-fetch-negotiator.sh
t/t5601-clone.sh
t/t5610-clone-detached.sh
t/t5611-clone-config.sh
t/t5614-clone-submodules-shallow.sh
t/t5616-partial-clone.sh
t/t5617-clone-submodules-remote.sh
t/t5618-alternate-refs.sh
t/t5701-git-serve.sh
t/t5730-protocol-v2-bundle-uri-file.sh [new file with mode: 0755]
t/t5731-protocol-v2-bundle-uri-git.sh [new file with mode: 0755]
t/t5732-protocol-v2-bundle-uri-http.sh [new file with mode: 0755]
t/t5750-bundle-uri-parse.sh
t/t6020-bundle-misc.sh
t/t6030-bisect-porcelain.sh
t/t6060-merge-index.sh
t/t6300-for-each-ref.sh
t/t6301-for-each-ref-errors.sh
t/t6401-merge-criss-cross.sh
t/t6406-merge-attr.sh
t/t6407-merge-binary.sh
t/t6415-merge-dir-to-symlink.sh
t/t6435-merge-sparse.sh
t/t7103-reset-bare.sh
t/t7412-submodule-absorbgitdirs.sh
t/t7504-commit-msg-hook.sh
t/t7508-status.sh
t/t7517-per-repo-email.sh
t/t7520-ignored-hook-warning.sh
t/t7527-builtin-fsmonitor.sh
t/t7600-merge.sh
t/t7605-merge-resolve.sh
t/t7614-merge-signoff.sh
t/t9003-help-autocorrect.sh
t/t9115-git-svn-dcommit-funky-renames.sh
t/t9119-git-svn-info.sh
t/t9146-git-svn-empty-dirs.sh
t/t9148-git-svn-propset.sh
t/t9160-git-svn-preserve-empty-dirs.sh
t/t9902-completion.sh
t/test-lib-functions.sh
t/test-lib.sh
transport-helper.c
transport-internal.h
transport.c
transport.h
unpack-trees.c
unpack-trees.h
userdiff.c
ws.c
wt-status.c
xdiff/xdiffi.c
xdiff/xemit.c

index ad3466ad16e36cda7e6218811163adfbde74d266..a58e2dc8ad6279d792635c4034ed0d113d489695 100644 (file)
@@ -9,42 +9,83 @@ on:
   pull_request:
     types: [opened, synchronize]
 
+# Avoid unnecessary builds. Unlike the main CI jobs, these are not
+# ci-configurable (but could be).
+concurrency:
+  group: ${{ github.workflow }}-${{ github.ref }}
+  cancel-in-progress: true
+
 jobs:
   check-whitespace:
     runs-on: ubuntu-latest
     steps:
-    - uses: actions/checkout@v2
+    - uses: actions/checkout@v3
       with:
         fetch-depth: 0
 
     - name: git log --check
       id: check_out
       run: |
-        log=
+        baseSha=${{github.event.pull_request.base.sha}}
+        problems=()
         commit=
-        while read dash etc
+        commitText=
+        commitTextmd=
+        goodparent=
+        while read dash sha etc
         do
           case "${dash}" in
           "---")
-            commit="${etc}"
+            if test -z "${commit}"
+            then
+              goodparent=${sha}
+            fi
+            commit="${sha}"
+            commitText="${sha} ${etc}"
+            commitTextmd="[${sha}](https://github.com/${{ github.repository }}/commit/${sha}) ${etc}"
             ;;
           "")
             ;;
           *)
             if test -n "${commit}"
             then
-              log="${log}\n${commit}"
+              problems+=("1) --- ${commitTextmd}")
               echo ""
-              echo "--- ${commit}"
+              echo "--- ${commitText}"
+              commit=
             fi
-            commit=
-            log="${log}\n${dash} ${etc}"
-            echo "${dash} ${etc}"
+            case "${dash}" in
+            *:[1-9]*:) # contains file and line number information
+              dashend=${dash#*:}
+              problems+=("[${dash}](https://github.com/${{ github.repository }}/blob/${{github.event.pull_request.head.ref}}/${dash%%:*}#L${dashend%:}) ${sha} ${etc}")
+              ;;
+            *)
+              problems+=("\`${dash} ${sha} ${etc}\`")
+              ;;
+            esac
+            echo "${dash} ${sha} ${etc}"
             ;;
           esac
-        done <<< $(git log --check --pretty=format:"---% h% s" ${{github.event.pull_request.base.sha}}..)
+        done <<< $(git log --check --pretty=format:"---% h% s" ${baseSha}..)
 
-        if test -n "${log}"
+        if test ${#problems[*]} -gt 0
         then
+          if test -z "${commit}"
+          then
+            goodparent=${baseSha: 0:7}
+          fi
+          echo "🛑 Please review the Summary output for further information."
+          echo "### :x: A whitespace issue was found in one or more of the commits." >$GITHUB_STEP_SUMMARY
+          echo "" >>$GITHUB_STEP_SUMMARY
+          echo "Run these commands to correct the problem:" >>$GITHUB_STEP_SUMMARY
+          echo "1. \`git rebase --whitespace=fix ${goodparent}\`" >>$GITHUB_STEP_SUMMARY
+          echo "1. \`git push --force\`" >>$GITHUB_STEP_SUMMARY
+          echo " " >>$GITHUB_STEP_SUMMARY
+          echo "Errors:" >>$GITHUB_STEP_SUMMARY
+          for i in "${problems[@]}"
+          do
+            echo "${i}" >>$GITHUB_STEP_SUMMARY
+          done
+
           exit 2
         fi
index f7ea0f00a449cfafd5dc006e59267e464f753387..6c3849658aa061b89ced8e0464b3b580519c3a34 100644 (file)
@@ -2,6 +2,12 @@ name: git-l10n
 
 on: [push, pull_request_target]
 
+# Avoid unnecessary builds. Unlike the main CI jobs, these are not
+# ci-configurable (but could be).
+concurrency:
+  group: ${{ github.workflow }}-${{ github.ref }}
+  cancel-in-progress: true
+
 jobs:
   git-po-helper:
     if: >-
index e67847a682cea6da2d85e94cf96ea7f34db28fe2..30492eacddfc7db484e9c73af3ba49960d77f5cb 100644 (file)
@@ -11,6 +11,7 @@ jobs:
     runs-on: ubuntu-latest
     outputs:
       enabled: ${{ steps.check-ref.outputs.enabled }}${{ steps.skip-if-redundant.outputs.enabled }}
+      skip_concurrent: ${{ steps.check-ref.outputs.skip_concurrent }}
     steps:
       - name: try to clone ci-config branch
         run: |
@@ -34,7 +35,15 @@ jobs:
           then
             enabled=no
           fi
+
+          skip_concurrent=yes
+          if test -x config-repo/ci/config/skip-concurrent &&
+             ! config-repo/ci/config/skip-concurrent '${{ github.ref }}'
+          then
+            skip_concurrent=no
+          fi
           echo "enabled=$enabled" >>$GITHUB_OUTPUT
+          echo "skip_concurrent=$skip_concurrent" >>$GITHUB_OUTPUT
       - name: skip if the commit or tree was already tested
         id: skip-if-redundant
         uses: actions/github-script@v6
@@ -82,6 +91,9 @@ jobs:
     needs: ci-config
     if: needs.ci-config.outputs.enabled == 'yes'
     runs-on: windows-latest
+    concurrency:
+      group: windows-build-${{ github.ref }}
+      cancel-in-progress: ${{ needs.ci-config.outputs.skip_concurrent == 'yes' }}
     steps:
     - uses: actions/checkout@v3
     - uses: git-for-windows/setup-git-for-windows-sdk@v1
@@ -101,11 +113,14 @@ jobs:
   windows-test:
     name: win test
     runs-on: windows-latest
-    needs: [windows-build]
+    needs: [ci-config, windows-build]
     strategy:
       fail-fast: false
       matrix:
         nr: [0, 1, 2, 3, 4, 5, 6, 7, 8, 9]
+    concurrency:
+      group: windows-test-${{ matrix.nr }}-${{ github.ref }}
+      cancel-in-progress: ${{ needs.ci-config.outputs.skip_concurrent == 'yes' }}
     steps:
     - name: download tracked files and build artifacts
       uses: actions/download-artifact@v3
@@ -132,11 +147,14 @@ jobs:
   vs-build:
     name: win+VS build
     needs: ci-config
-    if: needs.ci-config.outputs.enabled == 'yes'
+    if: github.event.repository.owner.login == 'git-for-windows' && needs.ci-config.outputs.enabled == 'yes'
     env:
       NO_PERL: 1
       GIT_CONFIG_PARAMETERS: "'user.name=CI' 'user.email=ci@git'"
     runs-on: windows-latest
+    concurrency:
+      group: vs-build-${{ github.ref }}
+      cancel-in-progress: ${{ needs.ci-config.outputs.skip_concurrent == 'yes' }}
     steps:
     - uses: actions/checkout@v3
     - uses: git-for-windows/setup-git-for-windows-sdk@v1
@@ -184,11 +202,14 @@ jobs:
   vs-test:
     name: win+VS test
     runs-on: windows-latest
-    needs: vs-build
+    needs: [ci-config, vs-build]
     strategy:
       fail-fast: false
       matrix:
         nr: [0, 1, 2, 3, 4, 5, 6, 7, 8, 9]
+    concurrency:
+      group: vs-test-${{ matrix.nr }}-${{ github.ref }}
+      cancel-in-progress: ${{ needs.ci-config.outputs.skip_concurrent == 'yes' }}
     steps:
     - uses: git-for-windows/setup-git-for-windows-sdk@v1
     - name: download tracked files and build artifacts
@@ -218,6 +239,9 @@ jobs:
     name: ${{matrix.vector.jobname}} (${{matrix.vector.pool}})
     needs: ci-config
     if: needs.ci-config.outputs.enabled == 'yes'
+    concurrency:
+      group: ${{ matrix.vector.jobname }}-${{ matrix.vector.pool }}-${{ github.ref }}
+      cancel-in-progress: ${{ needs.ci-config.outputs.skip_concurrent == 'yes' }}
     strategy:
       fail-fast: false
       matrix:
@@ -265,8 +289,9 @@ jobs:
     - uses: actions/checkout@v3
     - run: ci/install-dependencies.sh
     - run: ci/run-build-and-tests.sh
-    - run: ci/print-test-failures.sh
+    - name: print test failures
       if: failure() && env.FAILED_TEST_ARTIFACTS != ''
+      run: ci/print-test-failures.sh
     - name: Upload failed tests' directories
       if: failure() && env.FAILED_TEST_ARTIFACTS != ''
       uses: actions/upload-artifact@v3
@@ -277,6 +302,9 @@ jobs:
     name: ${{matrix.vector.jobname}} (${{matrix.vector.image}})
     needs: ci-config
     if: needs.ci-config.outputs.enabled == 'yes'
+    concurrency:
+      group: dockerized-${{ matrix.vector.jobname }}-${{ matrix.vector.image }}-${{ github.ref }}
+      cancel-in-progress: ${{ needs.ci-config.outputs.skip_concurrent == 'yes' }}
     strategy:
       fail-fast: false
       matrix:
@@ -298,8 +326,9 @@ jobs:
       if: matrix.vector.jobname == 'linux32'
     - run: ci/install-docker-dependencies.sh
     - run: ci/run-build-and-tests.sh
-    - run: ci/print-test-failures.sh
+    - name: print test failures
       if: failure() && env.FAILED_TEST_ARTIFACTS != ''
+      run: ci/print-test-failures.sh
     - name: Upload failed tests' directories
       if: failure() && env.FAILED_TEST_ARTIFACTS != '' && matrix.vector.jobname != 'linux32'
       uses: actions/upload-artifact@v3
@@ -318,6 +347,9 @@ jobs:
     env:
       jobname: StaticAnalysis
     runs-on: ubuntu-22.04
+    concurrency:
+      group: static-analysis-${{ github.ref }}
+      cancel-in-progress: ${{ needs.ci-config.outputs.skip_concurrent == 'yes' }}
     steps:
     - uses: actions/checkout@v3
     - run: ci/install-dependencies.sh
@@ -329,6 +361,9 @@ jobs:
     env:
       jobname: sparse
     runs-on: ubuntu-20.04
+    concurrency:
+      group: sparse-${{ github.ref }}
+      cancel-in-progress: ${{ needs.ci-config.outputs.skip_concurrent == 'yes' }}
     steps:
     - name: Download a current `sparse` package
       # Ubuntu's `sparse` version is too old for us
@@ -347,6 +382,9 @@ jobs:
     name: documentation
     needs: ci-config
     if: needs.ci-config.outputs.enabled == 'yes'
+    concurrency:
+      group: documentation-${{ github.ref }}
+      cancel-in-progress: ${{ needs.ci-config.outputs.skip_concurrent == 'yes' }}
     env:
       jobname: Documentation
     runs-on: ubuntu-latest
index 0832f1da77b0a1903d2db65689864c627899c5e5..e942a83b45b1bf79b25c60df571e06c75f8924c6 100644 (file)
@@ -21,7 +21,6 @@
 /git-archimport
 /git-archive
 /git-bisect
-/git-bisect--helper
 /git-blame
 /git-branch
 /git-bugreport
diff --git a/Documentation/RelNotes/2.40.0.txt b/Documentation/RelNotes/2.40.0.txt
new file mode 100644 (file)
index 0000000..d7685b8
--- /dev/null
@@ -0,0 +1,143 @@
+Git v2.40 Release Notes
+=======================
+
+UI, Workflows & Features
+
+ * "merge-tree" learns a new `--merge-base` option.
+
+ * "git jump" (in contrib/) learned to present the "quickfix list" to
+   its standard output (instead of letting it consumed by the editor
+   it invokes), and learned to also drive emacs/emacsclient.
+
+ * "git var UNKNOWN_VARIABLE" and "git var VARIABLE" with the variable
+   given an empty value used to behave identically.  Now the latter
+   just gives an empty output, while the former still gives an error
+   message.
+
+ * Introduce a case insensitive mode to the Bash completion helpers.
+
+ * The advice message given by "git status" when it takes long time to
+   enumerate untracked paths has been updated.
+
+ * Just like "git var GIT_EDITOR" abstracts the complex logic to
+   choose which editor gets used behind it, "git var" now give support
+   to GIT_SEQUENCE_EDITOR.
+
+ * "git format-patch" learned to honor format.mboxrd even when sending
+   patches to the standard output stream,
+
+ * 'cat-file' gains mailmap support for its '--batch-check' and '-s'
+   options.
+
+
+Performance, Internal Implementation, Development Support etc.
+
+ * `git bisect` becomes a builtin.
+
+ * The pack-bitmap machinery is taught to log the paths of redundant
+   bitmap(s) to trace2 instead of stderr.
+
+ * Use the SHA1DC implementation on macOS, just like other platforms,
+   by default.
+
+ * Even in a repository with promisor remote, it is useless to
+   attempt to lazily attempt fetching an object that is expected to be
+   commit, because no "filter" mode omits commit objects.  Take
+   advantage of this assumption to fail fast on errors.
+
+ * Stop using "git --super-prefix" and narrow the scope of its use to
+   the submodule--helper.
+
+ * Stop running win+VS build by default.
+   (merge a0da6deeec js/ci-disable-cmake-by-default later to maint).
+
+ * CI updates.  We probably want a clean-up to move the long shell
+   script embedded in yaml file into a separate file, but that can
+   come later.
+   (merge 4542582e59 cw/ci-whitespace later to maint).
+
+ * Use `git diff --no-index` as a test_cmp on Windows.
+
+   We'd probably need to revisit "do we really want to, and have to,
+   lose CRLF vs LF?" later, at which time we may be able to further
+   clean this up by replacing "git diff --no-index" with "diff -u".
+
+
+Fixes since v2.39
+-----------------
+
+ * Various leak fixes.
+   (merge ac95f5d36a ab/various-leak-fixes later to maint).
+
+ * Fix a bug where `pack-objects` would not respect multiple `--filter`
+   arguments when invoked directly.
+   (merge d4f7036887 rs/multi-filter-args later to maint).
+
+ * Make fsmonitor more robust to avoid the flakiness seen in t7527.
+   (merge 6692d45477 jh/t7527-unflake-by-forcing-cookie later to maint).
+
+ * Stop using deprecated macOS API in fsmonitor.
+   (merge b0226007f0 jh/fsmonitor-darwin-modernize later to maint).
+
+ * Redefining system functions for a few functions did not follow our
+   usual "implement git_foo() and #define foo(args) git_foo(args)"
+   pattern, which has broken build for some folks.
+   (merge e1a95b78d8 jk/avoid-redef-system-functions-2.30 later to maint).
+   (merge 395bec6b39 jk/avoid-redef-system-functions later to maint).
+
+ * The way the diff machinery prepares the options array for the
+   parse_options API has been refactored to avoid resource leaks.
+   (merge 189e97bc4b rs/diff-parseopts later to maint).
+
+ * Correct pthread API usage.
+   (merge 786e67611d sx/pthread-error-check-fix later to maint).
+
+ * The code to auto-correct a misspelt subcommand unnecessarily called
+   into git_default_config() from the early config codepath, which was
+   a no-no.  This has bee corrected.
+   (merge 0918d08887 sg/help-autocorrect-config-fix later to maint).
+
+ * "git http-fetch" (which is rarely used) forgot to identify itself
+   in the trace2 output.
+   (merge 7abb43cbc8 jt/http-fetch-trace2-report-name later to maint).
+
+ * The output from "git diff --stat" on an unmerged path lost the
+   terminating LF in Git 2.39, which has been corrected.
+   (merge 209d9cb011 pg/diff-stat-unmerged-regression-fix later to maint).
+
+ * "git pull -v --recurse-submodules" attempted to pass "-v" down to
+   underlying "git submodule update", which did not understand the
+   request and barfed, which has been corrected.
+   (merge 6f65f84766 ss/pull-v-recurse-fix later to maint).
+
+ * When given a pattern that matches an empty string at the end of a
+   line, the code to parse the "git diff" line-ranges fell into an
+   infinite loop, which has been corrected.
+   (merge 4e57c88e02 lk/line-range-parsing-fix later to maint).
+
+ * Fix the sequence to fsync $GIT_DIR/packed-refs file that forgot to
+   flush its output to the disk..
+   (merge ce54672f9b ps/fsync-refs-fix later to maint).
+
+ * Fix to a small regression in 2.38 days.
+   (merge 6d5e9e53aa ab/bundle-wo-args later to maint).
+
+ * Other code cleanup, docfix, build fix, etc.
+   (merge 77e04b2ed4 rs/t4205-do-not-exit-in-test-script later to maint).
+   (merge faebba436e rs/plug-pattern-list-leak-in-lof later to maint).
+   (merge 243caa8982 ab/t5314-avoid-losing-exit-status later to maint).
+   (merge 4d81ce1b99 ab/t7600-avoid-losing-exit-status-of-git later to maint).
+   (merge 5f3bfdc4f3 ab/t4023-avoid-losing-exit-status-of-diff later to maint).
+   (merge 500317ae03 js/t3920-shell-and-or-fix later to maint).
+   (merge 86325d36e6 rs/t3920-crlf-eating-grep-fix later to maint).
+   (merge cfbd173ccb rj/branch-copy-and-rename later to maint).
+   (merge c25d9e529d jk/unused-post-2.39 later to maint).
+   (merge a31cfe3283 jk/server-supports-v2-cleanup later to maint).
+   (merge a658e881c1 rs/am-parse-options-cleanup later to maint).
+   (merge 4cb39fcf19 rs/clear-commit-marks-cleanup later to maint).
+   (merge b07a819c05 rs/reflog-expiry-cleanup later to maint).
+   (merge d422d06167 rs/clarify-error-in-write-loose-object later to maint).
+   (merge 92cb135855 sk/remove-duplicate-includes later to maint).
+   (merge 4eb1ccecd4 dh/mingw-ownership-check-typofix later to maint).
+   (merge f95526419b ar/typofix-gitattributes-doc later to maint).
+   (merge 27875aeec9 km/doc-branch-start-point later to maint).
index c7303d8d9f004b5e413c4e5fe0dbb2f0df0c9171..3bd78269e29a544fa13369f9e5169b403be294ac 100644 (file)
@@ -139,3 +139,7 @@ For example,
 ------------
 +
 will only show notes from `refs/notes/bar`.
+
+format.mboxrd::
+       A boolean value which enables the robust "mboxrd" format when
+       `--stdout` is in use to escape "^>+From " lines.
index 264812cca4db9a0a7b70be3eed7703d3439c9f46..c3ac767d1e4de152865ec82de6a995a9b4934834 100644 (file)
@@ -115,3 +115,9 @@ transfer.unpackLimit::
 transfer.advertiseSID::
        Boolean. When true, client and server processes will advertise their
        unique session IDs to their remote counterpart. Defaults to false.
+
+transfer.bundleURI::
+       When `true`, local `git clone` commands will request bundle
+       information from the remote server (if advertised) and download
+       bundles before continuing the clone through the Git protocol.
+       Defaults to `false`.
index 326276e51ce5734675f842771d41290057c53f90..0c1dfb3c98b440f87e561a0d44c240a41d80dc14 100644 (file)
@@ -9,7 +9,7 @@ git-am - Apply a series of patches from a mailbox
 SYNOPSIS
 --------
 [verse]
-'git am' [--signoff] [--keep] [--[no-]keep-cr] [--[no-]utf8]
+'git am' [--signoff] [--keep] [--[no-]keep-cr] [--[no-]utf8] [--no-verify]
         [--[no-]3way] [--interactive] [--committer-date-is-author-date]
         [--ignore-date] [--ignore-space-change | --ignore-whitespace]
         [--whitespace=<option>] [-C<n>] [-p<n>] [--directory=<dir>]
@@ -138,6 +138,12 @@ include::rerere-options.txt[]
 --interactive::
        Run interactively.
 
+-n::
+--no-verify::
+       By default, the pre-applypatch and applypatch-msg hooks are run.
+       When any of `--no-verify` or `-n` is given, these are bypassed.
+       See also linkgit:githooks[5].
+
 --committer-date-is-author-date::
        By default the command records the date from the e-mail
        message as the commit author date, and uses the time of
index 12c5f84e3bef5c83c80edec9850315dd334beffa..aa2f78c4c2a0af76133636bc25b943457548d6ea 100644 (file)
@@ -116,7 +116,7 @@ OPTIONS
 
 -f::
 --force::
-       Reset <branchname> to <startpoint>, even if <branchname> exists
+       Reset <branchname> to <start-point>, even if <branchname> exists
        already. Without `-f`, 'git branch' refuses to change an existing branch.
        In combination with `-d` (or `--delete`), allow deleting the
        branch irrespective of its merged status, or whether it even
index ec30b5c5743fd64ea9d42f4f451dc6ce0627d80a..830f0a2efff95bc1b48bf9b1cbec7b6f481be2f2 100644 (file)
@@ -45,7 +45,9 @@ OPTIONS
 
 -s::
        Instead of the content, show the object size identified by
-       `<object>`.
+       `<object>`. If used with `--use-mailmap` option, will show
+       the size of updated object after replacing idents using the
+       mailmap mechanism.
 
 -e::
        Exit with zero status if `<object>` exists and is a valid
@@ -89,26 +91,49 @@ OPTIONS
 --batch::
 --batch=<format>::
        Print object information and contents for each object provided
-       on stdin.  May not be combined with any other options or arguments
-       except `--textconv` or `--filters`, in which case the input lines
-       also need to specify the path, separated by whitespace.  See the
-       section `BATCH OUTPUT` below for details.
+       on stdin. May not be combined with any other options or arguments
+       except `--textconv`, `--filters`, or `--use-mailmap`.
+       +
+       * When used with `--textconv` or `--filters`, the input lines
+         must specify the path, separated by whitespace. See the section
+         `BATCH OUTPUT` below for details.
+       +
+       * When used with `--use-mailmap`, for commit and tag objects, the
+         contents part of the output shows the identities replaced using the
+         mailmap mechanism, while the information part of the output shows
+         the size of the object as if it actually recorded the replacement
+         identities.
 
 --batch-check::
 --batch-check=<format>::
-       Print object information for each object provided on stdin.  May
-       not be combined with any other options or arguments except
-       `--textconv` or `--filters`, in which case the input lines also
-       need to specify the path, separated by whitespace.  See the
-       section `BATCH OUTPUT` below for details.
+       Print object information for each object provided on stdin. May not be
+       combined with any other options or arguments except `--textconv`, `--filters`
+       or `--use-mailmap`.
+       +
+       * When used with `--textconv` or `--filters`, the input lines must
+        specify the path, separated by whitespace. See the section
+        `BATCH OUTPUT` below for details.
+       +
+       * When used with `--use-mailmap`, for commit and tag objects, the
+         printed object information shows the size of the object as if the
+         identities recorded in it were replaced by the mailmap mechanism.
 
 --batch-command::
 --batch-command=<format>::
        Enter a command mode that reads commands and arguments from stdin. May
-       only be combined with `--buffer`, `--textconv` or `--filters`. In the
-       case of `--textconv` or `--filters`, the input lines also need to specify
-       the path, separated by whitespace. See the section `BATCH OUTPUT` below
-       for details.
+       only be combined with `--buffer`, `--textconv`, `--use-mailmap` or
+       `--filters`.
+       +
+       * When used with `--textconv` or `--filters`, the input lines must
+         specify the path, separated by whitespace. See the section
+         `BATCH OUTPUT` below for details.
+       +
+       * When used with `--use-mailmap`, for commit and tag objects, the
+         `contents` command shows the identities replaced using the
+         mailmap mechanism, while the `info` command shows the size
+         of the object as if it actually recorded the replacement
+         identities.
+
 +
 `--batch-command` recognizes the following commands:
 +
index 04bcc416e6e2966b8b83d911cac61307e0f4feaf..88ee942101a7748ac2a7b5030b364ed7e27d5fdb 100644 (file)
@@ -64,6 +64,11 @@ OPTIONS
        share no common history.  This flag can be given to override that
        check and make the merge proceed anyway.
 
+--merge-base=<commit>::
+       Instead of finding the merge-bases for <branch1> and <branch2>,
+       specify a merge-base for the merge, and specifying multiple bases is
+       currently not supported. This option is incompatible with `--stdin`.
+
 [[OUTPUT]]
 OUTPUT
 ------
@@ -216,6 +221,17 @@ with linkgit:git-merge[1]:
   * any messages that would have been printed to stdout (the
     <<IM,Informational messages>>)
 
+INPUT FORMAT
+------------
+'git merge-tree --stdin' input format is fully text based. Each line
+has this format:
+
+       [<base-commit> -- ]<branch1> <branch2>
+
+If one line is separated by `--`, the string before the separator is
+used for specifying a merge-base for the merge and the string after
+the separator describes the branches to be merged.
+
 MISTAKES TO AVOID
 -----------------
 
index 5e438a7fdc1ca4e8684c52c3773a6f722b26c287..a051b1e8f383abc618397ea7e2ac2f173d30a111 100644 (file)
@@ -457,6 +457,66 @@ during the write may conflict with other simultaneous processes, causing
 them to fail. Scripts running `status` in the background should consider
 using `git --no-optional-locks status` (see linkgit:git[1] for details).
 
+UNTRACKED FILES AND PERFORMANCE
+-------------------------------
+
+`git status` can be very slow in large worktrees if/when it
+needs to search for untracked files and directories. There are
+many configuration options available to speed this up by either
+avoiding the work or making use of cached results from previous
+Git commands. There is no single optimum set of settings right
+for everyone. We'll list a summary of the relevant options to help
+you, but before going into the list, you may want to run `git status`
+again, because your configuration may already be caching `git status`
+results, so it could be faster on subsequent runs.
+
+* The `--untracked-files=no` flag or the
+       `status.showUntrackedfiles=false` config (see above for both):
+       indicate that `git status` should not report untracked
+       files. This is the fastest option. `git status` will not list
+       the untracked files, so you need to be careful to remember if
+       you create any new files and manually `git add` them.
+
+* `advice.statusUoption=false` (see linkgit:git-config[1]):
+       setting this variable to `false` disables the warning message
+       given when enumerating untracked files takes more than 2
+       seconds.  In a large project, it may take longer and the user
+       may have already accepted the trade off (e.g. using "-uno" may
+       not be an acceptable option for the user), in which case, there
+       is no point issuing the warning message, and in such a case,
+       disabling the warning may be the best.
+
+* `core.untrackedCache=true` (see linkgit:git-update-index[1]):
+       enable the untracked cache feature and only search directories
+       that have been modified since the previous `git status` command.
+       Git remembers the set of untracked files within each directory
+       and assumes that if a directory has not been modified, then
+       the set of untracked files within has not changed.  This is much
+       faster than enumerating the contents of every directory, but still
+       not without cost, because Git still has to search for the set of
+       modified directories. The untracked cache is stored in the
+       `.git/index` file. The reduced cost of searching for untracked
+       files is offset slightly by the increased size of the index and
+       the cost of keeping it up-to-date. That reduced search time is
+       usually worth the additional size.
+
+* `core.untrackedCache=true` and `core.fsmonitor=true` or
+       `core.fsmonitor=<hook_command_pathname>` (see
+       linkgit:git-update-index[1]): enable both the untracked cache
+       and FSMonitor features and only search directories that have
+       been modified since the previous `git status` command.  This
+       is faster than using just the untracked cache alone because
+       Git can also avoid searching for modified directories.  Git
+       only has to enumerate the exact set of directories that have
+       changed recently. While the FSMonitor feature can be enabled
+       without the untracked cache, the benefits are greatly reduced
+       in that case.
+
+Note that after you turn on the untracked cache and/or FSMonitor
+features it may take a few `git status` commands for the various
+caches to warm up before you see improved command times.  This is
+normal.
+
 SEE ALSO
 --------
 linkgit:gitignore[5]
index 6aa521fab2383e5e0d4aaf39d03f695ea53940d3..f40202b8e3ab521ea22c78986d3bf5bb44a67f09 100644 (file)
@@ -13,7 +13,8 @@ SYNOPSIS
 
 DESCRIPTION
 -----------
-Prints a Git logical variable.
+Prints a Git logical variable. Exits with code 1 if the variable has
+no value.
 
 OPTIONS
 -------
@@ -49,6 +50,14 @@ ifdef::git-default-editor[]
     The build you are using chose '{git-default-editor}' as the default.
 endif::git-default-editor[]
 
+GIT_SEQUENCE_EDITOR::
+    Text editor used to edit the 'todo' file while running `git rebase
+    -i`. Like `GIT_EDITOR`, the value is meant to be interpreted by
+    the shell when it is used. The order of preference is the
+    `$GIT_SEQUENCE_EDITOR` environment variable, then
+    `sequence.editor` configuration, and then the value of `git var
+    GIT_EDITOR`.
+
 GIT_PAGER::
     Text viewer for use by Git commands (e.g., 'less').  The value
     is meant to be interpreted by the shell.  The order of preference
index 1d33e083ab8ba104641001cd6e85b49505c3c374..f9a7a4554cd8ab1e66f5c1a80783809cc5f8373a 100644 (file)
@@ -13,8 +13,7 @@ SYNOPSIS
     [--exec-path[=<path>]] [--html-path] [--man-path] [--info-path]
     [-p|--paginate|-P|--no-pager] [--no-replace-objects] [--bare]
     [--git-dir=<path>] [--work-tree=<path>] [--namespace=<name>]
-    [--super-prefix=<path>] [--config-env=<name>=<envvar>]
-    <command> [<args>]
+    [--config-env=<name>=<envvar>] <command> [<args>]
 
 DESCRIPTION
 -----------
@@ -169,11 +168,6 @@ If you just want to run git as if it was started in `<path>` then use
        details.  Equivalent to setting the `GIT_NAMESPACE` environment
        variable.
 
---super-prefix=<path>::
-       Currently for internal use only.  Set a prefix which gives a path from
-       above a repository down to its root.  One use is to give submodules
-       context about the superproject that invoked it.
-
 --bare::
        Treat the repository as a bare repository.  If GIT_DIR
        environment is not set, it is set to the current working
index 4b36d51beb66f08bff4576ed3afcbc9a6d78d63a..c19e64ea0eff616010436fb9edb947ef20c8ecba 100644 (file)
@@ -1155,7 +1155,7 @@ Unspecified::
 
 String::
 
-       Specify a comma separate list of common whitespace problems to
+       Specify a comma separated list of common whitespace problems to
        notice in the same format as the `core.whitespace` configuration
        variable.
 
index 59bf41cefb9b958f57e6e95201784ec8cec56fb0..10bd2d40cecc48975bceedb258b9ed3f2fc0a896 100644 (file)
@@ -578,6 +578,207 @@ and associated requested information, each separated by a single space.
 
        obj-info = obj-id SP obj-size
 
+bundle-uri
+~~~~~~~~~~
+
+If the 'bundle-uri' capability is advertised, the server supports the
+`bundle-uri' command.
+
+The capability is currently advertised with no value (i.e. not
+"bundle-uri=somevalue"), a value may be added in the future for
+supporting command-wide extensions. Clients MUST ignore any unknown
+capability values and proceed with the 'bundle-uri` dialog they
+support.
+
+The 'bundle-uri' command is intended to be issued before `fetch` to
+get URIs to bundle files (see linkgit:git-bundle[1]) to "seed" and
+inform the subsequent `fetch` command.
+
+The client CAN issue `bundle-uri` before or after any other valid
+command. To be useful to clients it's expected that it'll be issued
+after an `ls-refs` and before `fetch`, but CAN be issued at any time
+in the dialog.
+
+DISCUSSION of bundle-uri
+^^^^^^^^^^^^^^^^^^^^^^^^
+
+The intent of the feature is optimize for server resource consumption
+in the common case by changing the common case of fetching a very
+large PACK during linkgit:git-clone[1] into a smaller incremental
+fetch.
+
+It also allows servers to achieve better caching in combination with
+an `uploadpack.packObjectsHook` (see linkgit:git-config[1]).
+
+By having new clones or fetches be a more predictable and common
+negotiation against the tips of recently produces *.bundle file(s).
+Servers might even pre-generate the results of such negotiations for
+the `uploadpack.packObjectsHook` as new pushes come in.
+
+One way that servers could take advantage of these bundles is that the
+server would anticipate that fresh clones will download a known bundle,
+followed by catching up to the current state of the repository using ref
+tips found in that bundle (or bundles).
+
+PROTOCOL for bundle-uri
+^^^^^^^^^^^^^^^^^^^^^^^
+
+A `bundle-uri` request takes no arguments, and as noted above does not
+currently advertise a capability value. Both may be added in the
+future.
+
+When the client issues a `command=bundle-uri` request, the response is a
+list of key-value pairs provided as packet lines with value
+`<key>=<value>`. Each `<key>` should be interpreted as a config key from
+the `bundle.*` namespace to construct a list of bundles. These keys are
+grouped by a `bundle.<id>.` subsection, where each key corresponding to a
+given `<id>` contributes attributes to the bundle defined by that `<id>`.
+See linkgit:git-config[1] for the specific details of these keys and how
+the Git client will interpret their values.
+
+Clients MUST parse the line according to the above format, lines that do
+not conform to the format SHOULD be discarded. The user MAY be warned in
+such a case.
+
+bundle-uri CLIENT AND SERVER EXPECTATIONS
+^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
+
+URI CONTENTS::
+The content at the advertised URIs MUST be one of two types.
++
+The advertised URI may contain a bundle file that `git bundle verify`
+would accept. I.e. they MUST contain one or more reference tips for
+use by the client, MUST indicate prerequisites (in any) with standard
+"-" prefixes, and MUST indicate their "object-format", if
+applicable.
++
+The advertised URI may alternatively contain a plaintext file that `git
+config --list` would accept (with the `--file` option). The key-value
+pairs in this list are in the `bundle.*` namespace (see
+linkgit:git-config[1]).
+
+bundle-uri CLIENT ERROR RECOVERY::
+A client MUST above all gracefully degrade on errors, whether that
+error is because of bad missing/data in the bundle URI(s), because
+that client is too dumb to e.g. understand and fully parse out bundle
+headers and their prerequisite relationships, or something else.
++
+Server operators should feel confident in turning on "bundle-uri" and
+not worry if e.g. their CDN goes down that clones or fetches will run
+into hard failures. Even if the server bundle bundle(s) are
+incomplete, or bad in some way the client should still end up with a
+functioning repository, just as if it had chosen not to use this
+protocol extension.
++
+All subsequent discussion on client and server interaction MUST keep
+this in mind.
+
+bundle-uri SERVER TO CLIENT::
+The ordering of the returned bundle uris is not significant. Clients
+MUST parse their headers to discover their contained OIDS and
+prerequisites. A client MUST consider the content of the bundle(s)
+themselves and their header as the ultimate source of truth.
++
+A server MAY even return bundle(s) that don't have any direct
+relationship to the repository being cloned (either through accident,
+or intentional "clever" configuration), and expect a client to sort
+out what data they'd like from the bundle(s), if any.
+
+bundle-uri CLIENT TO SERVER::
+The client SHOULD provide reference tips found in the bundle header(s)
+as 'have' lines in any subsequent `fetch` request. A client MAY also
+ignore the bundle(s) entirely if doing so is deemed worse for some
+reason, e.g. if the bundles can't be downloaded, it doesn't like the
+tips it finds etc.
+
+WHEN ADVERTISED BUNDLE(S) REQUIRE NO FURTHER NEGOTIATION::
+If after issuing `bundle-uri` and `ls-refs`, and getting the header(s)
+of the bundle(s) the client finds that the ref tips it wants can be
+retrieved entirely from advertised bundle(s), the client MAY disconnect
+from the Git server. The results of such a 'clone' or 'fetch' should be
+indistinguishable from the state attained without using bundle-uri.
+
+EARLY CLIENT DISCONNECTIONS AND ERROR RECOVERY::
+A client MAY perform an early disconnect while still downloading the
+bundle(s) (having streamed and parsed their headers). In such a case
+the client MUST gracefully recover from any errors related to
+finishing the download and validation of the bundle(s).
++
+I.e. a client might need to re-connect and issue a 'fetch' command,
+and possibly fall back to not making use of 'bundle-uri' at all.
++
+This "MAY" behavior is specified as such (and not a "SHOULD") on the
+assumption that a server advertising bundle uris is more likely than
+not to be serving up a relatively large repository, and to be pointing
+to URIs that have a good chance of being in working order. A client
+MAY e.g. look at the payload size of the bundles as a heuristic to see
+if an early disconnect is worth it, should falling back on a full
+"fetch" dialog be necessary.
+
+WHEN ADVERTISED BUNDLE(S) REQUIRE FURTHER NEGOTIATION::
+A client SHOULD commence a negotiation of a PACK from the server via
+the "fetch" command using the OID tips found in advertised bundles,
+even if's still in the process of downloading those bundle(s).
++
+This allows for aggressive early disconnects from any interactive
+server dialog. The client blindly trusts that the advertised OID tips
+are relevant, and issues them as 'have' lines, it then requests any
+tips it would like (usually from the "ls-refs" advertisement) via
+'want' lines. The server will then compute a (hopefully small) PACK
+with the expected difference between the tips from the bundle(s) and
+the data requested.
++
+The only connection the client then needs to keep active is to the
+concurrently downloading static bundle(s), when those and the
+incremental PACK are retrieved they should be inflated and
+validated. Any errors at this point should be gracefully recovered
+from, see above.
+
+bundle-uri PROTOCOL FEATURES
+^^^^^^^^^^^^^^^^^^^^^^^^^^^^
+
+The client constructs a bundle list from the `<key>=<value>` pairs
+provided by the server. These pairs are part of the `bundle.*` namespace
+as documented in linkgit:git-config[1]. In this section, we discuss some
+of these keys and describe the actions the client will do in response to
+this information.
+
+In particular, the `bundle.version` key specifies an integer value. The
+only accepted value at the moment is `1`, but if the client sees an
+unexpected value here then the client MUST ignore the bundle list.
+
+As long as `bundle.version` is understood, all other unknown keys MAY be
+ignored by the client. The server will guarantee compatibility with older
+clients, though newer clients may be better able to use the extra keys to
+minimize downloads.
+
+Any backwards-incompatible addition of pre-URI key-value will be
+guarded by a new `bundle.version` value or values in 'bundle-uri'
+capability advertisement itself, and/or by new future `bundle-uri`
+request arguments.
+
+Some example key-value pairs that are not currently implemented but could
+be implemented in the future include:
+
+ * Add a "hash=<val>" or "size=<bytes>" advertise the expected hash or
+   size of the bundle file.
+
+ * Advertise that one or more bundle files are the same (to e.g. have
+   clients round-robin or otherwise choose one of N possible files).
+
+ * A "oid=<OID>" shortcut and "prerequisite=<OID>" shortcut. For
+   expressing the common case of a bundle with one tip and no
+   prerequisites, or one tip and one prerequisite.
++
+This would allow for optimizing the common case of servers who'd like
+to provide one "big bundle" containing only their "main" branch,
+and/or incremental updates thereof.
++
+A client receiving such a a response MAY assume that they can skip
+retrieving the header from a bundle at the indicated URI, and thus
+save themselves and the server(s) the request(s) needed to inspect the
+headers of that bundle or bundles.
+
 GIT
 ---
 Part of the linkgit:git[1] suite
index b2580ab8513f3cfca206502a06c9708eb9d9af11..c4c2d3e022da810454b1aab7172603ea2aac2209 100755 (executable)
@@ -1,7 +1,7 @@
 #!/bin/sh
 
 GVF=GIT-VERSION-FILE
-DEF_VER=v2.39.0
+DEF_VER=v2.39.GIT
 
 LF='
 '
index b258fdbed8623d44b014d3ec0dde0350ecb73d43..db447d073839458b97ffe47a64b1a21a82c752d9 100644 (file)
--- a/Makefile
+++ b/Makefile
@@ -511,10 +511,8 @@ include shared.mak
 # Define BLK_SHA1 to make use of optimized C SHA-1 routines bundled
 # with git (in the block-sha1/ directory).
 #
-# Define NO_APPLE_COMMON_CRYPTO on OSX to opt-out of using the
-# "APPLE_COMMON_CRYPTO" backend for SHA-1, which is currently the
-# default on that OS. On macOS 01.4 (Tiger) or older,
-# NO_APPLE_COMMON_CRYPTO is defined by default.
+# Define APPLE_COMMON_CRYPTO_SHA1 to use Apple's CommonCrypto for
+# SHA-1.
 #
 # If don't enable any of the *_SHA1 settings in this section, Git will
 # default to its built-in sha1collisiondetection library, which is a
@@ -691,7 +689,6 @@ THIRD_PARTY_SOURCES =
 # interactive shell sessions without exporting it.
 unexport CDPATH
 
-SCRIPT_SH += git-bisect.sh
 SCRIPT_SH += git-difftool--helper.sh
 SCRIPT_SH += git-filter-branch.sh
 SCRIPT_SH += git-merge-octopus.sh
@@ -1202,7 +1199,7 @@ BUILTIN_OBJS += builtin/am.o
 BUILTIN_OBJS += builtin/annotate.o
 BUILTIN_OBJS += builtin/apply.o
 BUILTIN_OBJS += builtin/archive.o
-BUILTIN_OBJS += builtin/bisect--helper.o
+BUILTIN_OBJS += builtin/bisect.o
 BUILTIN_OBJS += builtin/blame.o
 BUILTIN_OBJS += builtin/branch.o
 BUILTIN_OBJS += builtin/bugreport.o
@@ -1912,7 +1909,7 @@ ifdef NO_POSIX_GOODIES
        BASIC_CFLAGS += -DNO_POSIX_GOODIES
 endif
 
-ifdef APPLE_COMMON_CRYPTO
+ifdef APPLE_COMMON_CRYPTO_SHA1
        # Apple CommonCrypto requires chunking
        SHA1_MAX_BLOCK_SIZE = 1024L*1024L*1024L
 endif
@@ -1929,7 +1926,7 @@ ifdef BLK_SHA1
        LIB_OBJS += block-sha1/sha1.o
        BASIC_CFLAGS += -DSHA1_BLK
 else
-ifdef APPLE_COMMON_CRYPTO
+ifdef APPLE_COMMON_CRYPTO_SHA1
        COMPAT_CFLAGS += -DCOMMON_DIGEST_FOR_OPENSSL
        BASIC_CFLAGS += -DSHA1_APPLE
 else
index 758368388a4641dd46cf221bea607f885d487e84..47807bf257fd5e52f95f1fb628867a349034571b 120000 (symlink)
--- a/RelNotes
+++ b/RelNotes
@@ -1 +1 @@
-Documentation/RelNotes/2.39.0.txt
\ No newline at end of file
+Documentation/RelNotes/2.40.0.txt
\ No newline at end of file
index ae1839c04a7fa27125e4a2308c971e63da18492a..00a0f6f96f3f5ebc54bbfb841921c5ceff88c1c3 100644 (file)
@@ -724,7 +724,7 @@ static int run_update(struct add_i_state *s, const struct pathspec *ps,
 }
 
 static void revert_from_diff(struct diff_queue_struct *q,
-                            struct diff_options *opt, void *data)
+                            struct diff_options *opt, void *data UNUSED)
 {
        int i, add_flags = ADD_CACHE_OK_TO_ADD | ADD_CACHE_OK_TO_REPLACE;
 
diff --git a/apply.c b/apply.c
index bc338143134528bcb5e8e7ab56494ad1bfe04a7c..85822280476dc9b729943119802c60d76fd74134 100644 (file)
--- a/apply.c
+++ b/apply.c
@@ -2913,7 +2913,7 @@ static int apply_one_fragment(struct apply_state *state,
                        break;
                case ' ':
                        if (plen && (ws_rule & WS_BLANK_AT_EOF) &&
-                           ws_blank_line(patch + 1, plen, ws_rule))
+                           ws_blank_line(patch + 1, plen))
                                is_blank_context = 1;
                        /* fallthrough */
                case '-':
@@ -2942,7 +2942,7 @@ static int apply_one_fragment(struct apply_state *state,
                                      (first == '+' ? 0 : LINE_COMMON));
                        if (first == '+' &&
                            (ws_rule & WS_BLANK_AT_EOF) &&
-                           ws_blank_line(patch + 1, plen, ws_rule))
+                           ws_blank_line(patch + 1, plen))
                                added_blank_line = 1;
                        break;
                case '@': case '\\':
diff --git a/blob.c b/blob.c
index 182718aba9fe320f1064a12fb5baf1969f563b29..8f83523b0cde6dbf1b808b3dcf8ab1c1c23085f3 100644 (file)
--- a/blob.c
+++ b/blob.c
@@ -13,8 +13,7 @@ struct blob *lookup_blob(struct repository *r, const struct object_id *oid)
        return object_as_type(obj, OBJ_BLOB, 0);
 }
 
-int parse_blob_buffer(struct blob *item, void *buffer, unsigned long size)
+void parse_blob_buffer(struct blob *item)
 {
        item->object.parsed = 1;
-       return 0;
 }
diff --git a/blob.h b/blob.h
index 1664872055783557e1836dcc64bc2f51c20c5b4f..74555c90c449ffa89226d264f46c433580707f61 100644 (file)
--- a/blob.h
+++ b/blob.h
@@ -11,8 +11,6 @@ struct blob {
 
 struct blob *lookup_blob(struct repository *r, const struct object_id *oid);
 
-int parse_blob_buffer(struct blob *item, void *buffer, unsigned long size);
-
 /**
  * Blobs do not contain references to other objects and do not have
  * structured data that needs parsing. However, code may use the
@@ -21,5 +19,6 @@ int parse_blob_buffer(struct blob *item, void *buffer, unsigned long size);
  * parse_blob_buffer() is used (by object.c) to flag that the object
  * has been read successfully from the database.
  **/
+void parse_blob_buffer(struct blob *item);
 
 #endif /* BLOB_H */
index 8901a34d6bf424680b9d13a1bdf332bedb4d8e20..46cc7897898a51a7039e5bc9d0c350f3fab2f75e 100644 (file)
--- a/builtin.h
+++ b/builtin.h
  *     on bare repositories.
  *     This only makes sense when `RUN_SETUP` is also set.
  *
- * `SUPPORT_SUPER_PREFIX`:
- *
- *     The built-in supports `--super-prefix`.
- *
  * `DELAY_PAGER_CONFIG`:
  *
  *     If RUN_SETUP or RUN_SETUP_GENTLY is set, git.c normally handles
@@ -116,7 +112,7 @@ int cmd_am(int argc, const char **argv, const char *prefix);
 int cmd_annotate(int argc, const char **argv, const char *prefix);
 int cmd_apply(int argc, const char **argv, const char *prefix);
 int cmd_archive(int argc, const char **argv, const char *prefix);
-int cmd_bisect__helper(int argc, const char **argv, const char *prefix);
+int cmd_bisect(int argc, const char **argv, const char *prefix);
 int cmd_blame(int argc, const char **argv, const char *prefix);
 int cmd_branch(int argc, const char **argv, const char *prefix);
 int cmd_bugreport(int argc, const char **argv, const char *prefix);
index 76277df326b4f47f594e4580f6f645ffa76455f3..0c60402267653c5aff56c8b2bd1702793e4170be 100644 (file)
@@ -88,7 +88,7 @@ static int fix_unmerged_status(struct diff_filepair *p,
 }
 
 static void update_callback(struct diff_queue_struct *q,
-                           struct diff_options *opt, void *cbdata)
+                           struct diff_options *opt UNUSED, void *cbdata)
 {
        int i;
        struct update_callback_data *data = cbdata;
@@ -695,6 +695,6 @@ finish:
                die(_("Unable to write new index file"));
 
        dir_clear(&dir);
-       UNLEAK(pathspec);
+       clear_pathspec(&pathspec);
        return exit_status;
 }
index 30c9b3a9cd72588fc2fb4495faedcc7cf3eda258..7e88d2426d7e92bfa10164fef0ebd5c9ef213758 100644 (file)
@@ -117,6 +117,7 @@ struct am_state {
 
        /* various operating modes and command line options */
        int interactive;
+       int no_verify;
        int threeway;
        int quiet;
        int signoff; /* enum signoff_type */
@@ -472,10 +473,12 @@ static void am_destroy(const struct am_state *state)
  */
 static int run_applypatch_msg_hook(struct am_state *state)
 {
-       int ret;
+       int ret = 0;
 
        assert(state->msg);
-       ret = run_hooks_l("applypatch-msg", am_path(state, "final-commit"), NULL);
+
+       if (!state->no_verify)
+               ret = run_hooks_l("applypatch-msg", am_path(state, "final-commit"), NULL);
 
        if (!ret) {
                FREE_AND_NULL(state->msg);
@@ -1476,6 +1479,7 @@ static int run_apply(const struct am_state *state, const char *index_file)
        int res, opts_left;
        int force_apply = 0;
        int options = 0;
+       const char **apply_argv;
 
        if (init_apply_state(&apply_state, the_repository, NULL))
                BUG("init_apply_state() failed");
@@ -1483,7 +1487,15 @@ static int run_apply(const struct am_state *state, const char *index_file)
        strvec_push(&apply_opts, "apply");
        strvec_pushv(&apply_opts, state->git_apply_opts.v);
 
-       opts_left = apply_parse_options(apply_opts.nr, apply_opts.v,
+       /*
+        * Build a copy that apply_parse_options() can rearrange.
+        * apply_opts.v keeps referencing the allocated strings for
+        * strvec_clear() to release.
+        */
+       ALLOC_ARRAY(apply_argv, apply_opts.nr);
+       COPY_ARRAY(apply_argv, apply_opts.v, apply_opts.nr);
+
+       opts_left = apply_parse_options(apply_opts.nr, apply_argv,
                                        &apply_state, &force_apply, &options,
                                        NULL);
 
@@ -1513,6 +1525,7 @@ static int run_apply(const struct am_state *state, const char *index_file)
        strvec_clear(&apply_paths);
        strvec_clear(&apply_opts);
        clear_apply_state(&apply_state);
+       free(apply_argv);
 
        if (res)
                return res;
@@ -1640,7 +1653,7 @@ static void do_commit(const struct am_state *state)
        const char *reflog_msg, *author, *committer = NULL;
        struct strbuf sb = STRBUF_INIT;
 
-       if (run_hooks("pre-applypatch"))
+       if (!state->no_verify && run_hooks("pre-applypatch"))
                exit(1);
 
        if (write_cache_as_tree(&tree, 0, NULL))
@@ -2330,6 +2343,8 @@ int cmd_am(int argc, const char **argv, const char *prefix)
        struct option options[] = {
                OPT_BOOL('i', "interactive", &state.interactive,
                        N_("run interactively")),
+               OPT_BOOL('n', "no-verify", &state.no_verify,
+                       N_("bypass pre-applypatch and applypatch-msg hooks")),
                OPT_HIDDEN_BOOL('b', "binary", &binary,
                        N_("historical option -- no-op")),
                OPT_BOOL('3', "3way", &state.threeway,
similarity index 92%
rename from builtin/bisect--helper.c
rename to builtin/bisect.c
index 6e41cbdb2d3785cadb5ab901586585daee8718e0..cc9483e85155f7d4f46f3f2440be855dd403fa85 100644 (file)
@@ -20,18 +20,40 @@ static GIT_PATH_FUNC(git_path_bisect_names, "BISECT_NAMES")
 static GIT_PATH_FUNC(git_path_bisect_first_parent, "BISECT_FIRST_PARENT")
 static GIT_PATH_FUNC(git_path_bisect_run, "BISECT_RUN")
 
-static const char * const git_bisect_helper_usage[] = {
-       N_("git bisect--helper --bisect-reset [<commit>]"),
-       "git bisect--helper --bisect-terms [--term-good | --term-old | --term-bad | --term-new]",
-       N_("git bisect--helper --bisect-start [--term-{new,bad}=<term> --term-{old,good}=<term>]"
-                                           " [--no-checkout] [--first-parent] [<bad> [<good>...]] [--] [<paths>...]"),
-       "git bisect--helper --bisect-next",
-       N_("git bisect--helper --bisect-state (bad|new) [<rev>]"),
-       N_("git bisect--helper --bisect-state (good|old) [<rev>...]"),
-       N_("git bisect--helper --bisect-replay <filename>"),
-       N_("git bisect--helper --bisect-skip [(<rev>|<range>)...]"),
-       "git bisect--helper --bisect-visualize",
-       N_("git bisect--helper --bisect-run <cmd>..."),
+#define BUILTIN_GIT_BISECT_START_USAGE \
+       N_("git bisect start [--term-{new,bad}=<term> --term-{old,good}=<term>]" \
+          "    [--no-checkout] [--first-parent] [<bad> [<good>...]] [--]" \
+          "    [<pathspec>...]")
+#define BUILTIN_GIT_BISECT_STATE_USAGE \
+       N_("git bisect (good|bad) [<rev>...]")
+#define BUILTIN_GIT_BISECT_TERMS_USAGE \
+       "git bisect terms [--term-good | --term-bad]"
+#define BUILTIN_GIT_BISECT_SKIP_USAGE \
+       N_("git bisect skip [(<rev>|<range>)...]")
+#define BUILTIN_GIT_BISECT_NEXT_USAGE \
+       "git bisect next"
+#define BUILTIN_GIT_BISECT_RESET_USAGE \
+       N_("git bisect reset [<commit>]")
+#define BUILTIN_GIT_BISECT_VISUALIZE_USAGE \
+       "git bisect visualize"
+#define BUILTIN_GIT_BISECT_REPLAY_USAGE \
+       N_("git bisect replay <logfile>")
+#define BUILTIN_GIT_BISECT_LOG_USAGE \
+       "git bisect log"
+#define BUILTIN_GIT_BISECT_RUN_USAGE \
+       N_("git bisect run <cmd>...")
+
+static const char * const git_bisect_usage[] = {
+       BUILTIN_GIT_BISECT_START_USAGE,
+       BUILTIN_GIT_BISECT_STATE_USAGE,
+       BUILTIN_GIT_BISECT_TERMS_USAGE,
+       BUILTIN_GIT_BISECT_SKIP_USAGE,
+       BUILTIN_GIT_BISECT_NEXT_USAGE,
+       BUILTIN_GIT_BISECT_RESET_USAGE,
+       BUILTIN_GIT_BISECT_VISUALIZE_USAGE,
+       BUILTIN_GIT_BISECT_REPLAY_USAGE,
+       BUILTIN_GIT_BISECT_LOG_USAGE,
+       BUILTIN_GIT_BISECT_RUN_USAGE,
        NULL
 };
 
@@ -1191,13 +1213,13 @@ static int bisect_run(struct bisect_terms *terms, const char **argv, int argc)
        if (bisect_next_check(terms, NULL))
                return BISECT_FAILED;
 
-       if (argc)
-               sq_quote_argv(&command, argv);
-       else {
+       if (!argc) {
                error(_("bisect run failed: no command provided."));
                return BISECT_FAILED;
        }
 
+       sq_quote_argv(&command, argv);
+       strbuf_ltrim(&command);
        while (1) {
                res = do_bisect_run(command.buf);
 
@@ -1211,8 +1233,8 @@ static int bisect_run(struct bisect_terms *terms, const char **argv, int argc)
                if (is_first_run && (res == 126 || res == 127)) {
                        int rc = verify_good(terms, command.buf);
                        is_first_run = 0;
-                       if (rc < 0) {
-                               error(_("unable to verify '%s' on good"
+                       if (rc < 0 || 128 <= rc) {
+                               error(_("unable to verify %s on good"
                                        " revision"), command.buf);
                                res = BISECT_FAILED;
                                break;
@@ -1227,7 +1249,7 @@ static int bisect_run(struct bisect_terms *terms, const char **argv, int argc)
 
                if (res < 0 || 128 <= res) {
                        error(_("bisect run failed: exit code %d from"
-                               " '%s' is < 0 or >= 128"), res, command.buf);
+                               " %s is < 0 or >= 128"), res, command.buf);
                        break;
                }
 
@@ -1261,14 +1283,14 @@ static int bisect_run(struct bisect_terms *terms, const char **argv, int argc)
                if (res == BISECT_ONLY_SKIPPED_LEFT)
                        error(_("bisect run cannot continue any more"));
                else if (res == BISECT_INTERNAL_SUCCESS_MERGE_BASE) {
-                       printf(_("bisect run success"));
+                       puts(_("bisect run success"));
                        res = BISECT_OK;
                } else if (res == BISECT_INTERNAL_SUCCESS_1ST_BAD_FOUND) {
-                       printf(_("bisect found first bad commit"));
+                       puts(_("bisect found first bad commit"));
                        res = BISECT_OK;
                } else if (res) {
-                       error(_("bisect run failed: 'git bisect--helper --bisect-state"
-                       " %s' exited with error code %d"), new_state, res);
+                       error(_("bisect run failed: 'bisect-state %s'"
+                               " exited with error code %d"), new_state, res);
                } else {
                        continue;
                }
@@ -1282,7 +1304,8 @@ static int bisect_run(struct bisect_terms *terms, const char **argv, int argc)
 static int cmd_bisect__reset(int argc, const char **argv, const char *prefix UNUSED)
 {
        if (argc > 1)
-               return error(_("--bisect-reset requires either no argument or a commit"));
+               return error(_("'%s' requires either no argument or a commit"),
+                            "git bisect reset");
        return bisect_reset(argc ? argv[0] : NULL);
 }
 
@@ -1292,7 +1315,8 @@ static int cmd_bisect__terms(int argc, const char **argv, const char *prefix UNU
        struct bisect_terms terms = { 0 };
 
        if (argc > 1)
-               return error(_("--bisect-terms requires 0 or 1 argument"));
+               return error(_("'%s' requires 0 or 1 argument"),
+                            "git bisect terms");
        res = bisect_terms(&terms, argc == 1 ? argv[0] : NULL);
        free_terms(&terms);
        return res;
@@ -1315,29 +1339,16 @@ static int cmd_bisect__next(int argc, const char **argv UNUSED, const char *pref
        struct bisect_terms terms = { 0 };
 
        if (argc)
-               return error(_("--bisect-next requires 0 arguments"));
+               return error(_("'%s' requires 0 arguments"),
+                            "git bisect next");
        get_terms(&terms);
        res = bisect_next(&terms, prefix);
        free_terms(&terms);
        return res;
 }
 
-static int cmd_bisect__state(int argc, const char **argv, const char *prefix UNUSED)
+static int cmd_bisect__log(int argc UNUSED, const char **argv UNUSED, const char *prefix UNUSED)
 {
-       int res;
-       struct bisect_terms terms = { 0 };
-
-       set_terms(&terms, "bad", "good");
-       get_terms(&terms);
-       res = bisect_state(&terms, argv, argc);
-       free_terms(&terms);
-       return res;
-}
-
-static int cmd_bisect__log(int argc, const char **argv UNUSED, const char *prefix UNUSED)
-{
-       if (argc)
-               return error(_("--bisect-log requires 0 arguments"));
        return bisect_log();
 }
 
@@ -1383,14 +1394,14 @@ static int cmd_bisect__run(int argc, const char **argv, const char *prefix UNUSE
        struct bisect_terms terms = { 0 };
 
        if (!argc)
-               return error(_("bisect run failed: no command provided."));
+               return error(_("'%s' failed: no command provided."), "git bisect run");
        get_terms(&terms);
        res = bisect_run(&terms, argv, argc);
        free_terms(&terms);
        return res;
 }
 
-int cmd_bisect__helper(int argc, const char **argv, const char *prefix)
+int cmd_bisect(int argc, const char **argv, const char *prefix)
 {
        int res = 0;
        parse_opt_subcommand_fn *fn = NULL;
@@ -1399,7 +1410,6 @@ int cmd_bisect__helper(int argc, const char **argv, const char *prefix)
                OPT_SUBCOMMAND("terms", &fn, cmd_bisect__terms),
                OPT_SUBCOMMAND("start", &fn, cmd_bisect__start),
                OPT_SUBCOMMAND("next", &fn, cmd_bisect__next),
-               OPT_SUBCOMMAND("state", &fn, cmd_bisect__state),
                OPT_SUBCOMMAND("log", &fn, cmd_bisect__log),
                OPT_SUBCOMMAND("replay", &fn, cmd_bisect__replay),
                OPT_SUBCOMMAND("skip", &fn, cmd_bisect__skip),
@@ -1408,15 +1418,27 @@ int cmd_bisect__helper(int argc, const char **argv, const char *prefix)
                OPT_SUBCOMMAND("run", &fn, cmd_bisect__run),
                OPT_END()
        };
-       argc = parse_options(argc, argv, prefix, options,
-                            git_bisect_helper_usage, 0);
-
-       if (!fn)
-               usage_with_options(git_bisect_helper_usage, options);
-       argc--;
-       argv++;
-
-       res = fn(argc, argv, prefix);
+       argc = parse_options(argc, argv, prefix, options, git_bisect_usage,
+                            PARSE_OPT_SUBCOMMAND_OPTIONAL);
+
+       if (!fn) {
+               struct bisect_terms terms = { 0 };
+
+               if (!argc)
+                       usage_msg_opt(_("need a command"), git_bisect_usage, options);
+
+               set_terms(&terms, "bad", "good");
+               get_terms(&terms);
+               if (check_and_set_terms(&terms, argv[0]))
+                       usage_msg_optf(_("unknown command: '%s'"), git_bisect_usage,
+                                      options, argv[0]);
+               res = bisect_state(&terms, argv, argc);
+               free_terms(&terms);
+       } else {
+               argc--;
+               argv++;
+               res = fn(argc, argv, prefix);
+       }
 
        /*
         * Handle early success
index 9470c980c15dca8a96c264772991b51813f1b355..f63fd45edb96b513c24e1a9cd7c8a55e5a5f7b13 100644 (file)
@@ -581,13 +581,13 @@ static void copy_or_rename_branch(const char *oldname, const char *newname, int
        strbuf_release(&logmsg);
 
        strbuf_addf(&oldsection, "branch.%s", interpreted_oldname);
-       strbuf_release(&oldref);
        strbuf_addf(&newsection, "branch.%s", interpreted_newname);
-       strbuf_release(&newref);
        if (!copy && git_config_rename_section(oldsection.buf, newsection.buf) < 0)
                die(_("Branch is renamed, but update of config-file failed"));
-       if (copy && strcmp(oldname, newname) && git_config_copy_section(oldsection.buf, newsection.buf) < 0)
+       if (copy && strcmp(interpreted_oldname, interpreted_newname) && git_config_copy_section(oldsection.buf, newsection.buf) < 0)
                die(_("Branch is copied, but update of config-file failed"));
+       strbuf_release(&oldref);
+       strbuf_release(&newref);
        strbuf_release(&oldsection);
        strbuf_release(&newsection);
 }
index 96052541cbfe74cda5aff77396c94f4fb6397ae7..5bc254be80f97e7bb33c732039b7bab620fb71ad 100644 (file)
@@ -106,6 +106,7 @@ int cmd_bugreport(int argc, const char **argv, const char *prefix)
        const char *user_relative_path = NULL;
        char *prefixed_filename;
        size_t output_path_len;
+       int ret;
 
        const struct option bugreport_options[] = {
                OPT_CALLBACK_F(0, "diagnose", &diagnose, N_("mode"),
@@ -182,7 +183,9 @@ int cmd_bugreport(int argc, const char **argv, const char *prefix)
                user_relative_path);
 
        free(prefixed_filename);
-       UNLEAK(buffer);
-       UNLEAK(report_path);
-       return !!launch_editor(report_path.buf, NULL, NULL);
+       strbuf_release(&buffer);
+
+       ret = !!launch_editor(report_path.buf, NULL, NULL);
+       strbuf_release(&report_path);
+       return ret;
 }
index c12c09f8549940774c2882f18259cfc13318bdb2..acceef6200172cf7a44de45bfa6f3e940b6444c9 100644 (file)
@@ -55,13 +55,12 @@ static int parse_options_cmd_bundle(int argc,
                const char * const usagestr[],
                const struct option options[],
                char **bundle_file) {
-       int newargc;
-       newargc = parse_options(argc, argv, NULL, options, usagestr,
+       argc = parse_options(argc, argv, NULL, options, usagestr,
                             PARSE_OPT_STOP_AT_NON_OPTION);
-       if (argc < 1)
-               usage_with_options(usagestr, options);
+       if (!argc)
+               usage_msg_opt(_("need a <file> argument"), usagestr, options);
        *bundle_file = prefix_filename(prefix, argv[0]);
-       return newargc;
+       return argc;
 }
 
 static int cmd_bundle_create(int argc, const char **argv, const char *prefix) {
index b3be58b1fb06678376568753f9fc221d8cfb3750..cc17635e76536471a0a756903a8187eaa01d5733 100644 (file)
@@ -132,8 +132,21 @@ static int cat_one_file(int opt, const char *exp_type, const char *obj_name,
 
        case 's':
                oi.sizep = &size;
+
+               if (use_mailmap) {
+                       oi.typep = &type;
+                       oi.contentp = (void**)&buf;
+               }
+
                if (oid_object_info_extended(the_repository, &oid, &oi, flags) < 0)
                        die("git cat-file: could not get object info");
+
+               if (use_mailmap && (type == OBJ_COMMIT || type == OBJ_TAG)) {
+                       size_t s = size;
+                       buf = replace_idents_using_mailmap(buf, &s);
+                       size = cast_size_t_to_ulong(s);
+               }
+
                printf("%"PRIuMAX"\n", (uintmax_t)size);
                ret = 0;
                goto cleanup;
@@ -431,6 +444,9 @@ static void batch_object_write(const char *obj_name,
        if (!data->skip_object_info) {
                int ret;
 
+               if (use_mailmap)
+                       data->info.typep = &data->type;
+
                if (pack)
                        ret = packed_object_info(the_repository, pack, offset,
                                                 &data->info);
@@ -444,6 +460,18 @@ static void batch_object_write(const char *obj_name,
                        fflush(stdout);
                        return;
                }
+
+               if (use_mailmap && (data->type == OBJ_COMMIT || data->type == OBJ_TAG)) {
+                       size_t s = data->size;
+                       char *buf = NULL;
+
+                       buf = repo_read_object_file(the_repository, &data->oid, &data->type,
+                                                   &data->size);
+                       buf = replace_idents_using_mailmap(buf, &s);
+                       data->size = cast_size_t_to_ulong(s);
+
+                       free(buf);
+               }
        }
 
        strbuf_reset(scratch);
index 3fa29a08eea1f13ddfe96c1634f920284b87ad11..cd04f0bf572dd7e434da0d7cff6e67ae4cd72af7 100644 (file)
@@ -232,7 +232,7 @@ static int checkout_stage(int stage, const struct cache_entry *ce, int pos,
                pos++;
        }
        if (!overlay_mode) {
-               unlink_entry(ce);
+               unlink_entry(ce, NULL);
                return 0;
        }
        if (stage == 2)
@@ -1471,6 +1471,8 @@ static void die_if_some_operation_in_progress(void)
                      "or \"git worktree add\"."));
        if (state.bisect_in_progress)
                warning(_("you are switching branch while bisecting"));
+
+       wt_status_state_free_buffers(&state);
 }
 
 static int checkout_branch(struct checkout_opts *opts,
index f518bb2dc1fbad357e93511d8deee851ef56b812..5453ba5277f968cbac419b7aee9911182beb41e6 100644 (file)
@@ -1271,6 +1271,27 @@ int cmd_clone(int argc, const char **argv, const char *prefix)
        if (refs)
                mapped_refs = wanted_peer_refs(refs, &remote->fetch);
 
+       if (!bundle_uri) {
+               /*
+               * Populate transport->got_remote_bundle_uri and
+               * transport->bundle_uri. We might get nothing.
+               */
+               transport_get_remote_bundle_uri(transport);
+
+               if (transport->bundles &&
+                   hashmap_get_size(&transport->bundles->bundles)) {
+                       /* At this point, we need the_repository to match the cloned repo. */
+                       if (repo_init(the_repository, git_dir, work_tree))
+                               warning(_("failed to initialize the repo, skipping bundle URI"));
+                       else if (fetch_bundle_list(the_repository,
+                                                  transport->bundles))
+                               warning(_("failed to fetch advertised bundles"));
+               } else {
+                       clear_bundle_list(transport->bundles);
+                       FREE_AND_NULL(transport->bundles);
+               }
+       }
+
        if (mapped_refs) {
                int hash_algo = hash_algo_by_ptr(transport_get_hash_algo(transport));
 
index 06b1330346f895af5103fdb4b52243b57e6f4f18..44b763d7cd025db45581d60902502ae2a2a9fe11 100644 (file)
@@ -991,8 +991,11 @@ static int prepare_to_commit(const char *index_file, const char *prefix,
                struct object_id oid;
                const char *parent = "HEAD";
 
-               if (!active_nr && read_cache() < 0)
-                       die(_("Cannot read index"));
+               if (!active_nr) {
+                       discard_cache();
+                       if (read_cache() < 0)
+                               die(_("Cannot read index"));
+               }
 
                if (amend)
                        parent = "HEAD^1";
@@ -1875,8 +1878,8 @@ int cmd_commit(int argc, const char **argv, const char *prefix)
        apply_autostash(git_path_merge_autostash(the_repository));
 
 cleanup:
-       UNLEAK(author_ident);
-       UNLEAK(err);
-       UNLEAK(sb);
+       strbuf_release(&author_ident);
+       strbuf_release(&err);
+       strbuf_release(&sb);
        return ret;
 }
index 753e5fac297e08763317adcab2fefe386ea68421..060cf9f3e05e6718ae02923e132e4804dbdcb291 100644 (file)
@@ -639,8 +639,9 @@ static char *default_user_config(void)
 int cmd_config(int argc, const char **argv, const char *prefix)
 {
        int nongit = !startup_info->have_repository;
-       char *value;
+       char *value = NULL;
        int flags = 0;
+       int ret = 0;
 
        given_config_source.file = xstrdup_or_null(getenv(CONFIG_ENVIRONMENT));
 
@@ -856,44 +857,38 @@ int cmd_config(int argc, const char **argv, const char *prefix)
                free(config_file);
        }
        else if (actions == ACTION_SET) {
-               int ret;
                check_write();
                check_argc(argc, 2, 2);
                value = normalize_value(argv[0], argv[1]);
-               UNLEAK(value);
                ret = git_config_set_in_file_gently(given_config_source.file, argv[0], value);
                if (ret == CONFIG_NOTHING_SET)
                        error(_("cannot overwrite multiple values with a single value\n"
                        "       Use a regexp, --add or --replace-all to change %s."), argv[0]);
-               return ret;
        }
        else if (actions == ACTION_SET_ALL) {
                check_write();
                check_argc(argc, 2, 3);
                value = normalize_value(argv[0], argv[1]);
-               UNLEAK(value);
-               return git_config_set_multivar_in_file_gently(given_config_source.file,
-                                                             argv[0], value, argv[2],
-                                                             flags);
+               ret = git_config_set_multivar_in_file_gently(given_config_source.file,
+                                                            argv[0], value, argv[2],
+                                                            flags);
        }
        else if (actions == ACTION_ADD) {
                check_write();
                check_argc(argc, 2, 2);
                value = normalize_value(argv[0], argv[1]);
-               UNLEAK(value);
-               return git_config_set_multivar_in_file_gently(given_config_source.file,
-                                                             argv[0], value,
-                                                             CONFIG_REGEX_NONE,
-                                                             flags);
+               ret = git_config_set_multivar_in_file_gently(given_config_source.file,
+                                                            argv[0], value,
+                                                            CONFIG_REGEX_NONE,
+                                                            flags);
        }
        else if (actions == ACTION_REPLACE_ALL) {
                check_write();
                check_argc(argc, 2, 3);
                value = normalize_value(argv[0], argv[1]);
-               UNLEAK(value);
-               return git_config_set_multivar_in_file_gently(given_config_source.file,
-                                                             argv[0], value, argv[2],
-                                                             flags | CONFIG_FLAGS_MULTI_REPLACE);
+               ret = git_config_set_multivar_in_file_gently(given_config_source.file,
+                                                            argv[0], value, argv[2],
+                                                            flags | CONFIG_FLAGS_MULTI_REPLACE);
        }
        else if (actions == ACTION_GET) {
                check_argc(argc, 1, 2);
@@ -934,26 +929,28 @@ int cmd_config(int argc, const char **argv, const char *prefix)
                                                              flags | CONFIG_FLAGS_MULTI_REPLACE);
        }
        else if (actions == ACTION_RENAME_SECTION) {
-               int ret;
                check_write();
                check_argc(argc, 2, 2);
                ret = git_config_rename_section_in_file(given_config_source.file,
                                                        argv[0], argv[1]);
                if (ret < 0)
                        return ret;
-               if (ret == 0)
+               else if (!ret)
                        die(_("no such section: %s"), argv[0]);
+               else
+                       ret = 0;
        }
        else if (actions == ACTION_REMOVE_SECTION) {
-               int ret;
                check_write();
                check_argc(argc, 1, 1);
                ret = git_config_rename_section_in_file(given_config_source.file,
                                                        argv[0], NULL);
                if (ret < 0)
                        return ret;
-               if (ret == 0)
+               else if (!ret)
                        die(_("no such section: %s"), argv[0]);
+               else
+                       ret = 0;
        }
        else if (actions == ACTION_GET_COLOR) {
                check_argc(argc, 1, 2);
@@ -966,5 +963,6 @@ int cmd_config(int argc, const char **argv, const char *prefix)
                return get_colorbool(argv[0], argc == 2);
        }
 
-       return 0;
+       free(value);
+       return ret;
 }
index 163f2c6a874266f183b276fa30961e76feb11316..26f1e532c66b173fb4b29d220564fcd90c03d0d2 100644 (file)
@@ -612,7 +612,7 @@ int cmd_diff(int argc, const char **argv, const char *prefix)
        if (1 < rev.diffopt.skip_stat_unmatch)
                refresh_index_quietly();
        release_revisions(&rev);
-       UNLEAK(ent);
+       object_array_clear(&ent);
        UNLEAK(blob);
        return result;
 }
index 3b3314e7b2a3343b78d14ee3bc9d4a92fb578fd6..39a890fc005192a5082a1c170550f4be102efbc1 100644 (file)
@@ -409,7 +409,7 @@ static const char *anonymize_oid(const char *oid_hex)
 }
 
 static void show_filemodify(struct diff_queue_struct *q,
-                           struct diff_options *options, void *data)
+                           struct diff_options *options UNUSED, void *data)
 {
        int i;
        struct string_list *changed = data;
index 6f30a4f93a7fa93873e40312a0fcb569056daf88..0feef8caf6d5e27d508b3d4ea3e2679cd50a9966 100644 (file)
@@ -710,6 +710,7 @@ static int do_handle_client(struct fsmonitor_daemon_state *state,
                                  "fsmonitor: unsupported V1 protocol '%s'"),
                                 command);
                do_trivial = 1;
+               do_cookie = 1;
 
        } else {
                /* We have "builtin:*" */
@@ -719,6 +720,7 @@ static int do_handle_client(struct fsmonitor_daemon_state *state,
                                         "fsmonitor: invalid V2 protocol token '%s'",
                                         command);
                        do_trivial = 1;
+                       do_cookie = 1;
 
                } else {
                        /*
@@ -1209,7 +1211,7 @@ static int fsmonitor_run_daemon_1(struct fsmonitor_daemon_state *state)
         * events.
         */
        if (pthread_create(&state->listener_thread, NULL,
-                          fsm_listen__thread_proc, state) < 0) {
+                          fsm_listen__thread_proc, state)) {
                ipc_server_stop_async(state->ipc_server_data);
                err = error(_("could not start fsmonitor listener thread"));
                goto cleanup;
@@ -1220,7 +1222,7 @@ static int fsmonitor_run_daemon_1(struct fsmonitor_daemon_state *state)
         * Start the health thread to watch over our process.
         */
        if (pthread_create(&state->health_thread, NULL,
-                          fsm_health__thread_proc, state) < 0) {
+                          fsm_health__thread_proc, state)) {
                ipc_server_stop_async(state->ipc_server_data);
                err = error(_("could not start fsmonitor health thread"));
                goto cleanup;
index 89447a50838562795bb80172db8102ff7a42e91b..057e299c245c7d4a043fb37ff0e6d6e5a8a6e6d9 100644 (file)
@@ -52,6 +52,7 @@ static int decoration_style;
 static int decoration_given;
 static int use_mailmap_config = 1;
 static unsigned int force_in_body_from;
+static int stdout_mboxrd;
 static const char *fmt_patch_subject_prefix = "PATCH";
 static int fmt_patch_name_max = FORMAT_PATCH_NAME_MAX_DEFAULT;
 static const char *fmt_pretty;
@@ -1077,6 +1078,10 @@ static int git_format_config(const char *var, const char *value, void *cb)
                cover_from_description_mode = parse_cover_from_description(value);
                return 0;
        }
+       if (!strcmp(var, "format.mboxrd")) {
+               stdout_mboxrd = git_config_bool(var, value);
+               return 0;
+       }
 
        return git_log_config(var, value, cb);
 }
@@ -2105,6 +2110,9 @@ int cmd_format_patch(int argc, const char **argv, const char *prefix)
                                  rev.diffopt.close_file, "--output",
                                  !!output_directory, "--output-directory");
 
+       if (use_stdout && stdout_mboxrd)
+               rev.commit_format = CMIT_FMT_MBOXRD;
+
        if (use_stdout) {
                setup_pager();
        } else if (!rev.diffopt.close_file) {
index 4cf8a2364835c9fc1ae39b8d52fd09a8aeeb368e..a03b559ecaab4caeba9a9a18bb268f22274e7b2a 100644 (file)
@@ -613,6 +613,7 @@ void overlay_tree_on_index(struct index_state *istate,
        if (!fn)
                fn = read_one_entry_quick;
        err = read_tree(the_repository, tree, &pathspec, fn, istate);
+       clear_pathspec(&pathspec);
        if (err)
                die("unable to read tree entries %s", tree_name);
 
index e3767087bbfdeb065e9a2806df4ba99b922eb2ec..828dc81c426514d822e6bbce172495373f9a92c5 100644 (file)
@@ -3,6 +3,7 @@
 #include "tree-walk.h"
 #include "xdiff-interface.h"
 #include "help.h"
+#include "commit.h"
 #include "commit-reach.h"
 #include "merge-ort.h"
 #include "object-store.h"
@@ -98,7 +99,7 @@ static void *origin(struct merge_list *entry, unsigned long *size)
        return NULL;
 }
 
-static int show_outf(void *priv_, mmbuffer_t *mb, int nbuf)
+static int show_outf(void *priv UNUSED, mmbuffer_t *mb, int nbuf)
 {
        int i;
        for (i = 0; i < nbuf; i++)
@@ -406,6 +407,7 @@ struct merge_tree_options {
 };
 
 static int real_merge(struct merge_tree_options *o,
+                     const char *merge_base,
                      const char *branch1, const char *branch2,
                      const char *prefix)
 {
@@ -432,16 +434,31 @@ static int real_merge(struct merge_tree_options *o,
        opt.branch1 = branch1;
        opt.branch2 = branch2;
 
-       /*
-        * Get the merge bases, in reverse order; see comment above
-        * merge_incore_recursive in merge-ort.h
-        */
-       merge_bases = get_merge_bases(parent1, parent2);
-       if (!merge_bases && !o->allow_unrelated_histories)
-               die(_("refusing to merge unrelated histories"));
-       merge_bases = reverse_commit_list(merge_bases);
+       if (merge_base) {
+               struct commit *base_commit;
+               struct tree *base_tree, *parent1_tree, *parent2_tree;
+
+               base_commit = lookup_commit_reference_by_name(merge_base);
+               if (!base_commit)
+                       die(_("could not lookup commit %s"), merge_base);
+
+               opt.ancestor = merge_base;
+               base_tree = get_commit_tree(base_commit);
+               parent1_tree = get_commit_tree(parent1);
+               parent2_tree = get_commit_tree(parent2);
+               merge_incore_nonrecursive(&opt, base_tree, parent1_tree, parent2_tree, &result);
+       } else {
+               /*
+                * Get the merge bases, in reverse order; see comment above
+                * merge_incore_recursive in merge-ort.h
+                */
+               merge_bases = get_merge_bases(parent1, parent2);
+               if (!merge_bases && !o->allow_unrelated_histories)
+                       die(_("refusing to merge unrelated histories"));
+               merge_bases = reverse_commit_list(merge_bases);
+               merge_incore_recursive(&opt, merge_bases, parent1, parent2, &result);
+       }
 
-       merge_incore_recursive(&opt, merge_bases, parent1, parent2, &result);
        if (result.clean < 0)
                die(_("failure to merge"));
 
@@ -487,6 +504,7 @@ int cmd_merge_tree(int argc, const char **argv, const char *prefix)
        struct merge_tree_options o = { .show_messages = -1 };
        int expected_remaining_argc;
        int original_argc;
+       const char *merge_base = NULL;
 
        const char * const merge_tree_usage[] = {
                N_("git merge-tree [--write-tree] [<options>] <branch1> <branch2>"),
@@ -515,6 +533,10 @@ int cmd_merge_tree(int argc, const char **argv, const char *prefix)
                           &o.use_stdin,
                           N_("perform multiple merges, one per line of input"),
                           PARSE_OPT_NONEG),
+               OPT_STRING(0, "merge-base",
+                          &merge_base,
+                          N_("commit"),
+                          N_("specify a merge-base for the merge")),
                OPT_END()
        };
 
@@ -529,16 +551,35 @@ int cmd_merge_tree(int argc, const char **argv, const char *prefix)
 
                if (o.mode == MODE_TRIVIAL)
                        die(_("--trivial-merge is incompatible with all other options"));
+               if (merge_base)
+                       die(_("--merge-base is incompatible with --stdin"));
                line_termination = '\0';
                while (strbuf_getline_lf(&buf, stdin) != EOF) {
                        struct strbuf **split;
                        int result;
+                       const char *input_merge_base = NULL;
 
                        split = strbuf_split(&buf, ' ');
-                       if (!split[0] || !split[1] || split[2])
+                       if (!split[0] || !split[1])
                                die(_("malformed input line: '%s'."), buf.buf);
                        strbuf_rtrim(split[0]);
-                       result = real_merge(&o, split[0]->buf, split[1]->buf, prefix);
+                       strbuf_rtrim(split[1]);
+
+                       /* parse the merge-base */
+                       if (!strcmp(split[1]->buf, "--")) {
+                               input_merge_base = split[0]->buf;
+                       }
+
+                       if (input_merge_base && split[2] && split[3] && !split[4]) {
+                               strbuf_rtrim(split[2]);
+                               strbuf_rtrim(split[3]);
+                               result = real_merge(&o, input_merge_base, split[2]->buf, split[3]->buf, prefix);
+                       } else if (!input_merge_base && !split[2]) {
+                               result = real_merge(&o, NULL, split[0]->buf, split[1]->buf, prefix);
+                       } else {
+                               die(_("malformed input line: '%s'."), buf.buf);
+                       }
+
                        if (result < 0)
                                die(_("merging cannot continue; got unclean result of %d"), result);
                        strbuf_list_free(split);
@@ -581,7 +622,7 @@ int cmd_merge_tree(int argc, const char **argv, const char *prefix)
 
        /* Do the relevant type of merge */
        if (o.mode == MODE_REAL)
-               return real_merge(&o, argv[0], argv[1], prefix);
+               return real_merge(&o, merge_base, argv[0], argv[1], prefix);
        else
                return trivial_merge(argv[0], argv[1], argv[2]);
 }
index dd474371a256a6170e4f3f1073db2dae2a0f8791..0f093f2a4f221c2ee22e629cd9e069ee201f268a 100644 (file)
@@ -776,7 +776,7 @@ static int try_merge_strategy(const char *strategy, struct commit_list *common,
 }
 
 static void count_diff_files(struct diff_queue_struct *q,
-                            struct diff_options *opt, void *data)
+                            struct diff_options *opt UNUSED, void *data)
 {
        int *count = data;
 
@@ -1789,5 +1789,6 @@ done:
        }
        strbuf_release(&buf);
        free(branch_to_free);
+       discard_index(&the_index);
        return ret;
 }
index 573d0b20b76b5115c99781b28f65ae1b5d7d8dc3..2193f80b897270deda441885fcfe82af4997ad91 100644 (file)
@@ -4149,21 +4149,6 @@ static int option_parse_cruft_expiration(const struct option *opt,
        return 0;
 }
 
-struct po_filter_data {
-       unsigned have_revs:1;
-       struct rev_info revs;
-};
-
-static struct list_objects_filter_options *po_filter_revs_init(void *value)
-{
-       struct po_filter_data *data = value;
-
-       repo_init_revisions(the_repository, &data->revs, NULL);
-       data->have_revs = 1;
-
-       return &data->revs.filter;
-}
-
 int cmd_pack_objects(int argc, const char **argv, const char *prefix)
 {
        int use_internal_rev_list = 0;
@@ -4174,7 +4159,8 @@ int cmd_pack_objects(int argc, const char **argv, const char *prefix)
        int rev_list_index = 0;
        int stdin_packs = 0;
        struct string_list keep_pack_list = STRING_LIST_INIT_NODUP;
-       struct po_filter_data pfd = { .have_revs = 0 };
+       struct list_objects_filter_options filter_options =
+               LIST_OBJECTS_FILTER_INIT;
 
        struct option pack_objects_options[] = {
                OPT_SET_INT('q', "quiet", &progress,
@@ -4265,7 +4251,7 @@ int cmd_pack_objects(int argc, const char **argv, const char *prefix)
                              &write_bitmap_index,
                              N_("write a bitmap index if possible"),
                              WRITE_BITMAP_QUIET, PARSE_OPT_HIDDEN),
-               OPT_PARSE_LIST_OBJECTS_FILTER_INIT(&pfd, po_filter_revs_init),
+               OPT_PARSE_LIST_OBJECTS_FILTER(&filter_options),
                OPT_CALLBACK_F(0, "missing", NULL, N_("action"),
                  N_("handling for missing objects"), PARSE_OPT_NONEG,
                  option_parse_missing_action),
@@ -4385,7 +4371,7 @@ int cmd_pack_objects(int argc, const char **argv, const char *prefix)
        if (!rev_list_all || !rev_list_reflog || !rev_list_index)
                unpack_unreachable_expiration = 0;
 
-       if (pfd.have_revs && pfd.revs.filter.choice) {
+       if (filter_options.choice) {
                if (!pack_to_stdout)
                        die(_("cannot use --filter without --stdout"));
                if (stdin_packs)
@@ -4472,13 +4458,11 @@ int cmd_pack_objects(int argc, const char **argv, const char *prefix)
                read_cruft_objects();
        } else if (!use_internal_rev_list) {
                read_object_list_from_stdin();
-       } else if (pfd.have_revs) {
-               get_object_list(&pfd.revs, rp.nr, rp.v);
-               release_revisions(&pfd.revs);
        } else {
                struct rev_info revs;
 
                repo_init_revisions(the_repository, &revs, NULL);
+               list_objects_filter_copy(&revs.filter, &filter_options);
                get_object_list(&revs, rp.nr, rp.v);
                release_revisions(&revs);
        }
@@ -4513,6 +4497,7 @@ int cmd_pack_objects(int argc, const char **argv, const char *prefix)
                           reuse_packfile_objects);
 
 cleanup:
+       list_objects_filter_release(&filter_options);
        strvec_clear(&rp);
 
        return 0;
index e2a74efb42a795a5139932bd052999d050cab420..aecfae12d3ae0777a57bec2c02dffa7cffde3d8f 100644 (file)
@@ -47,7 +47,7 @@ int cmd_range_diff(int argc, const char **argv, const char *prefix)
 
        repo_diff_setup(the_repository, &diffopt);
 
-       options = parse_options_concat(range_diff_options, diffopt.parseopts);
+       options = add_diff_options(range_diff_options, &diffopt);
        argc = parse_options(argc, argv, prefix, options,
                             builtin_range_diff_usage, PARSE_OPT_KEEP_DASHDASH);
 
index f702f9d47bbfae249497aeeb0e2411ea97fc9f48..3ce75417833bf9f57ca1ddfa17a53c67ef39a9d8 100644 (file)
@@ -114,6 +114,7 @@ int cmd_read_tree(int argc, const char **argv, const char *cmd_prefix)
        int prefix_set = 0;
        struct lock_file lock_file = LOCK_INIT;
        const struct option read_tree_options[] = {
+               OPT__SUPER_PREFIX(&opts.super_prefix),
                OPT_CALLBACK_F(0, "index-output", NULL, N_("file"),
                  N_("write resulting index to <file>"),
                  PARSE_OPT_NONEG, index_output_cb),
index b22768ca5b9fb0ff1b66f0a031bd2494ee1932d7..1481c5b6a5b3d2e66cd02b84b22f6655c59d6f23 100644 (file)
@@ -1322,6 +1322,7 @@ int cmd_rebase(int argc, const char **argv, const char *prefix)
                if (reset_head(the_repository, &ropts) < 0)
                        die(_("could not move back to %s"),
                            oid_to_hex(&options.orig_head->object.oid));
+               strbuf_release(&head_msg);
                remove_branch_state(the_repository, 0);
                ret = finish_rebase(&options);
                goto cleanup;
@@ -1828,10 +1829,13 @@ cleanup:
        strbuf_release(&revisions);
        free(options.reflog_action);
        free(options.head_name);
+       strvec_clear(&options.git_am_opts);
        free(options.gpg_sign_opt);
        free(options.cmd);
        free(options.strategy);
        strbuf_release(&options.git_format_patch_opt);
        free(squash_onto_name);
+       string_list_clear(&exec, 0);
+       string_list_clear(&strategy_options, 0);
        return !!ret;
 }
index 65eb1b8bd22b4119908abe700cfd3a6d143289f1..c1402ad038f2d4e2a60441db11fb1209e7d63c62 100644 (file)
@@ -973,6 +973,7 @@ int cmd_repack(int argc, const char **argv, const char *prefix)
                item = string_list_append(&names, line.buf);
                item->util = populate_pack_exts(item->string);
        }
+       strbuf_release(&line);
        fclose(out);
        ret = finish_command(&cmd);
        if (ret)
@@ -1175,7 +1176,6 @@ int cmd_repack(int argc, const char **argv, const char *prefix)
        string_list_clear(&existing_nonkept_packs, 0);
        string_list_clear(&existing_kept_packs, 0);
        clear_pack_geometry(geometry);
-       strbuf_release(&line);
 
        return 0;
 }
index 8b7392d5b4483ff961e14fc7ae5194d97d188d8b..94ffb8c21ab1e08f1775ad15315dd689723a2db8 100644 (file)
@@ -14,7 +14,7 @@ static const char * const rerere_usage[] = {
        NULL,
 };
 
-static int outf(void *dummy, mmbuffer_t *ptr, int nbuf)
+static int outf(void *dummy UNUSED, mmbuffer_t *ptr, int nbuf)
 {
        int i;
        for (i = 0; i < nbuf; i++)
index 1fa86edb4c92f31a21d838bcbff76e687cd15c23..fea20a9ba0b99bc730f7c7fc7c1000fa57febf3e 100644 (file)
@@ -133,7 +133,8 @@ static void print_new_head_line(struct commit *commit)
 }
 
 static void update_index_from_diff(struct diff_queue_struct *q,
-               struct diff_options *opt, void *data)
+                                  struct diff_options *opt UNUSED,
+                                  void *data)
 {
        int i;
        int intent_to_add = *(int *)data;
@@ -485,5 +486,7 @@ int cmd_reset(int argc, const char **argv, const char *prefix)
        if (!pathspec.nr)
                remove_branch_state(the_repository, 0);
 
+       discard_index(&the_index);
+
        return update_ref_status;
 }
index 1c344d74a7950fa57abd4d6db4c7e692f930e5b7..e67999e5ebc5ef640fe5dcd45dfa1b4126a9c7a8 100644 (file)
@@ -530,6 +530,7 @@ static int cmd_parseopt(int argc, const char **argv, const char *prefix)
        strbuf_addstr(&parsed, " --");
        sq_quote_argv(&parsed, argv);
        puts(parsed.buf);
+       strbuf_release(&parsed);
        return 0;
 }
 
index 6a9b550a61facc27777678b93f61999b5be53dd1..f2d86d2a8f999bbad13f51319e1ba3c0e4561d53 100644 (file)
@@ -221,6 +221,7 @@ static int run_sequencer(int argc, const char **argv, struct replay_opts *opts)
        opts->strategy = xstrdup_or_null(opts->strategy);
        if (!opts->strategy && getenv("GIT_TEST_MERGE_ALGORITHM"))
                opts->strategy = xstrdup(getenv("GIT_TEST_MERGE_ALGORITHM"));
+       free(options);
 
        if (cmd == 'q') {
                int ret = sequencer_remove_state(opts);
@@ -261,6 +262,9 @@ int cmd_cherry_pick(int argc, const char **argv, const char *prefix)
        opts.action = REPLAY_PICK;
        sequencer_init_config(&opts);
        res = run_sequencer(argc, argv, &opts);
+       if (opts.revs)
+               release_revisions(opts.revs);
+       free(opts.revs);
        if (res < 0)
                die(_("cherry-pick failed"));
        return res;
index d4989d4d8637f6054f763ba516366a7eca7f316e..4a4aec0d00e07e5d0260efc5f6195221075ea61d 100644 (file)
@@ -86,7 +86,7 @@ static void submodules_absorb_gitdir_if_needed(void)
                        continue;
 
                if (!submodule_uses_gitfile(name))
-                       absorb_git_dir_into_superproject(name);
+                       absorb_git_dir_into_superproject(name, NULL);
        }
 }
 
index 62e36718d381fef809e381e886aa78e6108439d9..bb0fd861434aae2e9dcdd7b83ec9dfdada38a457 100644 (file)
@@ -1689,8 +1689,10 @@ static int do_push_stash(const struct pathspec *ps, const char *stash_msg, int q
        }
 
 done:
+       strbuf_release(&patch);
        free_stash_info(&info);
        strbuf_release(&stash_msg_buf);
+       strbuf_release(&untracked_files);
        return ret;
 }
 
index 05f2c9bc9858d98ae855b566d6128030e002fa18..450680fc70895ea54ab36b70e24e88191d65a380 100644 (file)
@@ -113,10 +113,9 @@ static char *resolve_relative_url(const char *rel_url, const char *up_path, int
 }
 
 /* the result should be freed by the caller. */
-static char *get_submodule_displaypath(const char *path, const char *prefix)
+static char *get_submodule_displaypath(const char *path, const char *prefix,
+                                      const char *super_prefix)
 {
-       const char *super_prefix = get_super_prefix();
-
        if (prefix && super_prefix) {
                BUG("cannot have prefix '%s' and superprefix '%s'",
                    prefix, super_prefix);
@@ -279,6 +278,7 @@ struct foreach_cb {
        int argc;
        const char **argv;
        const char *prefix;
+       const char *super_prefix;
        int quiet;
        int recursive;
 };
@@ -294,7 +294,8 @@ static void runcommand_in_submodule_cb(const struct cache_entry *list_item,
        struct child_process cp = CHILD_PROCESS_INIT;
        char *displaypath;
 
-       displaypath = get_submodule_displaypath(path, info->prefix);
+       displaypath = get_submodule_displaypath(path, info->prefix,
+                                               info->super_prefix);
 
        sub = submodule_from_path(the_repository, null_oid(), path);
 
@@ -364,10 +365,10 @@ static void runcommand_in_submodule_cb(const struct cache_entry *list_item,
                cpr.dir = path;
                prepare_submodule_repo_env(&cpr.env);
 
-               strvec_pushl(&cpr.args, "--super-prefix", NULL);
-               strvec_pushf(&cpr.args, "%s/", displaypath);
                strvec_pushl(&cpr.args, "submodule--helper", "foreach", "--recursive",
                             NULL);
+               strvec_pushl(&cpr.args, "--super-prefix", NULL);
+               strvec_pushf(&cpr.args, "%s/", displaypath);
 
                if (info->quiet)
                        strvec_push(&cpr.args, "--quiet");
@@ -391,6 +392,7 @@ static int module_foreach(int argc, const char **argv, const char *prefix)
        struct pathspec pathspec = { 0 };
        struct module_list list = MODULE_LIST_INIT;
        struct option module_foreach_options[] = {
+               OPT__SUPER_PREFIX(&info.super_prefix),
                OPT__QUIET(&info.quiet, N_("suppress output of entering each submodule command")),
                OPT_BOOL(0, "recursive", &info.recursive,
                         N_("recurse into nested submodules")),
@@ -435,11 +437,13 @@ static int starts_with_dot_dot_slash(const char *const path)
 
 struct init_cb {
        const char *prefix;
+       const char *super_prefix;
        unsigned int flags;
 };
 #define INIT_CB_INIT { 0 }
 
 static void init_submodule(const char *path, const char *prefix,
+                          const char *super_prefix,
                           unsigned int flags)
 {
        const struct submodule *sub;
@@ -447,7 +451,7 @@ static void init_submodule(const char *path, const char *prefix,
        const char *upd;
        char *url = NULL, *displaypath;
 
-       displaypath = get_submodule_displaypath(path, prefix);
+       displaypath = get_submodule_displaypath(path, prefix, super_prefix);
 
        sub = submodule_from_path(the_repository, null_oid(), path);
 
@@ -523,7 +527,8 @@ static void init_submodule_cb(const struct cache_entry *list_item, void *cb_data
 {
        struct init_cb *info = cb_data;
 
-       init_submodule(list_item->name, info->prefix, info->flags);
+       init_submodule(list_item->name, info->prefix, info->super_prefix,
+                      info->flags);
 }
 
 static int module_init(int argc, const char **argv, const char *prefix)
@@ -570,6 +575,7 @@ cleanup:
 
 struct status_cb {
        const char *prefix;
+       const char *super_prefix;
        unsigned int flags;
 };
 #define STATUS_CB_INIT { 0 }
@@ -608,7 +614,7 @@ static int handle_submodule_head_ref(const char *refname UNUSED,
 
 static void status_submodule(const char *path, const struct object_id *ce_oid,
                             unsigned int ce_flags, const char *prefix,
-                            unsigned int flags)
+                            const char *super_prefix, unsigned int flags)
 {
        char *displaypath;
        struct strvec diff_files_args = STRVEC_INIT;
@@ -624,7 +630,7 @@ static void status_submodule(const char *path, const struct object_id *ce_oid,
                die(_("no submodule mapping found in .gitmodules for path '%s'"),
                      path);
 
-       displaypath = get_submodule_displaypath(path, prefix);
+       displaypath = get_submodule_displaypath(path, prefix, super_prefix);
 
        if ((CE_STAGEMASK & ce_flags) >> CE_STAGESHIFT) {
                print_status(flags, 'U', path, null_oid(), displaypath);
@@ -682,10 +688,10 @@ static void status_submodule(const char *path, const struct object_id *ce_oid,
                cpr.dir = path;
                prepare_submodule_repo_env(&cpr.env);
 
-               strvec_push(&cpr.args, "--super-prefix");
-               strvec_pushf(&cpr.args, "%s/", displaypath);
                strvec_pushl(&cpr.args, "submodule--helper", "status",
                             "--recursive", NULL);
+               strvec_push(&cpr.args, "--super-prefix");
+               strvec_pushf(&cpr.args, "%s/", displaypath);
 
                if (flags & OPT_CACHED)
                        strvec_push(&cpr.args, "--cached");
@@ -709,7 +715,7 @@ static void status_submodule_cb(const struct cache_entry *list_item,
        struct status_cb *info = cb_data;
 
        status_submodule(list_item->name, &list_item->oid, list_item->ce_flags,
-                        info->prefix, info->flags);
+                        info->prefix, info->super_prefix, info->flags);
 }
 
 static int module_status(int argc, const char **argv, const char *prefix)
@@ -719,6 +725,7 @@ static int module_status(int argc, const char **argv, const char *prefix)
        struct module_list list = MODULE_LIST_INIT;
        int quiet = 0;
        struct option module_status_options[] = {
+               OPT__SUPER_PREFIX(&info.super_prefix),
                OPT__QUIET(&quiet, N_("suppress submodule status output")),
                OPT_BIT(0, "cached", &info.flags, N_("use commit stored in the index instead of the one stored in the submodule HEAD"), OPT_CACHED),
                OPT_BIT(0, "recursive", &info.flags, N_("recurse into nested submodules"), OPT_RECURSIVE),
@@ -787,6 +794,7 @@ struct summary_cb {
        int argc;
        const char **argv;
        const char *prefix;
+       const char *super_prefix;
        unsigned int cached: 1;
        unsigned int for_status: 1;
        unsigned int files: 1;
@@ -948,7 +956,8 @@ static void generate_submodule_summary(struct summary_cb *info,
                dst_abbrev = xstrndup(oid_to_hex(&p->oid_dst), 7);
        }
 
-       displaypath = get_submodule_displaypath(p->sm_path, info->prefix);
+       displaypath = get_submodule_displaypath(p->sm_path, info->prefix,
+                                               info->super_prefix);
 
        if (!missing_src && !missing_dst) {
                struct child_process cp_rev_list = CHILD_PROCESS_INIT;
@@ -1043,7 +1052,7 @@ static void prepare_submodule_summary(struct summary_cb *info,
 }
 
 static void submodule_summary_callback(struct diff_queue_struct *q,
-                                      struct diff_options *options,
+                                      struct diff_options *options UNUSED,
                                       void *data)
 {
        int i;
@@ -1203,12 +1212,13 @@ static int module_summary(int argc, const char **argv, const char *prefix)
 
 struct sync_cb {
        const char *prefix;
+       const char *super_prefix;
        unsigned int flags;
 };
 #define SYNC_CB_INIT { 0 }
 
 static void sync_submodule(const char *path, const char *prefix,
-                          unsigned int flags)
+                          const char *super_prefix, unsigned int flags)
 {
        const struct submodule *sub;
        char *remote_key = NULL;
@@ -1239,7 +1249,7 @@ static void sync_submodule(const char *path, const char *prefix,
                super_config_url = xstrdup("");
        }
 
-       displaypath = get_submodule_displaypath(path, prefix);
+       displaypath = get_submodule_displaypath(path, prefix, super_prefix);
 
        if (!(flags & OPT_QUIET))
                printf(_("Synchronizing submodule url for '%s'\n"),
@@ -1276,10 +1286,11 @@ static void sync_submodule(const char *path, const char *prefix,
                cpr.dir = path;
                prepare_submodule_repo_env(&cpr.env);
 
-               strvec_push(&cpr.args, "--super-prefix");
-               strvec_pushf(&cpr.args, "%s/", displaypath);
                strvec_pushl(&cpr.args, "submodule--helper", "sync",
                             "--recursive", NULL);
+               strvec_push(&cpr.args, "--super-prefix");
+               strvec_pushf(&cpr.args, "%s/", displaypath);
+
 
                if (flags & OPT_QUIET)
                        strvec_push(&cpr.args, "--quiet");
@@ -1302,7 +1313,8 @@ static void sync_submodule_cb(const struct cache_entry *list_item, void *cb_data
 {
        struct sync_cb *info = cb_data;
 
-       sync_submodule(list_item->name, info->prefix, info->flags);
+       sync_submodule(list_item->name, info->prefix, info->super_prefix,
+                      info->flags);
 }
 
 static int module_sync(int argc, const char **argv, const char *prefix)
@@ -1313,6 +1325,7 @@ static int module_sync(int argc, const char **argv, const char *prefix)
        int quiet = 0;
        int recursive = 0;
        struct option module_sync_options[] = {
+               OPT__SUPER_PREFIX(&info.super_prefix),
                OPT__QUIET(&quiet, N_("suppress output of synchronizing submodule url")),
                OPT_BOOL(0, "recursive", &recursive,
                        N_("recurse into nested submodules")),
@@ -1365,7 +1378,7 @@ static void deinit_submodule(const char *path, const char *prefix,
        if (!sub || !sub->name)
                goto cleanup;
 
-       displaypath = get_submodule_displaypath(path, prefix);
+       displaypath = get_submodule_displaypath(path, prefix, NULL);
 
        /* remove the submodule work tree (unless the user already did it) */
        if (is_directory(path)) {
@@ -1379,7 +1392,7 @@ static void deinit_submodule(const char *path, const char *prefix,
                                          ".git file by using absorbgitdirs."),
                                        displaypath);
 
-                       absorb_git_dir_into_superproject(path);
+                       absorb_git_dir_into_superproject(path, NULL);
 
                }
 
@@ -1883,6 +1896,7 @@ static void submodule_update_clone_release(struct submodule_update_clone *suc)
 
 struct update_data {
        const char *prefix;
+       const char *super_prefix;
        char *displaypath;
        enum submodule_update_type update_default;
        struct object_id suboid;
@@ -1958,7 +1972,8 @@ static int prepare_to_clone_next_submodule(const struct cache_entry *ce,
        enum submodule_update_type update_type;
        char *key;
        const struct update_data *ud = suc->update_data;
-       char *displaypath = get_submodule_displaypath(ce->name, ud->prefix);
+       char *displaypath = get_submodule_displaypath(ce->name, ud->prefix,
+                                                     ud->super_prefix);
        struct strbuf sb = STRBUF_INIT;
        int needs_cloning = 0;
        int need_free_url = 0;
@@ -2438,11 +2453,11 @@ static void update_data_to_args(const struct update_data *update_data,
 {
        enum submodule_update_type update_type = update_data->update_default;
 
+       strvec_pushl(args, "submodule--helper", "update", "--recursive", NULL);
        if (update_data->displaypath) {
                strvec_push(args, "--super-prefix");
                strvec_pushf(args, "%s/", update_data->displaypath);
        }
-       strvec_pushl(args, "submodule--helper", "update", "--recursive", NULL);
        strvec_pushf(args, "--jobs=%d", update_data->max_jobs);
        if (update_data->quiet)
                strvec_push(args, "--quiet");
@@ -2608,7 +2623,8 @@ static int update_submodules(struct update_data *update_data)
                        goto fail;
 
                update_data->displaypath = get_submodule_displaypath(
-                       update_data->sm_path, update_data->prefix);
+                       update_data->sm_path, update_data->prefix,
+                       update_data->super_prefix);
                code = update_submodule(update_data);
                FREE_AND_NULL(update_data->displaypath);
 fail:
@@ -2634,6 +2650,7 @@ static int module_update(int argc, const char **argv, const char *prefix)
                LIST_OBJECTS_FILTER_INIT;
        int ret;
        struct option module_update_options[] = {
+               OPT__SUPER_PREFIX(&opt.super_prefix),
                OPT__FORCE(&opt.force, N_("force checkout updates"), 0),
                OPT_BOOL(0, "init", &opt.init,
                         N_("initialize uninitialized submodules before update")),
@@ -2730,6 +2747,7 @@ static int module_update(int argc, const char **argv, const char *prefix)
                        module_list_active(&list);
 
                info.prefix = opt.prefix;
+               info.super_prefix = opt.super_prefix;
                if (opt.quiet)
                        info.flags |= OPT_QUIET;
 
@@ -2828,7 +2846,9 @@ static int absorb_git_dirs(int argc, const char **argv, const char *prefix)
        int i;
        struct pathspec pathspec = { 0 };
        struct module_list list = MODULE_LIST_INIT;
+       const char *super_prefix = NULL;
        struct option embed_gitdir_options[] = {
+               OPT__SUPER_PREFIX(&super_prefix),
                OPT_END()
        };
        const char *const git_submodule_helper_usage[] = {
@@ -2844,7 +2864,8 @@ static int absorb_git_dirs(int argc, const char **argv, const char *prefix)
                goto cleanup;
 
        for (i = 0; i < list.nr; i++)
-               absorb_git_dir_into_superproject(list.entries[i]->name);
+               absorb_git_dir_into_superproject(list.entries[i]->name,
+                                                super_prefix);
 
        ret = 0;
 cleanup:
@@ -2876,7 +2897,7 @@ static int module_set_url(int argc, const char **argv, const char *prefix)
        config_name = xstrfmt("submodule.%s.url", path);
 
        config_set_in_gitmodules_file_gently(config_name, newurl);
-       sync_submodule(path, prefix, quiet ? OPT_QUIET : 0);
+       sync_submodule(path, prefix, NULL, quiet ? OPT_QUIET : 0);
 
        free(config_name);
 
@@ -3353,8 +3374,6 @@ cleanup:
 
 int cmd_submodule__helper(int argc, const char **argv, const char *prefix)
 {
-       const char *cmd = argv[0];
-       const char *subcmd;
        parse_opt_subcommand_fn *fn = NULL;
        const char *const usage[] = {
                N_("git submodule--helper <command>"),
@@ -3378,18 +3397,6 @@ int cmd_submodule__helper(int argc, const char **argv, const char *prefix)
                OPT_END()
        };
        argc = parse_options(argc, argv, prefix, options, usage, 0);
-       subcmd = argv[0];
-
-       if (strcmp(subcmd, "clone") && strcmp(subcmd, "update") &&
-           strcmp(subcmd, "foreach") && strcmp(subcmd, "status") &&
-           strcmp(subcmd, "sync") && strcmp(subcmd, "absorbgitdirs") &&
-           get_super_prefix())
-               /*
-                * xstrfmt() rather than "%s %s" to keep the translated
-                * string identical to git.c's.
-                */
-               die(_("%s doesn't support --super-prefix"),
-                   xstrfmt("'%s %s'", cmd, subcmd));
 
        return fn(argc, argv, prefix);
 }
index 9e8119dd35422a93c63dbe23f840bda5ec69c5e5..88de32b7d7e66e3d7e1c57775b5f0a1b0b045175 100644 (file)
@@ -19,6 +19,7 @@ static char *create_temp_file(struct object_id *oid)
        if (write_in_full(fd, buf, size) < 0)
                die_errno("unable to write temp-file");
        close(fd);
+       free(buf);
        return path;
 }
 
index 491db2742926dbd6fd08f681817821addcfafab7..a80c1df86fd532bb6353b990381bfb050ac5e688 100644 (file)
@@ -11,12 +11,12 @@ static const char var_usage[] = "git var (-l | <variable>)";
 
 static const char *editor(int flag)
 {
-       const char *pgm = git_editor();
-
-       if (!pgm && flag & IDENT_STRICT)
-               die("Terminal is dumb, but EDITOR unset");
+       return git_editor();
+}
 
-       return pgm;
+static const char *sequence_editor(int flag)
+{
+       return git_sequence_editor();
 }
 
 static const char *pager(int flag)
@@ -41,6 +41,7 @@ static struct git_var git_vars[] = {
        { "GIT_COMMITTER_IDENT", git_committer_info },
        { "GIT_AUTHOR_IDENT",   git_author_info },
        { "GIT_EDITOR", editor },
+       { "GIT_SEQUENCE_EDITOR", sequence_editor },
        { "GIT_PAGER", pager },
        { "GIT_DEFAULT_BRANCH", default_branch },
        { "", NULL },
@@ -56,18 +57,15 @@ static void list_vars(void)
                        printf("%s=%s\n", ptr->name, val);
 }
 
-static const char *read_var(const char *var)
+static const struct git_var *get_git_var(const char *var)
 {
        struct git_var *ptr;
-       const char *val;
-       val = NULL;
        for (ptr = git_vars; ptr->read; ptr++) {
                if (strcmp(var, ptr->name) == 0) {
-                       val = ptr->read(IDENT_STRICT);
-                       break;
+                       return ptr;
                }
        }
-       return val;
+       return NULL;
 }
 
 static int show_config(const char *var, const char *value, void *cb)
@@ -81,7 +79,9 @@ static int show_config(const char *var, const char *value, void *cb)
 
 int cmd_var(int argc, const char **argv, const char *prefix)
 {
-       const char *val = NULL;
+       const struct git_var *git_var;
+       const char *val;
+
        if (argc != 2)
                usage(var_usage);
 
@@ -91,10 +91,15 @@ int cmd_var(int argc, const char **argv, const char *prefix)
                return 0;
        }
        git_config(git_default_config, NULL);
-       val = read_var(argv[1]);
-       if (!val)
+
+       git_var = get_git_var(argv[1]);
+       if (!git_var)
                usage(var_usage);
 
+       val = git_var->read(IDENT_STRICT);
+       if (!val)
+               return 1;
+
        printf("%s\n", val);
 
        return 0;
index 4a24d53be15c3a74d6632c68ccacb813c8aa989b..591d659faea1684b19a5371b773c6d8493f8d796 100644 (file)
@@ -629,6 +629,7 @@ static int add(int ac, const char **av, const char *prefix)
                         N_("try to match the new branch name with a remote-tracking branch")),
                OPT_END()
        };
+       int ret;
 
        memset(&opts, 0, sizeof(opts));
        opts.checkout = 1;
@@ -705,9 +706,9 @@ static int add(int ac, const char **av, const char *prefix)
                die(_("--[no-]track can only be used if a new branch is created"));
        }
 
-       UNLEAK(path);
-       UNLEAK(opts);
-       return add_worktree(path, branch, &opts);
+       ret = add_worktree(path, branch, &opts);
+       free(path);
+       return ret;
 }
 
 static void show_worktree_porcelain(struct worktree *wt, int line_terminator)
index 79a914f961b134cccc6029aa403d7a626841c81b..36268dda172511250cdda7a24db8cfdb00548f6f 100644 (file)
@@ -7,6 +7,7 @@
 #include "hashmap.h"
 #include "pkt-line.h"
 #include "config.h"
+#include "remote.h"
 
 static int compare_bundles(const void *hashmap_cmp_fn_data,
                           const struct hashmap_entry *he1,
@@ -49,6 +50,7 @@ void clear_bundle_list(struct bundle_list *list)
 
        for_all_bundles_in_list(list, clear_remote_bundle_info, NULL);
        hashmap_clear_and_free(&list->bundles, struct remote_bundle_info, ent);
+       free(list->baseURI);
 }
 
 int for_all_bundles_in_list(struct bundle_list *list,
@@ -163,7 +165,7 @@ static int bundle_list_update(const char *key, const char *value,
        if (!strcmp(subkey, "uri")) {
                if (bundle->uri)
                        return -1;
-               bundle->uri = xstrdup(value);
+               bundle->uri = relative_url(list->baseURI, value, NULL);
                return 0;
        }
 
@@ -190,6 +192,18 @@ int bundle_uri_parse_config_format(const char *uri,
                .error_action = CONFIG_ERROR_ERROR,
        };
 
+       if (!list->baseURI) {
+               struct strbuf baseURI = STRBUF_INIT;
+               strbuf_addstr(&baseURI, uri);
+
+               /*
+                * If the URI does not end with a trailing slash, then
+                * remove the filename portion of the path. This is
+                * important for relative URIs.
+                */
+               strbuf_strip_file_from_path(&baseURI);
+               list->baseURI = strbuf_detach(&baseURI, NULL);
+       }
        result = git_config_from_file_with_options(config_to_bundle_list,
                                                   filename, list,
                                                   &opts);
@@ -563,6 +577,77 @@ cleanup:
        return result;
 }
 
+int fetch_bundle_list(struct repository *r, struct bundle_list *list)
+{
+       int result;
+       struct bundle_list global_list;
+
+       init_bundle_list(&global_list);
+
+       /* If a bundle is added to this global list, then it is required. */
+       global_list.mode = BUNDLE_MODE_ALL;
+
+       if ((result = download_bundle_list(r, list, &global_list, 0)))
+               goto cleanup;
+
+       result = unbundle_all_bundles(r, &global_list);
+
+cleanup:
+       for_all_bundles_in_list(&global_list, unlink_bundle, NULL);
+       clear_bundle_list(&global_list);
+       return result;
+}
+
+/**
+ * API for serve.c.
+ */
+
+int bundle_uri_advertise(struct repository *r, struct strbuf *value UNUSED)
+{
+       static int advertise_bundle_uri = -1;
+
+       if (advertise_bundle_uri != -1)
+               goto cached;
+
+       advertise_bundle_uri = 0;
+       repo_config_get_maybe_bool(r, "uploadpack.advertisebundleuris", &advertise_bundle_uri);
+
+cached:
+       return advertise_bundle_uri;
+}
+
+static int config_to_packet_line(const char *key, const char *value, void *data)
+{
+       struct packet_reader *writer = data;
+
+       if (!strncmp(key, "bundle.", 7))
+               packet_write_fmt(writer->fd, "%s=%s", key, value);
+
+       return 0;
+}
+
+int bundle_uri_command(struct repository *r,
+                      struct packet_reader *request)
+{
+       struct packet_writer writer;
+       packet_writer_init(&writer, 1);
+
+       while (packet_reader_read(request) == PACKET_READ_NORMAL)
+               die(_("bundle-uri: unexpected argument: '%s'"), request->line);
+       if (request->status != PACKET_READ_FLUSH)
+               die(_("bundle-uri: expected flush after arguments"));
+
+       /*
+        * Read all "bundle.*" config lines to the client as key=value
+        * packet lines.
+        */
+       git_config(config_to_packet_line, &writer);
+
+       packet_writer_flush(&writer);
+
+       return 0;
+}
+
 /**
  * General API for {transport,connect}.c etc.
  */
index 4dbc269823cc3a6b4a523b199a8a6d8a83219578..d5e89f1671caa3717b35242f20b8993cea32d14f 100644 (file)
@@ -4,6 +4,7 @@
 #include "hashmap.h"
 #include "strbuf.h"
 
+struct packet_reader;
 struct repository;
 struct string_list;
 
@@ -60,6 +61,20 @@ struct bundle_list {
        int version;
        enum bundle_list_mode mode;
        struct hashmap bundles;
+
+       /**
+        * The baseURI of a bundle_list is the URI that provided the list.
+        *
+        * In the case of the 'bundle-uri' protocol v2 command, the base
+        * URI is the URI of the Git remote.
+        *
+        * Otherwise, the bundle list was downloaded over HTTP from some
+        * known URI. 'baseURI' is set to that value.
+        *
+        * The baseURI is used as the base for any relative URIs
+        * advertised by the bundle list at that location.
+        */
+       char *baseURI;
 };
 
 void init_bundle_list(struct bundle_list *list);
@@ -92,6 +107,26 @@ int bundle_uri_parse_config_format(const char *uri,
  */
 int fetch_bundle_uri(struct repository *r, const char *uri);
 
+/**
+ * Given a bundle list that was already advertised (likely by the
+ * bundle-uri protocol v2 verb) at the given uri, fetch and unbundle the
+ * bundles according to the bundle strategy of that list.
+ *
+ * It is expected that the given 'list' is initialized, including its
+ * 'baseURI' value.
+ *
+ * Returns non-zero if there was an error trying to download the list
+ * or any of its advertised bundles.
+ */
+int fetch_bundle_list(struct repository *r,
+                     struct bundle_list *list);
+
+/**
+ * API for serve.c.
+ */
+int bundle_uri_advertise(struct repository *r, struct strbuf *value);
+int bundle_uri_command(struct repository *r, struct packet_reader *request);
+
 /**
  * General API for {transport,connect}.c etc.
  */
diff --git a/cache.h b/cache.h
index 07d40b0964b12b9f4875c20f26540585ffdc792e..21ed9633b2a78193c02fda5cd91960a38fcbdf2e 100644 (file)
--- a/cache.h
+++ b/cache.h
@@ -480,7 +480,6 @@ static inline enum object_type object_type(unsigned int mode)
 #define GIT_NAMESPACE_ENVIRONMENT "GIT_NAMESPACE"
 #define GIT_WORK_TREE_ENVIRONMENT "GIT_WORK_TREE"
 #define GIT_PREFIX_ENVIRONMENT "GIT_PREFIX"
-#define GIT_SUPER_PREFIX_ENVIRONMENT "GIT_INTERNAL_SUPER_PREFIX"
 #define DEFAULT_GIT_DIR_ENVIRONMENT ".git"
 #define DB_ENVIRONMENT "GIT_OBJECT_DIRECTORY"
 #define INDEX_ENVIRONMENT "GIT_INDEX_FILE"
@@ -566,7 +565,6 @@ int get_common_dir_noenv(struct strbuf *sb, const char *gitdir);
 int get_common_dir(struct strbuf *sb, const char *gitdir);
 const char *get_git_namespace(void);
 const char *strip_namespace(const char *namespaced_ref);
-const char *get_super_prefix(void);
 const char *get_git_work_tree(void);
 
 /*
@@ -1865,7 +1863,7 @@ unsigned ws_check(const char *line, int len, unsigned ws_rule);
 void ws_check_emit(const char *line, int len, unsigned ws_rule, FILE *stream, const char *set, const char *reset, const char *ws);
 char *whitespace_error_string(unsigned ws);
 void ws_fix_copy(struct strbuf *, const char *, int, unsigned, int *);
-int ws_blank_line(const char *line, int len, unsigned ws_rule);
+int ws_blank_line(const char *line, int len);
 #define ws_tab_width(rule)     ((rule) & WS_TAB_WIDTH_MASK)
 
 /* ls-files */
index 706e3ba7e938d83091d1557ed20f7356e557f61d..db7105e8a8dcdff1432d5ec972e3262e4f787c16 100755 (executable)
--- a/ci/lib.sh
+++ b/ci/lib.sh
@@ -258,8 +258,7 @@ macos-*)
                MAKEFLAGS="$MAKEFLAGS PYTHON_PATH=$(which python3)"
        else
                MAKEFLAGS="$MAKEFLAGS PYTHON_PATH=$(which python2)"
-               MAKEFLAGS="$MAKEFLAGS NO_APPLE_COMMON_CRYPTO=NoThanks"
-               MAKEFLAGS="$MAKEFLAGS NO_OPENSSL=NoThanks"
+               MAKEFLAGS="$MAKEFLAGS APPLE_COMMON_CRYPTO_SHA1=Yes"
        fi
        ;;
 esac
index b0ece9548082c17510b6abd17427434bbf66cae9..1a39b5dde0946df0c1e36c7b1f4c48552885985a 100644 (file)
@@ -372,7 +372,7 @@ struct combine_diff_state {
 static void consume_hunk(void *state_,
                         long ob, long on,
                         long nb, long nn,
-                        const char *funcline, long funclen)
+                        const char *func UNUSED, long funclen UNUSED)
 {
        struct combine_diff_state *state = state_;
 
index 572301b80a2a3723daea5e16bc6d77977c95543c..14538a811ae0e000f140caf2f4738c89116684a5 100644 (file)
--- a/commit.c
+++ b/commit.c
@@ -508,6 +508,17 @@ int repo_parse_commit_internal(struct repository *r,
        enum object_type type;
        void *buffer;
        unsigned long size;
+       struct object_info oi = {
+               .typep = &type,
+               .sizep = &size,
+               .contentp = &buffer,
+       };
+       /*
+        * Git does not support partial clones that exclude commits, so set
+        * OBJECT_INFO_SKIP_FETCH_OBJECT to fail fast when an object is missing.
+        */
+       int flags = OBJECT_INFO_LOOKUP_REPLACE | OBJECT_INFO_SKIP_FETCH_OBJECT |
+               OBJECT_INFO_DIE_IF_CORRUPT;
        int ret;
 
        if (!item)
@@ -516,8 +527,8 @@ int repo_parse_commit_internal(struct repository *r,
                return 0;
        if (use_commit_graph && parse_commit_in_graph(r, item))
                return 0;
-       buffer = repo_read_object_file(r, &item->object.oid, &type, &size);
-       if (!buffer)
+
+       if (oid_object_info_extended(r, &item->object.oid, &oi, flags) < 0)
                return quiet_on_missing ? -1 :
                        error("Could not read %s",
                             oid_to_hex(&item->object.oid));
@@ -701,8 +712,10 @@ static void clear_commit_marks_1(struct commit_list **plist,
                if (!parents)
                        return;
 
-               while ((parents = parents->next))
-                       commit_list_insert(parents->item, plist);
+               while ((parents = parents->next)) {
+                       if (parents->item->object.flags & mark)
+                               commit_list_insert(parents->item, plist);
+               }
 
                commit = commit->parents->item;
        }
index 1c75c3d48e7d7c3d960e4cf7ff3bf672f4fe1432..3496e29b3a1f1b8ea04224b63b65539edda602df 100644 (file)
@@ -80,9 +80,7 @@ void CFRunLoopRun(void);
 void CFRunLoopStop(CFRunLoopRef run_loop);
 CFRunLoopRef CFRunLoopGetCurrent(void);
 extern CFStringRef kCFRunLoopDefaultMode;
-void FSEventStreamScheduleWithRunLoop(FSEventStreamRef stream,
-                                     CFRunLoopRef run_loop,
-                                     CFStringRef run_loop_mode);
+void FSEventStreamSetDispatchQueue(FSEventStreamRef stream, dispatch_queue_t q);
 unsigned char FSEventStreamStart(FSEventStreamRef stream);
 void FSEventStreamStop(FSEventStreamRef stream);
 void FSEventStreamInvalidate(FSEventStreamRef stream);
index cc9af1e3cb3b3afca7d519147c4ae252b9f5832a..97a55a6f0a42d8c0ba8b1d14e077e60e7fdd55df 100644 (file)
@@ -1,4 +1,5 @@
 #ifndef __clang__
+#include <dispatch/dispatch.h>
 #include "fsm-darwin-gcc.h"
 #else
 #include <CoreFoundation/CoreFoundation.h>
@@ -38,7 +39,9 @@ struct fsm_listen_data
 
        FSEventStreamRef stream;
 
-       CFRunLoopRef rl;
+       dispatch_queue_t dq;
+       pthread_cond_t dq_finished;
+       pthread_mutex_t dq_lock;
 
        enum shutdown_style {
                SHUTDOWN_EVENT = 0,
@@ -379,8 +382,11 @@ force_shutdown:
        fsmonitor_batch__free_list(batch);
        string_list_clear(&cookie_list, 0);
 
+       pthread_mutex_lock(&data->dq_lock);
        data->shutdown_style = FORCE_SHUTDOWN;
-       CFRunLoopStop(data->rl);
+       pthread_cond_broadcast(&data->dq_finished);
+       pthread_mutex_unlock(&data->dq_lock);
+
        strbuf_release(&tmp);
        return;
 }
@@ -441,10 +447,6 @@ int fsm_listen__ctor(struct fsmonitor_daemon_state *state)
        if (!data->stream)
                goto failed;
 
-       /*
-        * `data->rl` needs to be set inside the listener thread.
-        */
-
        return 0;
 
 failed:
@@ -471,6 +473,11 @@ void fsm_listen__dtor(struct fsmonitor_daemon_state *state)
                FSEventStreamRelease(data->stream);
        }
 
+       if (data->dq)
+               dispatch_release(data->dq);
+       pthread_cond_destroy(&data->dq_finished);
+       pthread_mutex_destroy(&data->dq_lock);
+
        FREE_AND_NULL(state->listen_data);
 }
 
@@ -479,9 +486,11 @@ void fsm_listen__stop_async(struct fsmonitor_daemon_state *state)
        struct fsm_listen_data *data;
 
        data = state->listen_data;
-       data->shutdown_style = SHUTDOWN_EVENT;
 
-       CFRunLoopStop(data->rl);
+       pthread_mutex_lock(&data->dq_lock);
+       data->shutdown_style = SHUTDOWN_EVENT;
+       pthread_cond_broadcast(&data->dq_finished);
+       pthread_mutex_unlock(&data->dq_lock);
 }
 
 void fsm_listen__loop(struct fsmonitor_daemon_state *state)
@@ -490,9 +499,11 @@ void fsm_listen__loop(struct fsmonitor_daemon_state *state)
 
        data = state->listen_data;
 
-       data->rl = CFRunLoopGetCurrent();
+       pthread_mutex_init(&data->dq_lock, NULL);
+       pthread_cond_init(&data->dq_finished, NULL);
+       data->dq = dispatch_queue_create("FSMonitor", NULL);
 
-       FSEventStreamScheduleWithRunLoop(data->stream, data->rl, kCFRunLoopDefaultMode);
+       FSEventStreamSetDispatchQueue(data->stream, data->dq);
        data->stream_scheduled = 1;
 
        if (!FSEventStreamStart(data->stream)) {
@@ -501,7 +512,9 @@ void fsm_listen__loop(struct fsmonitor_daemon_state *state)
        }
        data->stream_started = 1;
 
-       CFRunLoopRun();
+       pthread_mutex_lock(&data->dq_lock);
+       pthread_cond_wait(&data->dq_finished, &data->dq_lock);
+       pthread_mutex_unlock(&data->dq_lock);
 
        switch (data->shutdown_style) {
        case FORCE_ERROR_STOP:
index d614f156df1db31248e3a1de51273ffefdd671ff..af397e68a1dc0de834df7ee6ab98125622c29de3 100644 (file)
@@ -2752,7 +2752,7 @@ int is_path_owned_by_current_sid(const char *path, struct strbuf *report)
                        /*
                         * On FAT32 volumes, ownership is not actually recorded.
                         */
-                       strbuf_addf(report, "'%s' is on a file system that does"
+                       strbuf_addf(report, "'%s' is on a file system that does "
                                    "not record ownership\n", path);
                } else if (report) {
                        LPSTR str1, str2, to_free1 = NULL, to_free2 = NULL;
index 737983d00bae9190e6a328ccf9a2b1a4914fa109..cc3221cb2c8a84a454784c7fa0dc287dbad83983 100644 (file)
@@ -66,7 +66,7 @@ pthread_t pthread_self(void);
 
 static inline void NORETURN pthread_exit(void *ret)
 {
-       ExitThread((DWORD)(intptr_t)ret);
+       _endthreadex((unsigned)(uintptr_t)ret);
 }
 
 typedef DWORD pthread_key_t;
index 5ea53deda23527ded642449fa9d803cc12e34477..63e59641c0d4d5c60fc4bee2761e6acd471535a2 100644 (file)
--- a/connect.c
+++ b/connect.c
@@ -15,6 +15,7 @@
 #include "version.h"
 #include "protocol.h"
 #include "alias.h"
+#include "bundle-uri.h"
 
 static char *server_capabilities_v1;
 static struct strvec server_capabilities_v2 = STRVEC_INIT;
@@ -66,7 +67,7 @@ static NORETURN void die_initial_contact(int unexpected)
 }
 
 /* Checks if the server supports the capability 'c' */
-int server_supports_v2(const char *c, int die_on_error)
+int server_supports_v2(const char *c)
 {
        int i;
 
@@ -76,11 +77,13 @@ int server_supports_v2(const char *c, int die_on_error)
                    (!*out || *out == '='))
                        return 1;
        }
+       return 0;
+}
 
-       if (die_on_error)
+void ensure_server_supports_v2(const char *c)
+{
+       if (!server_supports_v2(c))
                die(_("server doesn't support '%s'"), c);
-
-       return 0;
 }
 
 int server_feature_v2(const char *c, const char **v)
@@ -477,7 +480,7 @@ static void send_capabilities(int fd_out, struct packet_reader *reader)
 {
        const char *hash_name;
 
-       if (server_supports_v2("agent", 0))
+       if (server_supports_v2("agent"))
                packet_write_fmt(fd_out, "agent=%s", git_user_agent_sanitized());
 
        if (server_feature_v2("object-format", &hash_name)) {
@@ -491,6 +494,49 @@ static void send_capabilities(int fd_out, struct packet_reader *reader)
        }
 }
 
+int get_remote_bundle_uri(int fd_out, struct packet_reader *reader,
+                         struct bundle_list *bundles, int stateless_rpc)
+{
+       int line_nr = 1;
+
+       /* Assert bundle-uri support */
+       ensure_server_supports_v2("bundle-uri");
+
+       /* (Re-)send capabilities */
+       send_capabilities(fd_out, reader);
+
+       /* Send command */
+       packet_write_fmt(fd_out, "command=bundle-uri\n");
+       packet_delim(fd_out);
+
+       packet_flush(fd_out);
+
+       /* Process response from server */
+       while (packet_reader_read(reader) == PACKET_READ_NORMAL) {
+               const char *line = reader->line;
+               line_nr++;
+
+               if (!bundle_uri_parse_line(bundles, line))
+                       continue;
+
+               return error(_("error on bundle-uri response line %d: %s"),
+                            line_nr, line);
+       }
+
+       if (reader->status != PACKET_READ_FLUSH)
+               return error(_("expected flush after bundle-uri listing"));
+
+       /*
+        * Might die(), but obscure enough that that's OK, e.g. in
+        * serve.c we'll call BUG() on its equivalent (the
+        * PACKET_READ_RESPONSE_END check).
+        */
+       check_stateless_delimiter(stateless_rpc, reader,
+                                 _("expected response end packet after ref listing"));
+
+       return 0;
+}
+
 struct ref **get_remote_refs(int fd_out, struct packet_reader *reader,
                             struct ref **list, int for_push,
                             struct transport_ls_refs_options *transport_options,
@@ -504,17 +550,18 @@ struct ref **get_remote_refs(int fd_out, struct packet_reader *reader,
                &transport_options->unborn_head_target : NULL;
        *list = NULL;
 
-       if (server_supports_v2("ls-refs", 1))
-               packet_write_fmt(fd_out, "command=ls-refs\n");
+       ensure_server_supports_v2("ls-refs");
+       packet_write_fmt(fd_out, "command=ls-refs\n");
 
        /* Send capabilities */
        send_capabilities(fd_out, reader);
 
-       if (server_options && server_options->nr &&
-           server_supports_v2("server-option", 1))
+       if (server_options && server_options->nr) {
+               ensure_server_supports_v2("server-option");
                for (i = 0; i < server_options->nr; i++)
                        packet_write_fmt(fd_out, "server-option=%s",
                                         server_options->items[i].string);
+       }
 
        packet_delim(fd_out);
        /* When pushing we don't want to request the peeled tags */
index c53586e929e8b6f11194d9aeb9934e0fbecd5983..b26f7de7841838e0d6e88bb81169191f579fb56d 100644 (file)
--- a/connect.h
+++ b/connect.h
@@ -20,7 +20,8 @@ enum protocol_version discover_version(struct packet_reader *reader);
 
 int server_supports_hash(const char *desired, int *feature_supported);
 const char *parse_feature_value(const char *feature_list, const char *feature, int *lenp, int *offset);
-int server_supports_v2(const char *c, int die_on_error);
+int server_supports_v2(const char *c);
+void ensure_server_supports_v2(const char *c);
 int server_feature_v2(const char *c, const char **v);
 int server_supports_feature(const char *c, const char *feature,
                            int die_on_error);
index 4f6388eed76f77ebacb0a4b1f68075fc69c50565..b90fd61790cef00d4141bb2eb30f33522e6e8c47 100644 (file)
@@ -85,6 +85,7 @@ int check_connected(oid_iterate_fn fn, void *cb_data,
 promisor_pack_found:
                        ;
                } while ((oid = fn(cb_data)) != NULL);
+               free(new_pack);
                return 0;
        }
 
@@ -121,8 +122,10 @@ no_promisor_pack_found:
        else
                rev_list.no_stderr = opt->quiet;
 
-       if (start_command(&rev_list))
+       if (start_command(&rev_list)) {
+               free(new_pack);
                return error(_("Could not run 'git rev-list'"));
+       }
 
        sigchain_push(SIGPIPE, SIG_IGN);
 
@@ -154,5 +157,6 @@ no_promisor_pack_found:
                err = error_errno(_("failed to close rev-list's stdin"));
 
        sigchain_pop(SIGPIPE);
+       free(new_pack);
        return finish_command(&rev_list) || err;
 }
index ba5c395d2d804f522d5730f7bca2fa6e17d63a96..dc95c34cc853557efd2a59a33825f834e8d934cf 100644 (file)
 #
 #     When set to "1" suggest all options, including options which are
 #     typically hidden (e.g. '--allow-empty' for 'git commit').
+#
+#   GIT_COMPLETION_IGNORE_CASE
+#
+#     When set, uses for-each-ref '--ignore-case' to find refs that match
+#     case insensitively, even on systems with case sensitive file systems
+#     (e.g., completing tag name "FOO" on "git checkout f<TAB>").
 
 case "$COMP_WORDBREAKS" in
 *:*) : great ;;
@@ -646,6 +652,7 @@ __git_heads ()
        local pfx="${1-}" cur_="${2-}" sfx="${3-}"
 
        __git for-each-ref --format="${pfx//\%/%%}%(refname:strip=2)$sfx" \
+                       ${GIT_COMPLETION_IGNORE_CASE+--ignore-case} \
                        "refs/heads/$cur_*" "refs/heads/$cur_*/**"
 }
 
@@ -659,6 +666,7 @@ __git_remote_heads ()
        local pfx="${1-}" cur_="${2-}" sfx="${3-}"
 
        __git for-each-ref --format="${pfx//\%/%%}%(refname:strip=2)$sfx" \
+                       ${GIT_COMPLETION_IGNORE_CASE+--ignore-case} \
                        "refs/remotes/$cur_*" "refs/remotes/$cur_*/**"
 }
 
@@ -669,6 +677,7 @@ __git_tags ()
        local pfx="${1-}" cur_="${2-}" sfx="${3-}"
 
        __git for-each-ref --format="${pfx//\%/%%}%(refname:strip=2)$sfx" \
+                       ${GIT_COMPLETION_IGNORE_CASE+--ignore-case} \
                        "refs/tags/$cur_*" "refs/tags/$cur_*/**"
 }
 
@@ -688,6 +697,7 @@ __git_dwim_remote_heads ()
        # but only output if the branch name is unique
        __git for-each-ref --format="$fer_pfx%(refname:strip=3)$sfx" \
                --sort="refname:strip=3" \
+               ${GIT_COMPLETION_IGNORE_CASE+--ignore-case} \
                "refs/remotes/*/$cur_*" "refs/remotes/*/$cur_*/**" | \
        uniq -u
 }
@@ -712,6 +722,7 @@ __git_refs ()
        local format refs
        local pfx="${3-}" cur_="${4-$cur}" sfx="${5-}"
        local match="${4-}"
+       local umatch="${4-}"
        local fer_pfx="${pfx//\%/%%}" # "escape" for-each-ref format specifiers
 
        __git_find_repo_path
@@ -735,12 +746,19 @@ __git_refs ()
                fi
        fi
 
+       if test "${GIT_COMPLETION_IGNORE_CASE:+1}" = "1"
+       then
+               # uppercase with tr instead of ${match,^^} for bash 3.2 compatibility
+               umatch=$(echo "$match" | tr a-z A-Z 2>/dev/null || echo "$match")
+       fi
+
        if [ "$list_refs_from" = path ]; then
                if [[ "$cur_" == ^* ]]; then
                        pfx="$pfx^"
                        fer_pfx="$fer_pfx^"
                        cur_=${cur_#^}
                        match=${match#^}
+                       umatch=${umatch#^}
                fi
                case "$cur_" in
                refs|refs/*)
@@ -751,7 +769,7 @@ __git_refs ()
                *)
                        for i in HEAD FETCH_HEAD ORIG_HEAD MERGE_HEAD REBASE_HEAD CHERRY_PICK_HEAD; do
                                case "$i" in
-                               $match*)
+                               $match*|$umatch*)
                                        if [ -e "$dir/$i" ]; then
                                                echo "$pfx$i$sfx"
                                        fi
@@ -765,6 +783,7 @@ __git_refs ()
                        ;;
                esac
                __git_dir="$dir" __git for-each-ref --format="$fer_pfx%($format)$sfx" \
+                       ${GIT_COMPLETION_IGNORE_CASE+--ignore-case} \
                        "${refs[@]}"
                if [ -n "$track" ]; then
                        __git_dwim_remote_heads "$pfx" "$match" "$sfx"
@@ -784,15 +803,16 @@ __git_refs ()
        *)
                if [ "$list_refs_from" = remote ]; then
                        case "HEAD" in
-                       $match*       echo "${pfx}HEAD$sfx" ;;
+                       $match*|$umatch*)       echo "${pfx}HEAD$sfx" ;;
                        esac
                        __git for-each-ref --format="$fer_pfx%(refname:strip=3)$sfx" \
+                               ${GIT_COMPLETION_IGNORE_CASE+--ignore-case} \
                                "refs/remotes/$remote/$match*" \
                                "refs/remotes/$remote/$match*/**"
                else
                        local query_symref
                        case "HEAD" in
-                       $match*       query_symref="HEAD" ;;
+                       $match*|$umatch*)       query_symref="HEAD" ;;
                        esac
                        __git ls-remote "$remote" $query_symref \
                                "refs/tags/$match*" "refs/heads/$match*" \
index 8bcace29d21eea15e224d9ef2a841b98fdf36fac..3211841305fcb3cc9cce22bcd6c445cf327e274f 100644 (file)
@@ -79,6 +79,14 @@ git jump grep -i foo_bar
 git config jump.grepCmd "ag --column"
 --------------------------------------------------
 
+You can use the optional argument '--stdout' to print the listing to
+standard output instead of feeding it to the editor. You can use the
+argument with M-x grep on Emacs:
+
+--------------------------------------------------
+# In Emacs, M-x grep and invoke "git jump --stdout <mode>"
+M-x grep<RET>git jump --stdout diff<RET>
+--------------------------------------------------
 
 Related Programs
 ----------------
@@ -100,7 +108,7 @@ Limitations
 -----------
 
 This script was written and tested with vim. Given that the quickfix
-format is the same as what gcc produces, I expect emacs users have a
+format is the same as what gcc produces, I expect other tools have a
 similar feature for iterating through the list, but I know nothing about
 how to activate it.
 
index 92dbd4cde18ea18f6933ed79648a61d0c7493796..40c4b0d111071676c278651f49669036e90f079a 100755 (executable)
@@ -2,7 +2,7 @@
 
 usage() {
        cat <<\EOF
-usage: git jump <mode> [<args>]
+usage: git jump [--stdout] <mode> [<args>]
 
 Jump to interesting elements in an editor.
 The <mode> parameter is one of:
@@ -15,12 +15,30 @@ grep: elements are grep hits. Arguments are given to git grep or, if
       configured, to the command in `jump.grepCmd`.
 
 ws: elements are whitespace errors. Arguments are given to diff --check.
+
+If the optional argument `--stdout` is given, print the quickfix
+lines to standard output instead of feeding it to the editor.
 EOF
 }
 
 open_editor() {
        editor=`git var GIT_EDITOR`
-       eval "$editor -q \$1"
+       case "$editor" in
+       *emacs*)
+               # Supported editor values are:
+               # - emacs
+               # - emacsclient
+               # - emacsclient -t
+               #
+               # Wait for completion of the asynchronously executed process
+               # to avoid race conditions in case of "emacsclient".
+               eval "$editor --eval \"(let ((buf (grep \\\"cat \$1\\\"))) (pop-to-buffer buf) (select-frame-set-input-focus (selected-frame)) (while (get-buffer-process buf) (sleep-for 0.1)))\""
+               ;;
+       *)
+               # assume anything else is vi-compatible
+               eval "$editor -q \$1"
+               ;;
+       esac
 }
 
 mode_diff() {
@@ -64,15 +82,36 @@ mode_ws() {
        git diff --check "$@"
 }
 
+use_stdout=
+while test $# -gt 0; do
+       case "$1" in
+       --stdout)
+               use_stdout=t
+               ;;
+       --*)
+               usage >&2
+               exit 1
+               ;;
+       *)
+               break
+               ;;
+       esac
+       shift
+done
 if test $# -lt 1; then
        usage >&2
        exit 1
 fi
 mode=$1; shift
+type "mode_$mode" >/dev/null 2>&1 || { usage >&2; exit 1; }
+
+if test "$use_stdout" = "t"; then
+       "mode_$mode" "$@"
+       exit 0
+fi
 
 trap 'rm -f "$tmp"' 0 1 2 3 15
 tmp=`mktemp -t git-jump.XXXXXX` || exit 1
-type "mode_$mode" >/dev/null 2>&1 || { usage >&2; exit 1; }
 "mode_$mode" "$@" >"$tmp"
 test -s "$tmp" || exit 0
 open_editor "$tmp"
index 2edea41a2345af623017d43e14a74d71a32bb873..dec040c366c82a71bc57bad9cf9db865182efe20 100644 (file)
@@ -673,7 +673,7 @@ int index_differs_from(struct repository *r,
        return (has_changes != 0);
 }
 
-static struct strbuf *idiff_prefix_cb(struct diff_options *opt, void *data)
+static struct strbuf *idiff_prefix_cb(struct diff_options *opt UNUSED, void *data)
 {
        return data;
 }
index 18edbdf4b59ec020cc58e3d16d01d92edfdc302e..05fafd0019b6d5acc7295b999a8a46c92e1a649e 100644 (file)
@@ -255,8 +255,7 @@ int diff_no_index(struct rev_info *revs,
        };
        struct option *options;
 
-       options = parse_options_concat(no_index_options,
-                                      revs->diffopt.parseopts);
+       options = add_diff_options(no_index_options, &revs->diffopt);
        argc = parse_options(argc, argv, revs->prefix, options,
                             diff_no_index_usage, 0);
        if (argc != 2) {
diff --git a/diff.c b/diff.c
index 1054a4b7329d0b70ed62311a728b5257df3a8143..329eebf16a0b100cb3de9afb530e4cac25fede4c 100644 (file)
--- a/diff.c
+++ b/diff.c
@@ -604,7 +604,7 @@ static unsigned long diff_filespec_size(struct repository *r,
        return one->size;
 }
 
-static int count_trailing_blank(mmfile_t *mf, unsigned ws_rule)
+static int count_trailing_blank(mmfile_t *mf)
 {
        char *ptr = mf->ptr;
        long size = mf->size;
@@ -622,7 +622,7 @@ static int count_trailing_blank(mmfile_t *mf, unsigned ws_rule)
                for (prev_eol = ptr; mf->ptr <= prev_eol; prev_eol--)
                        if (*prev_eol == '\n')
                                break;
-               if (!ws_blank_line(prev_eol + 1, ptr - prev_eol, ws_rule))
+               if (!ws_blank_line(prev_eol + 1, ptr - prev_eol))
                        break;
                cnt++;
                ptr = prev_eol - 1;
@@ -634,9 +634,8 @@ static void check_blank_at_eof(mmfile_t *mf1, mmfile_t *mf2,
                               struct emit_callback *ecbdata)
 {
        int l1, l2, at;
-       unsigned ws_rule = ecbdata->ws_rule;
-       l1 = count_trailing_blank(mf1, ws_rule);
-       l2 = count_trailing_blank(mf2, ws_rule);
+       l1 = count_trailing_blank(mf1);
+       l2 = count_trailing_blank(mf2);
        if (l2 <= l1) {
                ecbdata->blank_at_eof_in_preimage = 0;
                ecbdata->blank_at_eof_in_postimage = 0;
@@ -1583,7 +1582,7 @@ static int new_blank_line_at_eof(struct emit_callback *ecbdata, const char *line
              ecbdata->blank_at_eof_in_preimage <= ecbdata->lno_in_preimage &&
              ecbdata->blank_at_eof_in_postimage <= ecbdata->lno_in_postimage))
                return 0;
-       return ws_blank_line(line, len, ecbdata->ws_rule);
+       return ws_blank_line(line, len);
 }
 
 static void emit_add_line(struct emit_callback *ecbdata,
@@ -1955,7 +1954,7 @@ static int color_words_output_graph_prefix(struct diff_words_data *diff_words)
 static void fn_out_diff_words_aux(void *priv,
                                  long minus_first, long minus_len,
                                  long plus_first, long plus_len,
-                                 const char *func, long funclen)
+                                 const char *func UNUSED, long funclen UNUSED)
 {
        struct diff_words_data *diff_words = priv;
        struct diff_words_style *style = diff_words->style;
@@ -2801,7 +2800,7 @@ static void show_stats(struct diffstat_t *data, struct diff_options *options)
                else if (file->is_unmerged) {
                        strbuf_addf(&out, " %s%s%*s | %*s",
                                    prefix, name, padding, "",
-                                   number_width, "Unmerged");
+                                   number_width, "Unmerged\n");
                        emit_diff_symbol(options, DIFF_SYMBOL_STATS_LINE,
                                         out.buf, out.len, 0);
                        strbuf_reset(&out);
@@ -3185,8 +3184,9 @@ static int is_conflict_marker(const char *line, int marker_size, unsigned long l
 }
 
 static void checkdiff_consume_hunk(void *priv,
-                                  long ob, long on, long nb, long nn,
-                                  const char *func, long funclen)
+                                  long ob UNUSED, long on UNUSED,
+                                  long nb, long nn UNUSED,
+                                  const char *func UNUSED, long funclen UNUSED)
 
 {
        struct checkdiff_t *data = priv;
@@ -4213,7 +4213,6 @@ static void prep_temp_blob(struct index_state *istate,
 }
 
 static struct diff_tempfile *prepare_temp_file(struct repository *r,
-                                              const char *name,
                                               struct diff_filespec *one)
 {
        struct diff_tempfile *temp = claim_diff_tempfile();
@@ -4231,18 +4230,18 @@ static struct diff_tempfile *prepare_temp_file(struct repository *r,
 
        if (!S_ISGITLINK(one->mode) &&
            (!one->oid_valid ||
-            reuse_worktree_file(r->index, name, &one->oid, 1))) {
+            reuse_worktree_file(r->index, one->path, &one->oid, 1))) {
                struct stat st;
-               if (lstat(name, &st) < 0) {
+               if (lstat(one->path, &st) < 0) {
                        if (errno == ENOENT)
                                goto not_a_valid_file;
-                       die_errno("stat(%s)", name);
+                       die_errno("stat(%s)", one->path);
                }
                if (S_ISLNK(st.st_mode)) {
                        struct strbuf sb = STRBUF_INIT;
-                       if (strbuf_readlink(&sb, name, st.st_size) < 0)
-                               die_errno("readlink(%s)", name);
-                       prep_temp_blob(r->index, name, temp, sb.buf, sb.len,
+                       if (strbuf_readlink(&sb, one->path, st.st_size) < 0)
+                               die_errno("readlink(%s)", one->path);
+                       prep_temp_blob(r->index, one->path, temp, sb.buf, sb.len,
                                       (one->oid_valid ?
                                        &one->oid : null_oid()),
                                       (one->oid_valid ?
@@ -4251,7 +4250,7 @@ static struct diff_tempfile *prepare_temp_file(struct repository *r,
                }
                else {
                        /* we can borrow from the file in the work tree */
-                       temp->name = name;
+                       temp->name = one->path;
                        if (!one->oid_valid)
                                oid_to_hex_r(temp->hex, null_oid());
                        else
@@ -4269,7 +4268,7 @@ static struct diff_tempfile *prepare_temp_file(struct repository *r,
        else {
                if (diff_populate_filespec(r, one, NULL))
                        die("cannot read data blob for %s", one->path);
-               prep_temp_blob(r->index, name, temp,
+               prep_temp_blob(r->index, one->path, temp,
                               one->data, one->size,
                               &one->oid, one->mode);
        }
@@ -4278,10 +4277,9 @@ static struct diff_tempfile *prepare_temp_file(struct repository *r,
 
 static void add_external_diff_name(struct repository *r,
                                   struct strvec *argv,
-                                  const char *name,
                                   struct diff_filespec *df)
 {
-       struct diff_tempfile *temp = prepare_temp_file(r, name, df);
+       struct diff_tempfile *temp = prepare_temp_file(r, df);
        strvec_push(argv, temp->name);
        strvec_push(argv, temp->hex);
        strvec_push(argv, temp->mode);
@@ -4308,11 +4306,9 @@ static void run_external_diff(const char *pgm,
        strvec_push(&cmd.args, name);
 
        if (one && two) {
-               add_external_diff_name(o->repo, &cmd.args, name, one);
-               if (!other)
-                       add_external_diff_name(o->repo, &cmd.args, name, two);
-               else {
-                       add_external_diff_name(o->repo, &cmd.args, other, two);
+               add_external_diff_name(o->repo, &cmd.args, one);
+               add_external_diff_name(o->repo, &cmd.args, two);
+               if (other) {
                        strvec_push(&cmd.args, other);
                        strvec_push(&cmd.args, xfrm_msg);
                }
@@ -4615,8 +4611,6 @@ static void run_checkdiff(struct diff_filepair *p, struct diff_options *o)
        builtin_checkdiff(name, other, attr_path, p->one, p->two, o);
 }
 
-static void prep_parse_options(struct diff_options *options);
-
 void repo_diff_setup(struct repository *r, struct diff_options *options)
 {
        memcpy(options, &default_diff_options, sizeof(*options));
@@ -4662,8 +4656,6 @@ void repo_diff_setup(struct repository *r, struct diff_options *options)
 
        options->color_moved = diff_color_moved_default;
        options->color_moved_ws_handling = diff_color_moved_ws_default;
-
-       prep_parse_options(options);
 }
 
 static const char diff_status_letters[] = {
@@ -4821,8 +4813,6 @@ void diff_setup_done(struct diff_options *options)
                        options->filter = ~filter_bit[DIFF_STATUS_FILTER_AON];
                options->filter &= ~options->filter_not;
        }
-
-       FREE_AND_NULL(options->parseopts);
 }
 
 int parse_long_opt(const char *opt, const char **argv,
@@ -5419,7 +5409,8 @@ static int diff_opt_rotate_to(const struct option *opt, const char *arg, int uns
        return 0;
 }
 
-static void prep_parse_options(struct diff_options *options)
+struct option *add_diff_options(const struct option *opts,
+                               struct diff_options *options)
 {
        struct option parseopts[] = {
                OPT_GROUP(N_("Diff output format options")),
@@ -5689,22 +5680,25 @@ static void prep_parse_options(struct diff_options *options)
                OPT_END()
        };
 
-       ALLOC_ARRAY(options->parseopts, ARRAY_SIZE(parseopts));
-       memcpy(options->parseopts, parseopts, sizeof(parseopts));
+       return parse_options_concat(opts, parseopts);
 }
 
 int diff_opt_parse(struct diff_options *options,
                   const char **av, int ac, const char *prefix)
 {
+       struct option no_options[] = { OPT_END() };
+       struct option *parseopts = add_diff_options(no_options, options);
+
        if (!prefix)
                prefix = "";
 
-       ac = parse_options(ac, av, prefix, options->parseopts, NULL,
+       ac = parse_options(ac, av, prefix, parseopts, NULL,
                           PARSE_OPT_KEEP_DASHDASH |
                           PARSE_OPT_KEEP_UNKNOWN_OPT |
                           PARSE_OPT_NO_INTERNAL_HELP |
                           PARSE_OPT_ONE_SHOT |
                           PARSE_OPT_STOP_AT_NON_OPTION);
+       free(parseopts);
 
        return ac;
 }
@@ -6513,7 +6507,6 @@ void diff_free(struct diff_options *options)
        diff_free_file(options);
        diff_free_ignore_regex(options);
        clear_pathspec(&options->pathspec);
-       FREE_AND_NULL(options->parseopts);
 }
 
 void diff_flush(struct diff_options *options)
@@ -7037,7 +7030,7 @@ static char *run_textconv(struct repository *r,
        struct strbuf buf = STRBUF_INIT;
        int err = 0;
 
-       temp = prepare_temp_file(r, spec->path, spec);
+       temp = prepare_temp_file(r, spec);
        strvec_push(&child.args, pgm);
        strvec_push(&child.args, temp->name);
 
diff --git a/diff.h b/diff.h
index fd33caeb25dc84b53dbac18d59898bcd3cc36a7d..41eb2c3d428801130bc6ffe355cc723067689b85 100644 (file)
--- a/diff.h
+++ b/diff.h
@@ -394,7 +394,6 @@ struct diff_options {
        unsigned color_moved_ws_handling;
 
        struct repository *repo;
-       struct option *parseopts;
        struct strmap *additional_path_headers;
 
        int no_free;
@@ -539,6 +538,7 @@ int git_diff_ui_config(const char *var, const char *value, void *cb);
 #define diff_setup(diffopts) repo_diff_setup(the_repository, diffopts)
 #endif
 void repo_diff_setup(struct repository *, struct diff_options *);
+struct option *add_diff_options(const struct option *, struct diff_options *);
 int diff_opt_parse(struct diff_options *, const char **, int, const char *);
 void diff_setup_done(struct diff_options *);
 int git_config_rename(const char *var, const char *value);
diff --git a/dir.c b/dir.c
index d604d1bab982980fd5506a0fae072bac3510a5ae..4e99f0c868f3d7955c4a6f84bf22bbce0d917ae3 100644 (file)
--- a/dir.c
+++ b/dir.c
@@ -732,6 +732,13 @@ static void add_pattern_to_hashsets(struct pattern_list *pl, struct path_pattern
                goto clear_hashmaps;
        }
 
+       if (!(given->flags & PATTERN_FLAG_MUSTBEDIR) &&
+           strcmp(given->pattern, "/*")) {
+               /* Not a cone pattern. */
+               warning(_("unrecognized pattern: '%s'"), given->pattern);
+               goto clear_hashmaps;
+       }
+
        prev = given->pattern;
        cur = given->pattern + 1;
        next = given->pattern + 2;
@@ -3581,8 +3588,12 @@ static void free_untracked(struct untracked_cache_dir *ucd)
 
 void free_untracked_cache(struct untracked_cache *uc)
 {
-       if (uc)
-               free_untracked(uc->root);
+       if (!uc)
+               return;
+
+       free(uc->exclude_per_dir_to_free);
+       strbuf_release(&uc->ident);
+       free_untracked(uc->root);
        free(uc);
 }
 
@@ -3739,7 +3750,7 @@ struct untracked_cache *read_untracked_extension(const void *data, unsigned long
                      next + offset + hashsz);
        uc->dir_flags = get_be32(next + ouc_offset(dir_flags));
        exclude_per_dir = (const char *)next + exclude_per_dir_offset;
-       uc->exclude_per_dir = xstrdup(exclude_per_dir);
+       uc->exclude_per_dir = uc->exclude_per_dir_to_free = xstrdup(exclude_per_dir);
        /* NUL after exclude_per_dir is covered by sizeof(*ouc) */
        next += exclude_per_dir_offset + strlen(exclude_per_dir) + 1;
        if (next >= end)
diff --git a/dir.h b/dir.h
index 674747d93af7ad903b1e78a1418ff07c914a69ec..8acfc044181ca2f47aa39bef25713b1a811e987c 100644 (file)
--- a/dir.h
+++ b/dir.h
@@ -188,6 +188,7 @@ struct untracked_cache {
        struct oid_stat ss_info_exclude;
        struct oid_stat ss_excludes_file;
        const char *exclude_per_dir;
+       char *exclude_per_dir_to_free;
        struct strbuf ident;
        /*
         * dir_struct#flags must match dir_flags or the untracked
diff --git a/entry.c b/entry.c
index 616e4f073c1d6bd4016252423d4ced0958612432..971ab268714b96cdaa179ae7589c8377ff728e74 100644 (file)
--- a/entry.c
+++ b/entry.c
@@ -383,7 +383,7 @@ static int write_entry(struct cache_entry *ce, char *path, struct conv_attrs *ca
                        return error("cannot create submodule directory %s", path);
                sub = submodule_from_ce(ce);
                if (sub)
-                       return submodule_move_head(ce->name,
+                       return submodule_move_head(ce->name, state->super_prefix,
                                NULL, oid_to_hex(&ce->oid),
                                state->force ? SUBMODULE_MOVE_HEAD_FORCE : 0);
                break;
@@ -476,7 +476,7 @@ int checkout_entry_ca(struct cache_entry *ce, struct conv_attrs *ca,
                         * no pathname to return.
                         */
                        BUG("Can't remove entry to a path");
-               unlink_entry(ce);
+               unlink_entry(ce, state->super_prefix);
                return 0;
        }
 
@@ -510,10 +510,10 @@ int checkout_entry_ca(struct cache_entry *ce, struct conv_attrs *ca,
                                if (!(st.st_mode & S_IFDIR))
                                        unlink_or_warn(ce->name);
 
-                               return submodule_move_head(ce->name,
+                               return submodule_move_head(ce->name, state->super_prefix,
                                        NULL, oid_to_hex(&ce->oid), 0);
                        } else
-                               return submodule_move_head(ce->name,
+                               return submodule_move_head(ce->name, state->super_prefix,
                                        "HEAD", oid_to_hex(&ce->oid),
                                        state->force ? SUBMODULE_MOVE_HEAD_FORCE : 0);
                }
@@ -560,12 +560,12 @@ int checkout_entry_ca(struct cache_entry *ce, struct conv_attrs *ca,
        return write_entry(ce, path.buf, ca, state, 0, nr_checkouts);
 }
 
-void unlink_entry(const struct cache_entry *ce)
+void unlink_entry(const struct cache_entry *ce, const char *super_prefix)
 {
        const struct submodule *sub = submodule_from_ce(ce);
        if (sub) {
                /* state.force is set at the caller. */
-               submodule_move_head(ce->name, "HEAD", NULL,
+               submodule_move_head(ce->name, super_prefix, "HEAD", NULL,
                                    SUBMODULE_MOVE_HEAD_FORCE);
        }
        if (check_leading_path(ce->name, ce_namelen(ce), 1) >= 0)
diff --git a/entry.h b/entry.h
index 9be4659881e6ec979bb4ac5233c71af44b337586..2d4fbb88c8ffabbd36faec0247d5a39a30f99c4f 100644 (file)
--- a/entry.h
+++ b/entry.h
@@ -8,6 +8,7 @@ struct checkout {
        struct index_state *istate;
        const char *base_dir;
        int base_dir_len;
+       const char *super_prefix;
        struct delayed_checkout *delayed_checkout;
        struct checkout_metadata meta;
        unsigned force:1,
@@ -48,8 +49,11 @@ int finish_delayed_checkout(struct checkout *state, int show_progress);
 /*
  * Unlink the last component and schedule the leading directories for
  * removal, such that empty directories get removed.
+ *
+ * The "super_prefix" is either NULL, or the "--super-prefix" passed
+ * down from "read-tree" et al.
  */
-void unlink_entry(const struct cache_entry *ce);
+void unlink_entry(const struct cache_entry *ce, const char *super_prefix);
 
 void *read_blob_entry(const struct cache_entry *ce, size_t *size);
 int fstat_checkout_output(int fd, const struct checkout *state, struct stat *st);
index 18d042b467d26a9cb3b228db1ef5afa2d903e72c..1ee3686fd8a34270db8d47738f2ea12fcdbc1976 100644 (file)
@@ -102,8 +102,6 @@ char *git_work_tree_cfg;
 
 static char *git_namespace;
 
-static char *super_prefix;
-
 /*
  * Repository-local GIT_* environment variables; see cache.h for details.
  */
@@ -121,7 +119,6 @@ const char * const local_repo_env[] = {
        NO_REPLACE_OBJECTS_ENVIRONMENT,
        GIT_REPLACE_REF_BASE_ENVIRONMENT,
        GIT_PREFIX_ENVIRONMENT,
-       GIT_SUPER_PREFIX_ENVIRONMENT,
        GIT_SHALLOW_FILE_ENVIRONMENT,
        GIT_COMMON_DIR_ENVIRONMENT,
        NULL
@@ -234,16 +231,6 @@ const char *strip_namespace(const char *namespaced_ref)
        return NULL;
 }
 
-const char *get_super_prefix(void)
-{
-       static int initialized;
-       if (!initialized) {
-               super_prefix = xstrdup_or_null(getenv(GIT_SUPER_PREFIX_ENVIRONMENT));
-               initialized = 1;
-       }
-       return super_prefix;
-}
-
 static int git_work_tree_initialized;
 
 /*
index 998fc2fa1ed4abd32ad0e723115f568768134d3e..04016d1e3250f28ae6cd53a784913f02879769d8 100644 (file)
@@ -1317,15 +1317,15 @@ static void write_fetch_command_and_capabilities(struct strbuf *req_buf,
 {
        const char *hash_name;
 
-       if (server_supports_v2("fetch", 1))
-               packet_buf_write(req_buf, "command=fetch");
-       if (server_supports_v2("agent", 0))
+       ensure_server_supports_v2("fetch");
+       packet_buf_write(req_buf, "command=fetch");
+       if (server_supports_v2("agent"))
                packet_buf_write(req_buf, "agent=%s", git_user_agent_sanitized());
-       if (advertise_sid && server_supports_v2("session-id", 0))
+       if (advertise_sid && server_supports_v2("session-id"))
                packet_buf_write(req_buf, "session-id=%s", trace2_session_id());
-       if (server_options && server_options->nr &&
-           server_supports_v2("server-option", 1)) {
+       if (server_options && server_options->nr) {
                int i;
+               ensure_server_supports_v2("server-option");
                for (i = 0; i < server_options->nr; i++)
                        packet_buf_write(req_buf, "server-option=%s",
                                         server_options->items[i].string);
diff --git a/git-bisect.sh b/git-bisect.sh
deleted file mode 100755 (executable)
index dfce4b4..0000000
+++ /dev/null
@@ -1,67 +0,0 @@
-#!/bin/sh
-
-USAGE='[help|start|bad|good|new|old|terms|skip|next|reset|visualize|view|replay|log|run]'
-LONG_USAGE='git bisect help
-       print this long help message.
-git bisect start [--term-{new,bad}=<term> --term-{old,good}=<term>]
-                [--no-checkout] [--first-parent] [<bad> [<good>...]] [--] [<pathspec>...]
-       reset bisect state and start bisection.
-git bisect (bad|new) [<rev>]
-       mark <rev> a known-bad revision/
-               a revision after change in a given property.
-git bisect (good|old) [<rev>...]
-       mark <rev>... known-good revisions/
-               revisions before change in a given property.
-git bisect terms [--term-good | --term-bad]
-       show the terms used for old and new commits (default: bad, good)
-git bisect skip [(<rev>|<range>)...]
-       mark <rev>... untestable revisions.
-git bisect next
-       find next bisection to test and check it out.
-git bisect reset [<commit>]
-       finish bisection search and go back to commit.
-git bisect (visualize|view)
-       show bisect status in gitk.
-git bisect replay <logfile>
-       replay bisection log.
-git bisect log
-       show bisect log.
-git bisect run <cmd>...
-       use <cmd>... to automatically bisect.
-
-Please use "git help bisect" to get the full man page.'
-
-OPTIONS_SPEC=
-. git-sh-setup
-
-TERM_BAD=bad
-TERM_GOOD=good
-
-get_terms () {
-       if test -s "$GIT_DIR/BISECT_TERMS"
-       then
-               {
-               read TERM_BAD
-               read TERM_GOOD
-               } <"$GIT_DIR/BISECT_TERMS"
-       fi
-}
-
-case "$#" in
-0)
-       usage ;;
-*)
-       cmd="$1"
-       get_terms
-       shift
-       case "$cmd" in
-       help)
-               git bisect -h ;;
-       bad|good|new|old|"$TERM_BAD"|"$TERM_GOOD")
-               git bisect--helper state "$cmd" "$@" ;;
-       log)
-               git bisect--helper log || exit ;;
-       *)
-               git bisect--helper "$cmd" "$@" ;;
-       esac
-esac
index a76d0526f79266e1dc512072a4236876df8f8726..76e4b111315339449cd48d17f735f727935c08d8 100644 (file)
@@ -341,11 +341,13 @@ struct itimerval {
 #endif
 
 #ifdef NO_SETITIMER
-static inline int setitimer(int which UNUSED,
-                           const struct itimerval *value UNUSED,
-                           struct itimerval *newvalue UNUSED) {
+static inline int git_setitimer(int which UNUSED,
+                               const struct itimerval *value UNUSED,
+                               struct itimerval *newvalue UNUSED) {
        return 0; /* pretend success */
 }
+#undef setitimer
+#define setitimer(which,value,ovalue) git_setitimer(which,value,ovalue)
 #endif
 
 #ifndef NO_LIBGEN_H
@@ -1471,14 +1473,19 @@ int open_nofollow(const char *path, int flags);
 #endif
 
 #ifndef _POSIX_THREAD_SAFE_FUNCTIONS
-static inline void flockfile(FILE *fh UNUSED)
+static inline void git_flockfile(FILE *fh UNUSED)
 {
        ; /* nothing */
 }
-static inline void funlockfile(FILE *fh UNUSED)
+static inline void git_funlockfile(FILE *fh UNUSED)
 {
        ; /* nothing */
 }
+#undef flockfile
+#undef funlockfile
+#undef getc_unlocked
+#define flockfile(fh) git_flockfile(fh)
+#define funlockfile(fh) git_funlockfile(fh)
 #define getc_unlocked(fh) getc(fh)
 #endif
 
index 9a50f2e9124492589aa3a298a8bec74dbded8e32..7f9582d92343450e0c71ba72e5faee845cc1165b 100755 (executable)
@@ -244,6 +244,9 @@ cmd_update()
                -q|--quiet)
                        quiet=1
                        ;;
+               -v|--verbose)
+                       quiet=0
+                       ;;
                --progress)
                        progress=1
                        ;;
diff --git a/git.c b/git.c
index 10202a7f126aba88c41c51e140f8c0aa0b148c02..32a5be68a17064ca1b773675761e33e30bf4056b 100644 (file)
--- a/git.c
+++ b/git.c
@@ -14,9 +14,8 @@
  * RUN_SETUP for reading from the configuration file.
  */
 #define NEED_WORK_TREE         (1<<3)
-#define SUPPORT_SUPER_PREFIX   (1<<4)
-#define DELAY_PAGER_CONFIG     (1<<5)
-#define NO_PARSEOPT            (1<<6) /* parse-options is not used */
+#define DELAY_PAGER_CONFIG     (1<<4)
+#define NO_PARSEOPT            (1<<5) /* parse-options is not used */
 
 struct cmd_struct {
        const char *cmd;
@@ -29,8 +28,7 @@ const char git_usage_string[] =
           "           [--exec-path[=<path>]] [--html-path] [--man-path] [--info-path]\n"
           "           [-p | --paginate | -P | --no-pager] [--no-replace-objects] [--bare]\n"
           "           [--git-dir=<path>] [--work-tree=<path>] [--namespace=<name>]\n"
-          "           [--super-prefix=<path>] [--config-env=<name>=<envvar>]\n"
-          "           <command> [<args>]");
+          "           [--config-env=<name>=<envvar>] <command> [<args>]");
 
 const char git_more_info_string[] =
        N_("'git help -a' and 'git help -g' list available subcommands and some\n"
@@ -226,20 +224,6 @@ static int handle_options(const char ***argv, int *argc, int *envchanged)
                        setenv(GIT_WORK_TREE_ENVIRONMENT, cmd, 1);
                        if (envchanged)
                                *envchanged = 1;
-               } else if (!strcmp(cmd, "--super-prefix")) {
-                       if (*argc < 2) {
-                               fprintf(stderr, _("no prefix given for --super-prefix\n" ));
-                               usage(git_usage_string);
-                       }
-                       setenv(GIT_SUPER_PREFIX_ENVIRONMENT, (*argv)[1], 1);
-                       if (envchanged)
-                               *envchanged = 1;
-                       (*argv)++;
-                       (*argc)--;
-               } else if (skip_prefix(cmd, "--super-prefix=", &cmd)) {
-                       setenv(GIT_SUPER_PREFIX_ENVIRONMENT, cmd, 1);
-                       if (envchanged)
-                               *envchanged = 1;
                } else if (!strcmp(cmd, "--bare")) {
                        char *cwd = xgetcwd();
                        is_bare_repository_cfg = 1;
@@ -449,11 +433,6 @@ static int run_builtin(struct cmd_struct *p, int argc, const char **argv)
                trace_repo_setup(prefix);
        commit_pager_choice();
 
-       if (!help && get_super_prefix()) {
-               if (!(p->option & SUPPORT_SUPER_PREFIX))
-                       die(_("%s doesn't support --super-prefix"), p->cmd);
-       }
-
        if (!help && p->option & NEED_WORK_TREE)
                setup_work_tree();
 
@@ -492,7 +471,7 @@ static struct cmd_struct commands[] = {
        { "annotate", cmd_annotate, RUN_SETUP },
        { "apply", cmd_apply, RUN_SETUP_GENTLY },
        { "archive", cmd_archive, RUN_SETUP_GENTLY },
-       { "bisect--helper", cmd_bisect__helper, RUN_SETUP },
+       { "bisect", cmd_bisect, RUN_SETUP },
        { "blame", cmd_blame, RUN_SETUP },
        { "branch", cmd_branch, RUN_SETUP | DELAY_PAGER_CONFIG },
        { "bugreport", cmd_bugreport, RUN_SETUP_GENTLY },
@@ -504,7 +483,7 @@ static struct cmd_struct commands[] = {
        { "check-ref-format", cmd_check_ref_format, NO_PARSEOPT  },
        { "checkout", cmd_checkout, RUN_SETUP | NEED_WORK_TREE },
        { "checkout--worker", cmd_checkout__worker,
-               RUN_SETUP | NEED_WORK_TREE | SUPPORT_SUPER_PREFIX },
+               RUN_SETUP | NEED_WORK_TREE },
        { "checkout-index", cmd_checkout_index,
                RUN_SETUP | NEED_WORK_TREE},
        { "cherry", cmd_cherry, RUN_SETUP },
@@ -539,7 +518,7 @@ static struct cmd_struct commands[] = {
        { "format-patch", cmd_format_patch, RUN_SETUP },
        { "fsck", cmd_fsck, RUN_SETUP },
        { "fsck-objects", cmd_fsck, RUN_SETUP },
-       { "fsmonitor--daemon", cmd_fsmonitor__daemon, SUPPORT_SUPER_PREFIX | RUN_SETUP },
+       { "fsmonitor--daemon", cmd_fsmonitor__daemon, RUN_SETUP },
        { "gc", cmd_gc, RUN_SETUP },
        { "get-tar-commit-id", cmd_get_tar_commit_id, NO_PARSEOPT },
        { "grep", cmd_grep, RUN_SETUP_GENTLY },
@@ -583,7 +562,7 @@ static struct cmd_struct commands[] = {
        { "pull", cmd_pull, RUN_SETUP | NEED_WORK_TREE },
        { "push", cmd_push, RUN_SETUP },
        { "range-diff", cmd_range_diff, RUN_SETUP | USE_PAGER },
-       { "read-tree", cmd_read_tree, RUN_SETUP | SUPPORT_SUPER_PREFIX},
+       { "read-tree", cmd_read_tree, RUN_SETUP },
        { "rebase", cmd_rebase, RUN_SETUP | NEED_WORK_TREE },
        { "receive-pack", cmd_receive_pack },
        { "reflog", cmd_reflog, RUN_SETUP },
@@ -610,7 +589,7 @@ static struct cmd_struct commands[] = {
        { "stash", cmd_stash, RUN_SETUP | NEED_WORK_TREE },
        { "status", cmd_status, RUN_SETUP | NEED_WORK_TREE },
        { "stripspace", cmd_stripspace },
-       { "submodule--helper", cmd_submodule__helper, RUN_SETUP | SUPPORT_SUPER_PREFIX },
+       { "submodule--helper", cmd_submodule__helper, RUN_SETUP },
        { "switch", cmd_switch, RUN_SETUP | NEED_WORK_TREE },
        { "symbolic-ref", cmd_symbolic_ref, RUN_SETUP },
        { "tag", cmd_tag, RUN_SETUP | DELAY_PAGER_CONFIG },
@@ -727,9 +706,6 @@ static void execv_dashed_external(const char **argv)
        struct child_process cmd = CHILD_PROCESS_INIT;
        int status;
 
-       if (get_super_prefix())
-               die(_("%s doesn't support --super-prefix"), argv[0]);
-
        if (use_pager == -1 && !is_builtin(argv[0]))
                use_pager = check_pager_config(argv[0]);
        commit_pager_choice();
@@ -799,9 +775,6 @@ static int run_argv(int *argcp, const char ***argv)
                         */
                        trace2_cmd_name("_run_git_alias_");
 
-                       if (get_super_prefix())
-                               die("%s doesn't support --super-prefix", **argv);
-
                        commit_pager_choice();
 
                        strvec_push(&cmd.args, "git");
diff --git a/help.c b/help.c
index f1e090a4428f0e9e55b11a528e983d0908ffc70e..812af4cdea62129c12ec33813d934d677c61d714 100644 (file)
--- a/help.c
+++ b/help.c
@@ -563,7 +563,7 @@ static int git_unknown_cmd_config(const char *var, const char *value, void *cb)
        if (skip_prefix(var, "alias.", &p))
                add_cmdname(&aliases, p, strlen(p));
 
-       return git_default_config(var, value, cb);
+       return 0;
 }
 
 static int levenshtein_compare(const void *p1, const void *p2)
index 31bc5c7767ce865bb3fb1a89f311f7d2b03c7e6d..258fec2068262c815ae68b08c8729b0bc60ee498 100644 (file)
@@ -5,6 +5,7 @@
 #include "walker.h"
 #include "strvec.h"
 #include "urlmatch.h"
+#include "trace2.h"
 
 static const char http_fetch_usage[] = "git http-fetch "
 "[-c] [-t] [-a] [-v] [--recover] [-w ref] [--stdin | --packfile=hash | commit-id] url";
@@ -137,6 +138,8 @@ int cmd_main(int argc, const char **argv)
        if (nongit)
                die(_("not a git repository"));
 
+       trace2_cmd_name("http-fetch");
+
        git_config(git_default_config, NULL);
 
        if (packfile) {
index 955a8a9535575fe0a397497d765f3e0e97334259..47bf0d6f1a253a01baf6e3d0f68dfa5d5df6a0f0 100644 (file)
@@ -135,7 +135,7 @@ static const char *find_funcname_matching_regexp(xdemitconf_t *xecfg, const char
 {
        int reg_error;
        regmatch_t match[1];
-       while (1) {
+       while (*start) {
                const char *bol, *eol;
                reg_error = regexec(regexp, start, 1, match, 0);
                if (reg_error == REG_NOMATCH)
@@ -148,8 +148,8 @@ static const char *find_funcname_matching_regexp(xdemitconf_t *xecfg, const char
                /* determine extent of line matched */
                bol = start+match[0].rm_so;
                eol = start+match[0].rm_eo;
-               while (bol > start && *bol != '\n')
-                       bol--;
+               while (bol > start && *--bol != '\n')
+                       ; /* nothing */
                if (*bol == '\n')
                        bol++;
                while (*eol && *eol != '\n')
@@ -161,6 +161,7 @@ static const char *find_funcname_matching_regexp(xdemitconf_t *xecfg, const char
                        return bol;
                start = eol;
        }
+       return NULL;
 }
 
 static const char *parse_range_funcname(
index 533966023804776ec19ca06c7e84b4b264c07314..ee01bcd2cc365c12c72bcd3f5597ecceecb2a7d9 100644 (file)
@@ -290,10 +290,6 @@ int opt_parse_list_objects_filter(const struct option *opt,
                                  const char *arg, int unset)
 {
        struct list_objects_filter_options *filter_options = opt->value;
-       opt_lof_init init = (opt_lof_init)opt->defval;
-
-       if (init)
-               filter_options = init(opt->value);
 
        if (unset || !arg)
                list_objects_filter_set_no_filter(filter_options);
index 7eeadab2dd0551c5eea8f7f14cd741841317230e..1fe393f447303a6d94abbd571a1ea83a1bfc7620 100644 (file)
@@ -111,27 +111,13 @@ void parse_list_objects_filter(
  * The opt->value to opt_parse_list_objects_filter() is either a
  * "struct list_objects_filter_option *" when using
  * OPT_PARSE_LIST_OBJECTS_FILTER().
- *
- * Or, if using no "struct option" field is used by the callback,
- * except the "defval" which is expected to be an "opt_lof_init"
- * function, which is called with the "opt->value" and must return a
- * pointer to the ""struct list_objects_filter_option *" to be used.
- *
- * The OPT_PARSE_LIST_OBJECTS_FILTER_INIT() can be used e.g. the
- * "struct list_objects_filter_option" is embedded in a "struct
- * rev_info", which the "defval" could be tasked with lazily
- * initializing. See cmd_pack_objects() for an example.
  */
 int opt_parse_list_objects_filter(const struct option *opt,
                                  const char *arg, int unset);
-typedef struct list_objects_filter_options *(*opt_lof_init)(void *);
-#define OPT_PARSE_LIST_OBJECTS_FILTER_INIT(fo, init) \
-       { OPTION_CALLBACK, 0, "filter", (fo), N_("args"), \
-         N_("object filtering"), 0, opt_parse_list_objects_filter, \
-         (intptr_t)(init) }
 
 #define OPT_PARSE_LIST_OBJECTS_FILTER(fo) \
-       OPT_PARSE_LIST_OBJECTS_FILTER_INIT((fo), NULL)
+       OPT_CALLBACK(0, "filter", (fo), N_("args"), \
+                    N_("object filtering"), opt_parse_list_objects_filter)
 
 /*
  * Translates abbreviated numbers in the filter's filter_spec into their
index b9543545ca4af96e062ec69546c354e318e4ad0d..7ed21cb299c126fb2d2107eb3e418a18e64fbc6c 100644 (file)
@@ -70,13 +70,13 @@ struct filter {
 };
 
 static enum list_objects_filter_result filter_blobs_none(
-       struct repository *r,
+       struct repository *r UNUSED,
        enum list_objects_filter_situation filter_situation,
        struct object *obj,
-       const char *pathname,
-       const char *filename,
+       const char *pathname UNUSED,
+       const char *filename UNUSED,
        struct oidset *omits,
-       void *filter_data_)
+       void *filter_data_ UNUSED)
 {
        switch (filter_situation) {
        default:
@@ -112,7 +112,7 @@ static enum list_objects_filter_result filter_blobs_none(
 }
 
 static void filter_blobs_none__init(
-       struct list_objects_filter_options *filter_options,
+       struct list_objects_filter_options *filter_options UNUSED,
        struct filter *filter)
 {
        filter->filter_object_fn = filter_blobs_none;
@@ -159,11 +159,11 @@ static int filter_trees_update_omits(
 }
 
 static enum list_objects_filter_result filter_trees_depth(
-       struct repository *r,
+       struct repository *r UNUSED,
        enum list_objects_filter_situation filter_situation,
        struct object *obj,
-       const char *pathname,
-       const char *filename,
+       const char *pathname UNUSED,
+       const char *filename UNUSED,
        struct oidset *omits,
        void *filter_data_)
 {
@@ -274,8 +274,8 @@ static enum list_objects_filter_result filter_blobs_limit(
        struct repository *r,
        enum list_objects_filter_situation filter_situation,
        struct object *obj,
-       const char *pathname,
-       const char *filename,
+       const char *pathname UNUSED,
+       const char *filename UNUSED,
        struct oidset *omits,
        void *filter_data_)
 {
@@ -514,6 +514,7 @@ static enum list_objects_filter_result filter_sparse(
 static void filter_sparse_free(void *filter_data)
 {
        struct filter_sparse_data *d = filter_data;
+       clear_pattern_list(&d->pl);
        free(d->array_frame);
        free(d);
 }
@@ -554,12 +555,12 @@ struct filter_object_type_data {
 };
 
 static enum list_objects_filter_result filter_object_type(
-       struct repository *r,
+       struct repository *r UNUSED,
        enum list_objects_filter_situation filter_situation,
        struct object *obj,
-       const char *pathname,
-       const char *filename,
-       struct oidset *omits,
+       const char *pathname UNUSED,
+       const char *filename UNUSED,
+       struct oidset *omits UNUSED,
        void *filter_data_)
 {
        struct filter_object_type_data *filter_data = filter_data_;
@@ -675,7 +676,7 @@ static enum list_objects_filter_result filter_combine(
        struct object *obj,
        const char *pathname,
        const char *filename,
-       struct oidset *omits,
+       struct oidset *omits UNUSED,
        void *filter_data)
 {
        struct combine_filter_data *d = filter_data;
index 250d9de41cb56072e95420530b203ce417191a2b..7528fe1db2c85c0b4d923854ef3ec53c3a8f1fad 100644 (file)
@@ -81,36 +81,6 @@ static void process_blob(struct traversal_context *ctx,
        strbuf_setlen(path, pathlen);
 }
 
-/*
- * Processing a gitlink entry currently does nothing, since
- * we do not recurse into the subproject.
- *
- * We *could* eventually add a flag that actually does that,
- * which would involve:
- *  - is the subproject actually checked out?
- *  - if so, see if the subproject has already been added
- *    to the alternates list, and add it if not.
- *  - process the commit (or tag) the gitlink points to
- *    recursively.
- *
- * However, it's unclear whether there is really ever any
- * reason to see superprojects and subprojects as such a
- * "unified" object pool (potentially resulting in a totally
- * humongous pack - avoiding which was the whole point of
- * having gitlinks in the first place!).
- *
- * So for now, there is just a note that we *could* follow
- * the link, and how to do it. Whether it necessarily makes
- * any sense what-so-ever to ever do that is another issue.
- */
-static void process_gitlink(struct traversal_context *ctx,
-                           const unsigned char *sha1,
-                           struct strbuf *path,
-                           const char *name)
-{
-       /* Nothing to do */
-}
-
 static void process_tree(struct traversal_context *ctx,
                         struct tree *tree,
                         struct strbuf *base,
@@ -149,8 +119,7 @@ static void process_tree_contents(struct traversal_context *ctx,
                        process_tree(ctx, t, base, entry.path);
                }
                else if (S_ISGITLINK(entry.mode))
-                       process_gitlink(ctx, entry.oid.hash,
-                                       base, entry.path);
+                       ; /* ignore gitlink */
                else {
                        struct blob *b = lookup_blob(ctx->revs->repo, &entry.oid);
                        if (!b) {
index fb6769742cd85ebe3782b071634d45922305f5d0..697d4beb8de43a4a7ae48be8284e10c6be2a9bfd 100644 (file)
--- a/ls-refs.c
+++ b/ls-refs.c
@@ -194,8 +194,9 @@ int ls_refs(struct repository *r, struct packet_reader *request)
        send_possibly_unborn_head(&data);
        if (!data.prefixes.nr)
                strvec_push(&data.prefixes, "");
-       for_each_fullref_in_prefixes(get_git_namespace(), data.prefixes.v,
-                                    send_ref, &data);
+       refs_for_each_fullref_in_prefixes(get_main_ref_store(r),
+                                         get_git_namespace(), data.prefixes.v,
+                                         send_ref, &data);
        packet_fflush(stdout);
        strvec_clear(&data.prefixes);
        strbuf_release(&data.buf);
index 26290554bb4c5f1d679194fe7d3f2b5c78a273e2..80a0cd3b35183476bc56c8ce510d0863e171069e 100644 (file)
@@ -1211,35 +1211,25 @@ static int quick_has_loose(struct repository *r,
 }
 
 /*
- * Map the loose object at "path" if it is not NULL, or the path found by
- * searching for a loose object named "oid".
+ * Map and close the given loose object fd. The path argument is used for
+ * error reporting.
  */
-static void *map_loose_object_1(struct repository *r, const char *path,
-                            const struct object_id *oid, unsigned long *size)
+static void *map_fd(int fd, const char *path, unsigned long *size)
 {
-       void *map;
-       int fd;
-
-       if (path)
-               fd = git_open(path);
-       else
-               fd = open_loose_object(r, oid, &path);
-       map = NULL;
-       if (fd >= 0) {
-               struct stat st;
+       void *map = NULL;
+       struct stat st;
 
-               if (!fstat(fd, &st)) {
-                       *size = xsize_t(st.st_size);
-                       if (!*size) {
-                               /* mmap() is forbidden on empty files */
-                               error(_("object file %s is empty"), path);
-                               close(fd);
-                               return NULL;
-                       }
-                       map = xmmap(NULL, *size, PROT_READ, MAP_PRIVATE, fd, 0);
+       if (!fstat(fd, &st)) {
+               *size = xsize_t(st.st_size);
+               if (!*size) {
+                       /* mmap() is forbidden on empty files */
+                       error(_("object file %s is empty"), path);
+                       close(fd);
+                       return NULL;
                }
-               close(fd);
+               map = xmmap(NULL, *size, PROT_READ, MAP_PRIVATE, fd, 0);
        }
+       close(fd);
        return map;
 }
 
@@ -1247,7 +1237,12 @@ void *map_loose_object(struct repository *r,
                       const struct object_id *oid,
                       unsigned long *size)
 {
-       return map_loose_object_1(r, NULL, oid, size);
+       const char *p;
+       int fd = open_loose_object(r, oid, &p);
+
+       if (fd < 0)
+               return NULL;
+       return map_fd(fd, p, size);
 }
 
 enum unpack_loose_header_result unpack_loose_header(git_zstream *stream,
@@ -1427,7 +1422,9 @@ static int loose_object_info(struct repository *r,
                             struct object_info *oi, int flags)
 {
        int status = 0;
+       int fd;
        unsigned long mapsize;
+       const char *path;
        void *map;
        git_zstream stream;
        char hdr[MAX_HEADER_LEN];
@@ -1448,7 +1445,6 @@ static int loose_object_info(struct repository *r,
         * object even exists.
         */
        if (!oi->typep && !oi->type_name && !oi->sizep && !oi->contentp) {
-               const char *path;
                struct stat st;
                if (!oi->disk_sizep && (flags & OBJECT_INFO_QUICK))
                        return quick_has_loose(r, oid) ? 0 : -1;
@@ -1459,7 +1455,13 @@ static int loose_object_info(struct repository *r,
                return 0;
        }
 
-       map = map_loose_object(r, oid, &mapsize);
+       fd = open_loose_object(r, oid, &path);
+       if (fd < 0) {
+               if (errno != ENOENT)
+                       error_errno(_("unable to open loose object %s"), oid_to_hex(oid));
+               return -1;
+       }
+       map = map_fd(fd, path, &mapsize);
        if (!map)
                return -1;
 
@@ -1497,6 +1499,10 @@ static int loose_object_info(struct repository *r,
                break;
        }
 
+       if (status && (flags & OBJECT_INFO_DIE_IF_CORRUPT))
+               die(_("loose object %s (stored in %s) is corrupt"),
+                   oid_to_hex(oid), path);
+
        git_inflate_end(&stream);
 cleanup:
        munmap(map, mapsize);
@@ -1575,9 +1581,6 @@ static int do_oid_object_info_extended(struct repository *r,
                if (find_pack_entry(r, real, &e))
                        break;
 
-               if (flags & OBJECT_INFO_IGNORE_LOOSE)
-                       return -1;
-
                /* Most likely it's a loose object. */
                if (!loose_object_info(r, real, oi, flags))
                        return 0;
@@ -1609,6 +1612,15 @@ static int do_oid_object_info_extended(struct repository *r,
                        continue;
                }
 
+               if (flags & OBJECT_INFO_DIE_IF_CORRUPT) {
+                       const struct packed_git *p;
+                       if ((flags & OBJECT_INFO_LOOKUP_REPLACE) && !oideq(real, oid))
+                               die(_("replacement %s not found for %s"),
+                                   oid_to_hex(real), oid_to_hex(oid));
+                       if ((p = has_packed_and_bad(r, real)))
+                               die(_("packed object %s (stored in %s) is corrupt"),
+                                   oid_to_hex(real), p->pack_name);
+               }
                return -1;
        }
 
@@ -1661,7 +1673,8 @@ int oid_object_info(struct repository *r,
 
 static void *read_object(struct repository *r,
                         const struct object_id *oid, enum object_type *type,
-                        unsigned long *size)
+                        unsigned long *size,
+                        int die_if_corrupt)
 {
        struct object_info oi = OBJECT_INFO_INIT;
        void *content;
@@ -1669,7 +1682,8 @@ static void *read_object(struct repository *r,
        oi.sizep = size;
        oi.contentp = &content;
 
-       if (oid_object_info_extended(r, oid, &oi, 0) < 0)
+       if (oid_object_info_extended(r, oid, &oi, die_if_corrupt
+                                    ? OBJECT_INFO_DIE_IF_CORRUPT : 0) < 0)
                return NULL;
        return content;
 }
@@ -1705,35 +1719,14 @@ void *read_object_file_extended(struct repository *r,
                                int lookup_replace)
 {
        void *data;
-       const struct packed_git *p;
-       const char *path;
-       struct stat st;
        const struct object_id *repl = lookup_replace ?
                lookup_replace_object(r, oid) : oid;
 
        errno = 0;
-       data = read_object(r, repl, type, size);
+       data = read_object(r, repl, type, size, 1);
        if (data)
                return data;
 
-       obj_read_lock();
-       if (errno && errno != ENOENT)
-               die_errno(_("failed to read object %s"), oid_to_hex(oid));
-
-       /* die if we replaced an object with one that does not exist */
-       if (repl != oid)
-               die(_("replacement %s not found for %s"),
-                   oid_to_hex(repl), oid_to_hex(oid));
-
-       if (!stat_loose_object(r, repl, &st, &path))
-               die(_("loose object %s (stored in %s) is corrupt"),
-                   oid_to_hex(repl), path);
-
-       if ((p = has_packed_and_bad(r, repl)))
-               die(_("packed object %s (stored in %s) is corrupt"),
-                   oid_to_hex(repl), p->pack_name);
-       obj_read_unlock();
-
        return NULL;
 }
 
@@ -1864,13 +1857,6 @@ out:
        return 0;
 }
 
-static int write_buffer(int fd, const void *buf, size_t len)
-{
-       if (write_in_full(fd, buf, len) < 0)
-               return error_errno(_("file write error"));
-       return 0;
-}
-
 static void hash_object_file_literally(const struct git_hash_algo *algo,
                                       const void *buf, unsigned long len,
                                       const char *type, struct object_id *oid)
@@ -2015,8 +2001,8 @@ static int write_loose_object_common(git_hash_ctx *c,
 
        ret = git_deflate(stream, flush ? Z_FINISH : 0);
        the_hash_algo->update_fn(c, in0, stream->next_in - in0);
-       if (write_buffer(fd, compressed, stream->next_out - compressed) < 0)
-               die(_("unable to write loose object file"));
+       if (write_in_full(fd, compressed, stream->next_out - compressed) < 0)
+               die_errno(_("unable to write loose object file"));
        stream->next_out = compressed;
        stream->avail_out = compressed_len;
 
@@ -2276,7 +2262,7 @@ int force_object_loose(const struct object_id *oid, time_t mtime)
 
        if (has_loose_object(oid))
                return 0;
-       buf = read_object(the_repository, oid, &type, &len);
+       buf = read_object(the_repository, oid, &type, &len, 0);
        if (!buf)
                return error(_("cannot read object for %s"), oid_to_hex(oid));
        hdrlen = format_object_header(hdr, sizeof(hdr), type, len);
@@ -2792,13 +2778,16 @@ int read_loose_object(const char *path,
                      struct object_info *oi)
 {
        int ret = -1;
+       int fd;
        void *map = NULL;
        unsigned long mapsize;
        git_zstream stream;
        char hdr[MAX_HEADER_LEN];
        unsigned long *size = oi->sizep;
 
-       map = map_loose_object_1(the_repository, path, NULL, &mapsize);
+       fd = git_open(path);
+       if (fd >= 0)
+               map = map_fd(fd, path, &mapsize);
        if (!map) {
                error_errno(_("unable to mmap %s"), path);
                goto out;
index 1be57abaf10d7aa527df7f22ebaf660f6430d4d8..98c1d67946f7351120b1389f3ed124720e901a3a 100644 (file)
@@ -434,19 +434,20 @@ struct object_info {
 #define OBJECT_INFO_ALLOW_UNKNOWN_TYPE 2
 /* Do not retry packed storage after checking packed and loose storage */
 #define OBJECT_INFO_QUICK 8
-/* Do not check loose object */
-#define OBJECT_INFO_IGNORE_LOOSE 16
 /*
  * Do not attempt to fetch the object if missing (even if fetch_is_missing is
  * nonzero).
  */
-#define OBJECT_INFO_SKIP_FETCH_OBJECT 32
+#define OBJECT_INFO_SKIP_FETCH_OBJECT 16
 /*
  * This is meant for bulk prefetching of missing blobs in a partial
  * clone. Implies OBJECT_INFO_SKIP_FETCH_OBJECT and OBJECT_INFO_QUICK
  */
 #define OBJECT_INFO_FOR_PREFETCH (OBJECT_INFO_SKIP_FETCH_OBJECT | OBJECT_INFO_QUICK)
 
+/* Die if object corruption (not just an object being missing) was detected. */
+#define OBJECT_INFO_DIE_IF_CORRUPT 32
+
 int oid_object_info_extended(struct repository *r,
                             const struct object_id *,
                             struct object_info *, unsigned flags);
index 682b852a46c1aa89c85c24c66660a1948b5c2184..344087de4de5184fcbfb599277180cec8f39c474 100644 (file)
--- a/object.c
+++ b/object.c
@@ -212,8 +212,7 @@ struct object *parse_object_buffer(struct repository *r, const struct object_id
        if (type == OBJ_BLOB) {
                struct blob *blob = lookup_blob(r, oid);
                if (blob) {
-                       if (parse_blob_buffer(blob, buffer, size))
-                               return NULL;
+                       parse_blob_buffer(blob);
                        obj = &blob->object;
                }
        } else if (type == OBJ_TREE) {
@@ -292,7 +291,7 @@ struct object *parse_object_with_flags(struct repository *r,
                        error(_("hash mismatch %s"), oid_to_hex(oid));
                        return NULL;
                }
-               parse_blob_buffer(lookup_blob(r, oid), NULL, 0);
+               parse_blob_buffer(lookup_blob(r, oid));
                return lookup_object(r, oid);
        }
 
index 440407f1be742c7e2a19a62f5adb968d409bcc29..d2a42abf28cc7246b3d076b1c855a350edcded70 100644 (file)
@@ -354,8 +354,8 @@ static int open_midx_bitmap_1(struct bitmap_index *bitmap_git,
        if (bitmap_git->pack || bitmap_git->midx) {
                struct strbuf buf = STRBUF_INIT;
                get_midx_filename(&buf, midx->object_dir);
-               /* ignore extra bitmap file; we can only handle one */
-               warning(_("ignoring extra bitmap file: '%s'"), buf.buf);
+               trace2_data_string("bitmap", the_repository,
+                                  "ignoring extra midx bitmap file", buf.buf);
                close(fd);
                strbuf_release(&buf);
                return -1;
@@ -411,9 +411,6 @@ static int open_pack_bitmap_1(struct bitmap_index *bitmap_git, struct packed_git
        struct stat st;
        char *bitmap_name;
 
-       if (open_pack_index(packfile))
-               return -1;
-
        bitmap_name = pack_bitmap_filename(packfile);
        fd = git_open(bitmap_name);
 
@@ -432,8 +429,8 @@ static int open_pack_bitmap_1(struct bitmap_index *bitmap_git, struct packed_git
        }
 
        if (bitmap_git->pack || bitmap_git->midx) {
-               /* ignore extra bitmap file; we can only handle one */
-               warning(_("ignoring extra bitmap file: '%s'"), packfile->pack_name);
+               trace2_data_string("bitmap", the_repository,
+                                  "ignoring extra bitmap file", packfile->pack_name);
                close(fd);
                return -1;
        }
@@ -458,6 +455,8 @@ static int open_pack_bitmap_1(struct bitmap_index *bitmap_git, struct packed_git
                return -1;
        }
 
+       trace2_data_string("bitmap", the_repository, "opened bitmap file",
+                          packfile->pack_name);
        return 0;
 }
 
@@ -525,11 +524,16 @@ static int open_pack_bitmap(struct repository *r,
        struct packed_git *p;
        int ret = -1;
 
-       assert(!bitmap_git->map);
-
        for (p = get_all_packs(r); p; p = p->next) {
-               if (open_pack_bitmap_1(bitmap_git, p) == 0)
+               if (open_pack_bitmap_1(bitmap_git, p) == 0) {
                        ret = 0;
+                       /*
+                        * The only reason to keep looking is to report
+                        * duplicates.
+                        */
+                       if (!trace2_is_enabled())
+                               break;
+               }
        }
 
        return ret;
@@ -553,11 +557,20 @@ static int open_midx_bitmap(struct repository *r,
 static int open_bitmap(struct repository *r,
                       struct bitmap_index *bitmap_git)
 {
+       int found;
+
        assert(!bitmap_git->map);
 
-       if (!open_midx_bitmap(r, bitmap_git))
-               return 0;
-       return open_pack_bitmap(r, bitmap_git);
+       found = !open_midx_bitmap(r, bitmap_git);
+
+       /*
+        * these will all be skipped if we opened a midx bitmap; but run it
+        * anyway if tracing is enabled to report the duplicates
+        */
+       if (!found || trace2_is_enabled())
+               found |= !open_pack_bitmap(r, bitmap_git);
+
+       return found ? 0 : -1;
 }
 
 struct bitmap_index *prepare_bitmap_git(struct repository *r)
index 00787e306db4c010ae68845f27ed69966fb9e250..336372974872882a235598d254c876e31c49b24a 100644 (file)
@@ -5,7 +5,6 @@
 #include "chunk-format.h"
 #include "pack-mtimes.h"
 #include "oidmap.h"
-#include "chunk-format.h"
 #include "pack-objects.h"
 
 void reset_pack_idx_option(struct pack_idx_option *opts)
index b6ef86e0d15e3dc36364977dd3515fa118dee165..50d852f2991738c951bb447ce8206640aabb8a80 100644 (file)
@@ -369,6 +369,10 @@ int parse_opt_tracking_mode(const struct option *, const char *, int);
        { OPTION_CALLBACK, 0, "abbrev", (var), N_("n"), \
          N_("use <n> digits to display object names"), \
          PARSE_OPT_OPTARG, &parse_opt_abbrev_cb, 0 }
+#define OPT__SUPER_PREFIX(var) \
+       OPT_STRING_F(0, "super-prefix", (var), N_("prefix"), \
+               N_("prefixed path to initial superproject"), PARSE_OPT_HIDDEN)
+
 #define OPT__COLOR(var, h) \
        OPT_COLOR_FLAG(0, "color", (var), (h))
 #define OPT_COLUMN(s, l, v, h) \
index 8b7d81adc1be09d91a1551c370c08058766f76f6..8255ab4349c009bec7c2c7200365d2bd9b018c16 100644 (file)
@@ -269,14 +269,18 @@ static void find_exact_matches(struct string_list *a, struct string_list *b)
        hashmap_clear(&map);
 }
 
-static int diffsize_consume(void *data, char *line, unsigned long len)
+static int diffsize_consume(void *data,
+                            char *line UNUSED,
+                            unsigned long len UNUSED)
 {
        (*(int *)data)++;
        return 0;
 }
 
-static void diffsize_hunk(void *data, long ob, long on, long nb, long nn,
-                         const char *funcline, long funclen)
+static void diffsize_hunk(void *data,
+                         long ob UNUSED, long on UNUSED,
+                         long nb UNUSED, long nn UNUSED,
+                         const char *func UNUSED, long funclen UNUSED)
 {
        diffsize_consume(data, NULL, 0);
 }
@@ -461,7 +465,7 @@ static void patch_diff(const char *a, const char *b,
        diff_flush(diffopt);
 }
 
-static struct strbuf *output_prefix_cb(struct diff_options *opt, void *data)
+static struct strbuf *output_prefix_cb(struct diff_options *opt UNUSED, void *data)
 {
        return data;
 }
index feefa0f68ba74b7ffddd2ed6cf93c7991803728f..cf87ad7097021c9764d2faeb7c97fc8bfef91a22 100644 (file)
@@ -2566,6 +2566,11 @@ void discard_index(struct index_state *istate)
        free_untracked_cache(istate->untracked);
        istate->untracked = NULL;
 
+       if (istate->sparse_checkout_patterns) {
+               clear_pattern_list(istate->sparse_checkout_patterns);
+               FREE_AND_NULL(istate->sparse_checkout_patterns);
+       }
+
        if (istate->ce_mem_pool) {
                mem_pool_discard(istate->ce_mem_pool, should_validate_cache_entries());
                FREE_AND_NULL(istate->ce_mem_pool);
index 9dc2cd14519d07fb976a3007c1d8d47fa9a9a549..a24324123e727b762e2df915802920728ee8f63d 100644 (file)
@@ -228,6 +228,22 @@ static int strbuf_addf_ret(struct strbuf *sb, int ret, const char *fmt, ...)
        return ret;
 }
 
+static int err_no_arg(struct strbuf *sb, const char *name)
+{
+       size_t namelen = strchrnul(name, ':') - name;
+       strbuf_addf(sb, _("%%(%.*s) does not take arguments"),
+                   (int)namelen, name);
+       return -1;
+}
+
+static int err_bad_arg(struct strbuf *sb, const char *name, const char *arg)
+{
+       size_t namelen = strchrnul(name, ':') - name;
+       strbuf_addf(sb, _("unrecognized %%(%.*s) argument: %s"),
+                   (int)namelen, name, arg);
+       return -1;
+}
+
 static int color_atom_parser(struct ref_format *format, struct used_atom *atom,
                             const char *color_value, struct strbuf *err)
 {
@@ -262,7 +278,7 @@ static int refname_atom_parser_internal(struct refname_atom *atom, const char *a
                if (strtol_i(arg, 10, &atom->rstrip))
                        return strbuf_addf_ret(err, -1, _("Integer value expected refname:rstrip=%s"), arg);
        } else
-               return strbuf_addf_ret(err, -1, _("unrecognized %%(%s) argument: %s"), name, arg);
+               return err_bad_arg(err, name, arg);
        return 0;
 }
 
@@ -317,7 +333,7 @@ static int objecttype_atom_parser(struct ref_format *format, struct used_atom *a
                                  const char *arg, struct strbuf *err)
 {
        if (arg)
-               return strbuf_addf_ret(err, -1, _("%%(objecttype) does not take arguments"));
+               return err_no_arg(err, "objecttype");
        if (*atom->name == '*')
                oi_deref.info.typep = &oi_deref.type;
        else
@@ -341,7 +357,7 @@ static int objectsize_atom_parser(struct ref_format *format, struct used_atom *a
                else
                        oi.info.disk_sizep = &oi.disk_size;
        } else
-               return strbuf_addf_ret(err, -1, _("unrecognized %%(%s) argument: %s"), "objectsize", arg);
+               return err_bad_arg(err, "objectsize", arg);
        return 0;
 }
 
@@ -349,7 +365,7 @@ static int deltabase_atom_parser(struct ref_format *format, struct used_atom *at
                                 const char *arg, struct strbuf *err)
 {
        if (arg)
-               return strbuf_addf_ret(err, -1, _("%%(deltabase) does not take arguments"));
+               return err_no_arg(err, "deltabase");
        if (*atom->name == '*')
                oi_deref.info.delta_base_oid = &oi_deref.delta_base_oid;
        else
@@ -361,7 +377,7 @@ static int body_atom_parser(struct ref_format *format, struct used_atom *atom,
                            const char *arg, struct strbuf *err)
 {
        if (arg)
-               return strbuf_addf_ret(err, -1, _("%%(body) does not take arguments"));
+               return err_no_arg(err, "body");
        atom->u.contents.option = C_BODY_DEP;
        return 0;
 }
@@ -374,7 +390,7 @@ static int subject_atom_parser(struct ref_format *format, struct used_atom *atom
        else if (!strcmp(arg, "sanitize"))
                atom->u.contents.option = C_SUB_SANITIZE;
        else
-               return strbuf_addf_ret(err, -1, _("unrecognized %%(%s) argument: %s"), "subject", arg);
+               return err_bad_arg(err, "subject", arg);
        return 0;
 }
 
@@ -428,7 +444,7 @@ static int contents_atom_parser(struct ref_format *format, struct used_atom *ato
                if (strtoul_ui(arg, 10, &atom->u.contents.nlines))
                        return strbuf_addf_ret(err, -1, _("positive value expected contents:lines=%s"), arg);
        } else
-               return strbuf_addf_ret(err, -1, _("unrecognized %%(%s) argument: %s"), "contents", arg);
+               return err_bad_arg(err, "contents", arg);
        return 0;
 }
 
@@ -440,7 +456,7 @@ static int raw_atom_parser(struct ref_format *format, struct used_atom *atom,
        else if (!strcmp(arg, "size"))
                atom->u.raw_data.option = RAW_LENGTH;
        else
-               return strbuf_addf_ret(err, -1, _("unrecognized %%(%s) argument: %s"), "raw", arg);
+               return err_bad_arg(err, "raw", arg);
        return 0;
 }
 
@@ -459,7 +475,7 @@ static int oid_atom_parser(struct ref_format *format, struct used_atom *atom,
                if (atom->u.oid.length < MINIMUM_ABBREV)
                        atom->u.oid.length = MINIMUM_ABBREV;
        } else
-               return strbuf_addf_ret(err, -1, _("unrecognized %%(%s) argument: %s"), atom->name, arg);
+               return err_bad_arg(err, atom->name, arg);
        return 0;
 }
 
@@ -473,7 +489,7 @@ static int person_email_atom_parser(struct ref_format *format, struct used_atom
        else if (!strcmp(arg, "localpart"))
                atom->u.email_option.option = EO_LOCALPART;
        else
-               return strbuf_addf_ret(err, -1, _("unrecognized email option: %s"), arg);
+               return err_bad_arg(err, atom->name, arg);
        return 0;
 }
 
@@ -557,7 +573,7 @@ static int if_atom_parser(struct ref_format *format, struct used_atom *atom,
        } else if (skip_prefix(arg, "notequals=", &atom->u.if_then_else.str)) {
                atom->u.if_then_else.cmp_status = COMPARE_UNEQUAL;
        } else
-               return strbuf_addf_ret(err, -1, _("unrecognized %%(%s) argument: %s"), "if", arg);
+               return err_bad_arg(err, "if", arg);
        return 0;
 }
 
@@ -565,14 +581,16 @@ static int rest_atom_parser(struct ref_format *format, struct used_atom *atom,
                            const char *arg, struct strbuf *err)
 {
        if (arg)
-               return strbuf_addf_ret(err, -1, _("%%(rest) does not take arguments"));
+               return err_no_arg(err, "rest");
        format->use_rest = 1;
        return 0;
 }
 
 static int head_atom_parser(struct ref_format *format, struct used_atom *atom,
-                           const char *arg, struct strbuf *unused_err)
+                           const char *arg, struct strbuf *err)
 {
+       if (arg)
+               return err_no_arg(err, "HEAD");
        atom->u.head = resolve_refdup("HEAD", RESOLVE_REF_READING, NULL, NULL);
        return 0;
 }
@@ -1358,6 +1376,7 @@ static void find_subpos(const char *buf,
 
        /* parse signature first; we might not even have a subject line */
        parse_signature(buf, end - buf, &payload, &signature);
+       strbuf_release(&payload);
 
        /* skip past header until we hit empty line */
        while (*buf && *buf != '\n') {
@@ -2128,8 +2147,9 @@ static int for_each_fullref_in_pattern(struct ref_filter *filter,
                return for_each_fullref_in("", cb, cb_data);
        }
 
-       return for_each_fullref_in_prefixes(NULL, filter->name_patterns,
-                                           cb, cb_data);
+       return refs_for_each_fullref_in_prefixes(get_main_ref_store(the_repository),
+                                                NULL, filter->name_patterns,
+                                                cb, cb_data);
 }
 
 /*
index 78e9350e20526a279fd5f5cedfb3d5a726edd481..04630f56ec6b3661ecb027fa0a8a5f49155bb3da 100644 (file)
--- a/reflog.c
+++ b/reflog.c
@@ -193,7 +193,6 @@ static void mark_reachable(struct expire_reflog_policy_cb *cb)
                        commit_list_insert(commit, &leftover);
                        continue;
                }
-               commit->object.flags |= REACHABLE;
                parent = commit->parents;
                while (parent) {
                        commit = parent->item;
@@ -371,6 +370,9 @@ void reflog_expiry_cleanup(void *cb_data)
                clear_commit_marks(cb->tip_commit, REACHABLE);
                break;
        }
+       for (elem = cb->mark_list; elem; elem = elem->next)
+               clear_commit_marks(elem->item, REACHABLE);
+       free_commit_list(cb->mark_list);
 }
 
 int count_reflog_ent(struct object_id *ooid UNUSED,
diff --git a/refs.c b/refs.c
index 2c7e88b1902b8d35b63808d781f4076d5567560f..e31dbcda5990b8f2e7143547ce134b7218041fd2 100644 (file)
--- a/refs.c
+++ b/refs.c
@@ -1723,9 +1723,10 @@ static void find_longest_prefixes(struct string_list *out,
        strbuf_release(&prefix);
 }
 
-int for_each_fullref_in_prefixes(const char *namespace,
-                                const char **patterns,
-                                each_ref_fn fn, void *cb_data)
+int refs_for_each_fullref_in_prefixes(struct ref_store *ref_store,
+                                     const char *namespace,
+                                     const char **patterns,
+                                     each_ref_fn fn, void *cb_data)
 {
        struct string_list prefixes = STRING_LIST_INIT_DUP;
        struct string_list_item *prefix;
@@ -1740,7 +1741,7 @@ int for_each_fullref_in_prefixes(const char *namespace,
 
        for_each_string_list_item(prefix, &prefixes) {
                strbuf_addstr(&buf, prefix->string);
-               ret = for_each_fullref_in(buf.buf, fn, cb_data);
+               ret = refs_for_each_fullref_in(ref_store, buf.buf, fn, cb_data);
                if (ret)
                        break;
                strbuf_setlen(&buf, namespace_len);
diff --git a/refs.h b/refs.h
index 3266fd8f57d39938c2c496c8f869b14697c34c69..935cdd1ece3db3af483cb2225f204966c0e6cfa5 100644 (file)
--- a/refs.h
+++ b/refs.h
@@ -354,8 +354,10 @@ int for_each_fullref_in(const char *prefix, each_ref_fn fn, void *cb_data);
  *
  * callers should be prepared to ignore references that they did not ask for.
  */
-int for_each_fullref_in_prefixes(const char *namespace, const char **patterns,
-                                each_ref_fn fn, void *cb_data);
+int refs_for_each_fullref_in_prefixes(struct ref_store *refs,
+                                     const char *namespace, const char **patterns,
+                                     each_ref_fn fn, void *cb_data);
+
 /**
  * iterate refs from the respective area.
  */
index c1c71d183ea3c0baf082b16cf18bc373a7615fe3..6f5a0709fba65e80081fdb04563019e1f1e15400 100644 (file)
@@ -1263,7 +1263,8 @@ static int write_with_updates(struct packed_ref_store *refs,
                goto error;
        }
 
-       if (fsync_component(FSYNC_COMPONENT_REFERENCE, get_tempfile_fd(refs->tempfile)) ||
+       if (fflush(out) ||
+           fsync_component(FSYNC_COMPONENT_REFERENCE, get_tempfile_fd(refs->tempfile)) ||
            close_tempfile_gently(refs->tempfile)) {
                strbuf_addf(err, "error closing file %s: %s",
                            get_tempfile_path(refs->tempfile),
index 1c4621b414bdc0372613e58c3b2133b84004d1e7..1ebbe42792ec67230b62dbbe4cb8ed9646576858 100644 (file)
--- a/remote.h
+++ b/remote.h
@@ -234,6 +234,11 @@ struct ref **get_remote_refs(int fd_out, struct packet_reader *reader,
                             const struct string_list *server_options,
                             int stateless_rpc);
 
+/* Used for protocol v2 in order to retrieve refs from a remote */
+struct bundle_list;
+int get_remote_bundle_uri(int fd_out, struct packet_reader *reader,
+                         struct bundle_list *bundles, int stateless_rpc);
+
 int resolve_remote_symref(struct ref *ref, struct ref *list);
 
 /*
index 439e34a7c51803335c4f5677f67058b633c46339..100e5ad5110b1f851801c11639434da7191a229e 100644 (file)
@@ -600,10 +600,12 @@ static struct commit *one_relevant_parent(const struct rev_info *revs,
 static int tree_difference = REV_TREE_SAME;
 
 static void file_add_remove(struct diff_options *options,
-                   int addremove, unsigned mode,
-                   const struct object_id *oid,
-                   int oid_valid,
-                   const char *fullpath, unsigned dirty_submodule)
+                   int addremove,
+                   unsigned mode UNUSED,
+                   const struct object_id *oid UNUSED,
+                   int oid_valid UNUSED,
+                   const char *fullpath UNUSED,
+                   unsigned dirty_submodule UNUSED)
 {
        int diff = addremove == '+' ? REV_TREE_NEW : REV_TREE_OLD;
        struct rev_info *revs = options->change_fn_data;
@@ -614,12 +616,15 @@ static void file_add_remove(struct diff_options *options,
 }
 
 static void file_change(struct diff_options *options,
-                unsigned old_mode, unsigned new_mode,
-                const struct object_id *old_oid,
-                const struct object_id *new_oid,
-                int old_oid_valid, int new_oid_valid,
-                const char *fullpath,
-                unsigned old_dirty_submodule, unsigned new_dirty_submodule)
+                unsigned old_mode UNUSED,
+                unsigned new_mode UNUSED,
+                const struct object_id *old_oid UNUSED,
+                const struct object_id *new_oid UNUSED,
+                int old_oid_valid UNUSED,
+                int new_oid_valid UNUSED,
+                const char *fullpath UNUSED,
+                unsigned old_dirty_submodule UNUSED,
+                unsigned new_dirty_submodule UNUSED)
 {
        tree_difference = REV_TREE_DIFFERENT;
        options->flags.has_changes = 1;
@@ -3054,6 +3059,7 @@ void release_revisions(struct rev_info *revs)
        date_mode_release(&revs->date_mode);
        release_revisions_mailmap(revs->mailmap);
        free_grep_patterns(&revs->grep_filter);
+       graph_clear(revs->graph);
        /* TODO (need to handle "no_free"): diff_free(&revs->diffopt) */
        diff_free(&revs->pruning);
        reflog_walk_info_release(revs->reflog_info);
index 48b9ba6d6f077e5729f3efc4c1a88ab2be508c36..756f1839aab22b9c79337400a7424d0243088368 100644 (file)
@@ -1019,7 +1019,7 @@ static void *run_thread(void *data)
                sigset_t mask;
                sigemptyset(&mask);
                sigaddset(&mask, SIGPIPE);
-               if (pthread_sigmask(SIG_BLOCK, &mask, NULL) < 0) {
+               if (pthread_sigmask(SIG_BLOCK, &mask, NULL)) {
                        ret = error("unable to block SIGPIPE in async thread");
                        return (void *)ret;
                }
index 9827e1adcbf0de181b7e3eea9a411d60effc2124..bcb662e23be9e8d2962a88bd16b4cad93981619d 100644 (file)
@@ -36,7 +36,6 @@
 #include "rebase-interactive.h"
 #include "reset.h"
 #include "branch.h"
-#include "log-tree.h"
 
 #define GIT_REFLOG_ACTION "GIT_REFLOG_ACTION"
 
@@ -2897,6 +2896,7 @@ static void read_strategy_opts(struct replay_opts *opts, struct strbuf *buf)
        strbuf_reset(buf);
        if (!read_oneliner(buf, rebase_path_strategy(), 0))
                return;
+       free(opts->strategy);
        opts->strategy = strbuf_detach(buf, NULL);
        if (!read_oneliner(buf, rebase_path_strategy_opts(), 0))
                return;
diff --git a/serve.c b/serve.c
index 733347f602aa1ede3e45fee94050163790d65bf9..cbf4a143cfea9bd450d5d033b1ec051025af8a1f 100644 (file)
--- a/serve.c
+++ b/serve.c
@@ -7,6 +7,7 @@
 #include "protocol-caps.h"
 #include "serve.h"
 #include "upload-pack.h"
+#include "bundle-uri.h"
 
 static int advertise_sid = -1;
 static int client_hash_algo = GIT_HASH_SHA1;
@@ -135,6 +136,11 @@ static struct protocol_capability capabilities[] = {
                .advertise = always_advertise,
                .command = cap_object_info,
        },
+       {
+               .name = "bundle-uri",
+               .advertise = bundle_uri_advertise,
+               .command = bundle_uri_command,
+       },
 };
 
 void protocol_v2_advertise_capabilities(void)
index 0890b1405c5cc6888396bac8558549ebdb2e2977..c383f41a3c5ccc6c6228c4ed4f9631529633f878 100644 (file)
--- a/strbuf.c
+++ b/strbuf.c
@@ -1200,3 +1200,9 @@ int strbuf_edit_interactively(struct strbuf *buffer, const char *path,
        free(path2);
        return res;
 }
+
+void strbuf_strip_file_from_path(struct strbuf *sb)
+{
+       char *path_sep = find_last_dir_sep(sb->buf);
+       strbuf_setlen(sb, path_sep ? path_sep - sb->buf + 1 : 0);
+}
index 76965a17d444829b4a5b5edc099d12bf1fcc7ea3..f6dbb9681ee768221e707a39c37dc0a00f63b95d 100644 (file)
--- a/strbuf.h
+++ b/strbuf.h
@@ -664,6 +664,17 @@ int launch_sequence_editor(const char *path, struct strbuf *buffer,
 int strbuf_edit_interactively(struct strbuf *buffer, const char *path,
                              const char *const *env);
 
+/*
+ * Remove the filename from the provided path string. If the path
+ * contains a trailing separator, then the path is considered a directory
+ * and nothing is modified.
+ *
+ * Examples:
+ * - "/path/to/file" -> "/path/to/"
+ * - "/path/to/dir/" -> "/path/to/dir/"
+ */
+void strbuf_strip_file_from_path(struct strbuf *sb);
+
 void strbuf_add_lines(struct strbuf *sb,
                      const char *prefix,
                      const char *buf,
index 8ac2fca855d6c3615be2e5b8a7f16a4746441e1a..3a0dfc417c05b627a5100f518768a1b32ee22339 100644 (file)
@@ -832,7 +832,7 @@ static void changed_submodule_data_clear(struct changed_submodule_data *cs_data)
 }
 
 static void collect_changed_submodules_cb(struct diff_queue_struct *q,
-                                         struct diff_options *options,
+                                         struct diff_options *options UNUSED,
                                          void *data)
 {
        struct collect_changed_submodules_cb_data *me = data;
@@ -2054,14 +2054,6 @@ void submodule_unset_core_worktree(const struct submodule *sub)
        strbuf_release(&config_path);
 }
 
-static const char *get_super_prefix_or_empty(void)
-{
-       const char *s = get_super_prefix();
-       if (!s)
-               s = "";
-       return s;
-}
-
 static int submodule_has_dirty_index(const struct submodule *sub)
 {
        struct child_process cp = CHILD_PROCESS_INIT;
@@ -2080,7 +2072,7 @@ static int submodule_has_dirty_index(const struct submodule *sub)
        return finish_command(&cp);
 }
 
-static void submodule_reset_index(const char *path)
+static void submodule_reset_index(const char *path, const char *super_prefix)
 {
        struct child_process cp = CHILD_PROCESS_INIT;
        prepare_submodule_repo_env(&cp.env);
@@ -2089,10 +2081,10 @@ static void submodule_reset_index(const char *path)
        cp.no_stdin = 1;
        cp.dir = path;
 
-       strvec_pushf(&cp.args, "--super-prefix=%s%s/",
-                    get_super_prefix_or_empty(), path);
        /* TODO: determine if this might overwright untracked files */
        strvec_pushl(&cp.args, "read-tree", "-u", "--reset", NULL);
+       strvec_pushf(&cp.args, "--super-prefix=%s%s/",
+                    (super_prefix ? super_prefix : ""), path);
 
        strvec_push(&cp.args, empty_tree_oid_hex());
 
@@ -2105,10 +2097,9 @@ static void submodule_reset_index(const char *path)
  * For edge cases (a submodule coming into existence or removing a submodule)
  * pass NULL for old or new respectively.
  */
-int submodule_move_head(const char *path,
-                        const char *old_head,
-                        const char *new_head,
-                        unsigned flags)
+int submodule_move_head(const char *path, const char *super_prefix,
+                       const char *old_head, const char *new_head,
+                       unsigned flags)
 {
        int ret = 0;
        struct child_process cp = CHILD_PROCESS_INIT;
@@ -2145,7 +2136,8 @@ int submodule_move_head(const char *path,
        if (!(flags & SUBMODULE_MOVE_HEAD_DRY_RUN)) {
                if (old_head) {
                        if (!submodule_uses_gitfile(path))
-                               absorb_git_dir_into_superproject(path);
+                               absorb_git_dir_into_superproject(path,
+                                                                super_prefix);
                } else {
                        struct strbuf gitdir = STRBUF_INIT;
                        submodule_name_to_gitdir(&gitdir, the_repository,
@@ -2154,7 +2146,7 @@ int submodule_move_head(const char *path,
                        strbuf_release(&gitdir);
 
                        /* make sure the index is clean as well */
-                       submodule_reset_index(path);
+                       submodule_reset_index(path, super_prefix);
                }
 
                if (old_head && (flags & SUBMODULE_MOVE_HEAD_FORCE)) {
@@ -2172,9 +2164,9 @@ int submodule_move_head(const char *path,
        cp.no_stdin = 1;
        cp.dir = path;
 
-       strvec_pushf(&cp.args, "--super-prefix=%s%s/",
-                    get_super_prefix_or_empty(), path);
        strvec_pushl(&cp.args, "read-tree", "--recurse-submodules", NULL);
+       strvec_pushf(&cp.args, "--super-prefix=%s%s/",
+                    (super_prefix ? super_prefix : ""), path);
 
        if (flags & SUBMODULE_MOVE_HEAD_DRY_RUN)
                strvec_push(&cp.args, "-n");
@@ -2274,7 +2266,8 @@ int validate_submodule_git_dir(char *git_dir, const char *submodule_name)
  * Embeds a single submodules git directory into the superprojects git dir,
  * non recursively.
  */
-static void relocate_single_git_dir_into_superproject(const char *path)
+static void relocate_single_git_dir_into_superproject(const char *path,
+                                                     const char *super_prefix)
 {
        char *old_git_dir = NULL, *real_old_git_dir = NULL, *real_new_git_dir = NULL;
        struct strbuf new_gitdir = STRBUF_INIT;
@@ -2304,7 +2297,7 @@ static void relocate_single_git_dir_into_superproject(const char *path)
        real_new_git_dir = real_pathdup(new_gitdir.buf, 1);
 
        fprintf(stderr, _("Migrating git directory of '%s%s' from\n'%s' to\n'%s'\n"),
-               get_super_prefix_or_empty(), path,
+               super_prefix ? super_prefix : "", path,
                real_old_git_dir, real_new_git_dir);
 
        relocate_gitdir(path, real_old_git_dir, real_new_git_dir);
@@ -2315,7 +2308,8 @@ static void relocate_single_git_dir_into_superproject(const char *path)
        strbuf_release(&new_gitdir);
 }
 
-static void absorb_git_dir_into_superproject_recurse(const char *path)
+static void absorb_git_dir_into_superproject_recurse(const char *path,
+                                                    const char *super_prefix)
 {
 
        struct child_process cp = CHILD_PROCESS_INIT;
@@ -2323,10 +2317,11 @@ static void absorb_git_dir_into_superproject_recurse(const char *path)
        cp.dir = path;
        cp.git_cmd = 1;
        cp.no_stdin = 1;
-       strvec_pushf(&cp.args, "--super-prefix=%s%s/",
-                    get_super_prefix_or_empty(), path);
        strvec_pushl(&cp.args, "submodule--helper",
                     "absorbgitdirs", NULL);
+       strvec_pushf(&cp.args, "--super-prefix=%s%s/", super_prefix ?
+                    super_prefix : "", path);
+
        prepare_submodule_repo_env(&cp.env);
        if (run_command(&cp))
                die(_("could not recurse into submodule '%s'"), path);
@@ -2337,7 +2332,8 @@ static void absorb_git_dir_into_superproject_recurse(const char *path)
  * having its git directory within the working tree to the git dir nested
  * in its superprojects git dir under modules/.
  */
-void absorb_git_dir_into_superproject(const char *path)
+void absorb_git_dir_into_superproject(const char *path,
+                                     const char *super_prefix)
 {
        int err_code;
        const char *sub_git_dir;
@@ -2379,14 +2375,14 @@ void absorb_git_dir_into_superproject(const char *path)
                char *real_common_git_dir = real_pathdup(get_git_common_dir(), 1);
 
                if (!starts_with(real_sub_git_dir, real_common_git_dir))
-                       relocate_single_git_dir_into_superproject(path);
+                       relocate_single_git_dir_into_superproject(path, super_prefix);
 
                free(real_sub_git_dir);
                free(real_common_git_dir);
        }
        strbuf_release(&gitdir);
 
-       absorb_git_dir_into_superproject_recurse(path);
+       absorb_git_dir_into_superproject_recurse(path, super_prefix);
 }
 
 int get_superproject_working_tree(struct strbuf *buf)
index b52a4ff1e73e3b137b7cd1a01e420c7f302497e6..c55a25ca37d2425d398f59c98ff177c8dfa6fdaf 100644 (file)
@@ -150,9 +150,8 @@ int validate_submodule_git_dir(char *git_dir, const char *submodule_name);
 
 #define SUBMODULE_MOVE_HEAD_DRY_RUN (1<<0)
 #define SUBMODULE_MOVE_HEAD_FORCE   (1<<1)
-int submodule_move_head(const char *path,
-                       const char *old,
-                       const char *new_head,
+int submodule_move_head(const char *path, const char *super_prefix,
+                       const char *old_head, const char *new_head,
                        unsigned flags);
 
 void submodule_unset_core_worktree(const struct submodule *sub);
@@ -164,7 +163,8 @@ void submodule_unset_core_worktree(const struct submodule *sub);
  */
 void prepare_submodule_repo_env(struct strvec *env);
 
-void absorb_git_dir_into_superproject(const char *path);
+void absorb_git_dir_into_superproject(const char *path,
+                                     const char *super_prefix);
 
 /*
  * Return the absolute path of the working tree of the superproject, which this
index 25afd3934284ddc8bc0fdca7cca514a742e44e7e..b18e7603103817b931acf6ab2a8cbbae8538bb8e 100644 (file)
@@ -3,6 +3,10 @@
 #include "bundle-uri.h"
 #include "strbuf.h"
 #include "string-list.h"
+#include "transport.h"
+#include "ref-filter.h"
+#include "remote.h"
+#include "refs.h"
 
 enum input_mode {
        KEY_VALUE_PAIRS,
@@ -36,6 +40,8 @@ static int cmd__bundle_uri_parse(int argc, const char **argv, enum input_mode mo
 
        init_bundle_list(&list);
 
+       list.baseURI = xstrdup("<uri>");
+
        switch (mode) {
        case KEY_VALUE_PAIRS:
                if (argc != 1)
@@ -68,6 +74,39 @@ usage:
        usage_with_options(usage, options);
 }
 
+static int cmd_ls_remote(int argc, const char **argv)
+{
+       const char *dest;
+       struct remote *remote;
+       struct transport *transport;
+       int status = 0;
+
+       dest = argc > 1 ? argv[1] : NULL;
+
+       remote = remote_get(dest);
+       if (!remote) {
+               if (dest)
+                       die(_("bad repository '%s'"), dest);
+               die(_("no remote configured to get bundle URIs from"));
+       }
+       if (!remote->url_nr)
+               die(_("remote '%s' has no configured URL"), dest);
+
+       transport = transport_get(remote, NULL);
+       if (transport_get_remote_bundle_uri(transport) < 0) {
+               error(_("could not get the bundle-uri list"));
+               status = 1;
+               goto cleanup;
+       }
+
+       print_bundle_list(stdout, transport->bundles);
+
+cleanup:
+       if (transport_disconnect(transport))
+               return 1;
+       return status;
+}
+
 int cmd__bundle_uri(int argc, const char **argv)
 {
        const char *usage[] = {
@@ -88,6 +127,8 @@ int cmd__bundle_uri(int argc, const char **argv)
                return cmd__bundle_uri_parse(argc - 1, argv + 1, KEY_VALUE_PAIRS);
        if (!strcmp(argv[1], "parse-config"))
                return cmd__bundle_uri_parse(argc - 1, argv + 1, CONFIG_FILE);
+       if (!strcmp(argv[1], "ls-remote"))
+               return cmd_ls_remote(argc - 1, argv + 1);
        error("there is no test-tool bundle-uri tool '%s'", argv[1]);
 
 usage:
index 2e576bcc11e9a62dc01206d46e01574ad47193d4..27323cb3672c35152e83e8741585a6c9520159f0 100644 (file)
@@ -17,6 +17,7 @@ int cmd_main(int argc, const char **argv)
        f = fopen(buf.buf, "w");
        if (!f)
                die("Could not write to %s", buf.buf);
+       strbuf_release(&buf);
        for (i = 0; i < argc; i++)
                fprintf(f, "%s%s", i > 0 ? " " : "", i > 0 ? argv[i] : "ssh:");
        fprintf(f, "\n");
diff --git a/t/lib-bundle-uri-protocol.sh b/t/lib-bundle-uri-protocol.sh
new file mode 100644 (file)
index 0000000..a4a1af8
--- /dev/null
@@ -0,0 +1,216 @@
+# Set up and run tests of the 'bundle-uri' command in protocol v2
+#
+# The test that includes this script should set BUNDLE_URI_PROTOCOL
+# to one of "file", "git", or "http".
+
+BUNDLE_URI_TEST_PARENT=
+BUNDLE_URI_TEST_URI=
+BUNDLE_URI_TEST_BUNDLE_URI=
+case "$BUNDLE_URI_PROTOCOL" in
+file)
+       BUNDLE_URI_PARENT=file_parent
+       BUNDLE_URI_REPO_URI="file://$PWD/file_parent"
+       BUNDLE_URI_BUNDLE_URI="$BUNDLE_URI_REPO_URI/fake.bdl"
+       test_set_prereq BUNDLE_URI_FILE
+       ;;
+git)
+       . "$TEST_DIRECTORY"/lib-git-daemon.sh
+       start_git_daemon --export-all --enable=receive-pack
+       BUNDLE_URI_PARENT="$GIT_DAEMON_DOCUMENT_ROOT_PATH/parent"
+       BUNDLE_URI_REPO_URI="$GIT_DAEMON_URL/parent"
+       BUNDLE_URI_BUNDLE_URI="https://example.com/fake.bdl"
+       test_set_prereq BUNDLE_URI_GIT
+       ;;
+http)
+       . "$TEST_DIRECTORY"/lib-httpd.sh
+       start_httpd
+       BUNDLE_URI_PARENT="$HTTPD_DOCUMENT_ROOT_PATH/http_parent"
+       BUNDLE_URI_REPO_URI="$HTTPD_URL/smart/http_parent"
+       BUNDLE_URI_BUNDLE_URI="https://example.com/fake.bdl"
+       test_set_prereq BUNDLE_URI_HTTP
+       ;;
+*)
+       BUG "Need to pass valid BUNDLE_URI_PROTOCOL (was \"$BUNDLE_URI_PROTOCOL\")"
+       ;;
+esac
+
+test_expect_success "setup protocol v2 $BUNDLE_URI_PROTOCOL:// tests" '
+       git init "$BUNDLE_URI_PARENT" &&
+       test_commit -C "$BUNDLE_URI_PARENT" one &&
+       git -C "$BUNDLE_URI_PARENT" config uploadpack.advertiseBundleURIs true
+'
+
+case "$BUNDLE_URI_PROTOCOL" in
+http)
+       test_expect_success "setup config for $BUNDLE_URI_PROTOCOL:// tests" '
+               git -C "$BUNDLE_URI_PARENT" config http.receivepack true
+       '
+       ;;
+*)
+       ;;
+esac
+BUNDLE_URI_BUNDLE_URI_ESCAPED=$(echo "$BUNDLE_URI_BUNDLE_URI" | test_uri_escape)
+
+test_expect_success "connect with $BUNDLE_URI_PROTOCOL:// using protocol v2: no bundle-uri" '
+       test_when_finished "rm -f log" &&
+       test_when_finished "git -C \"$BUNDLE_URI_PARENT\" config uploadpack.advertiseBundleURIs true" &&
+       git -C "$BUNDLE_URI_PARENT" config uploadpack.advertiseBundleURIs false &&
+
+       GIT_TRACE_PACKET="$PWD/log" \
+       git \
+               -c protocol.version=2 \
+               ls-remote --symref "$BUNDLE_URI_REPO_URI" \
+               >actual 2>err &&
+
+       # Server responded using protocol v2
+       grep "< version 2" log &&
+
+       ! grep bundle-uri log
+'
+
+test_expect_success "connect with $BUNDLE_URI_PROTOCOL:// using protocol v2: have bundle-uri" '
+       test_when_finished "rm -f log" &&
+
+       GIT_TRACE_PACKET="$PWD/log" \
+       git \
+               -c protocol.version=2 \
+               ls-remote --symref "$BUNDLE_URI_REPO_URI" \
+               >actual 2>err &&
+
+       # Server responded using protocol v2
+       grep "< version 2" log &&
+
+       # Server advertised bundle-uri capability
+       grep "< bundle-uri" log
+'
+
+test_expect_success "clone with $BUNDLE_URI_PROTOCOL:// using protocol v2: request bundle-uris" '
+       test_when_finished "rm -rf log* cloned*" &&
+
+       GIT_TRACE_PACKET="$PWD/log" \
+       git \
+               -c transfer.bundleURI=false \
+               -c protocol.version=2 \
+               clone "$BUNDLE_URI_REPO_URI" cloned \
+               >actual 2>err &&
+
+       # Server responded using protocol v2
+       grep "< version 2" log &&
+
+       # Server advertised bundle-uri capability
+       grep "< bundle-uri" log &&
+
+       # Client did not issue bundle-uri command
+       ! grep "> command=bundle-uri" log &&
+
+       GIT_TRACE_PACKET="$PWD/log" \
+       git \
+               -c transfer.bundleURI=true \
+               -c protocol.version=2 \
+               clone "$BUNDLE_URI_REPO_URI" cloned2 \
+               >actual 2>err &&
+
+       # Server responded using protocol v2
+       grep "< version 2" log &&
+
+       # Server advertised bundle-uri capability
+       grep "< bundle-uri" log &&
+
+       # Client issued bundle-uri command
+       grep "> command=bundle-uri" log &&
+
+       GIT_TRACE_PACKET="$PWD/log3" \
+       git \
+               -c transfer.bundleURI=true \
+               -c protocol.version=2 \
+               clone --bundle-uri="$BUNDLE_URI_BUNDLE_URI" \
+               "$BUNDLE_URI_REPO_URI" cloned3 \
+               >actual 2>err &&
+
+       # Server responded using protocol v2
+       grep "< version 2" log3 &&
+
+       # Server advertised bundle-uri capability
+       grep "< bundle-uri" log3 &&
+
+       # Client did not issue bundle-uri command (--bundle-uri override)
+       ! grep "> command=bundle-uri" log3
+'
+
+# The remaining tests will all assume transfer.bundleURI=true
+#
+# This test can be removed when transfer.bundleURI is enabled by default.
+test_expect_success 'enable transfer.bundleURI for remaining tests' '
+       git config --global transfer.bundleURI true
+'
+
+test_expect_success "test bundle-uri with $BUNDLE_URI_PROTOCOL:// using protocol v2" '
+       test_config -C "$BUNDLE_URI_PARENT" \
+               bundle.only.uri "$BUNDLE_URI_BUNDLE_URI_ESCAPED" &&
+
+       # All data about bundle URIs
+       cat >expect <<-EOF &&
+       [bundle]
+               version = 1
+               mode = all
+       [bundle "only"]
+               uri = $BUNDLE_URI_BUNDLE_URI_ESCAPED
+       EOF
+
+       test-tool bundle-uri \
+               ls-remote \
+               "$BUNDLE_URI_REPO_URI" \
+               >actual &&
+       test_cmp_config_output expect actual
+'
+
+test_expect_success "test bundle-uri with $BUNDLE_URI_PROTOCOL:// using protocol v2 and extra data" '
+       test_config -C "$BUNDLE_URI_PARENT" \
+               bundle.only.uri "$BUNDLE_URI_BUNDLE_URI_ESCAPED" &&
+
+       # Extra data should be ignored
+       test_config -C "$BUNDLE_URI_PARENT" bundle.only.extra bogus &&
+
+       # All data about bundle URIs
+       cat >expect <<-EOF &&
+       [bundle]
+               version = 1
+               mode = all
+       [bundle "only"]
+               uri = $BUNDLE_URI_BUNDLE_URI_ESCAPED
+       EOF
+
+       test-tool bundle-uri \
+               ls-remote \
+               "$BUNDLE_URI_REPO_URI" \
+               >actual &&
+       test_cmp_config_output expect actual
+'
+
+test_expect_success "test bundle-uri with $BUNDLE_URI_PROTOCOL:// using protocol v2 with list" '
+       test_config -C "$BUNDLE_URI_PARENT" \
+               bundle.bundle1.uri "$BUNDLE_URI_BUNDLE_URI_ESCAPED-1.bdl" &&
+       test_config -C "$BUNDLE_URI_PARENT" \
+               bundle.bundle2.uri "$BUNDLE_URI_BUNDLE_URI_ESCAPED-2.bdl" &&
+       test_config -C "$BUNDLE_URI_PARENT" \
+               bundle.bundle3.uri "$BUNDLE_URI_BUNDLE_URI_ESCAPED-3.bdl" &&
+
+       # All data about bundle URIs
+       cat >expect <<-EOF &&
+       [bundle]
+               version = 1
+               mode = all
+       [bundle "bundle1"]
+               uri = $BUNDLE_URI_BUNDLE_URI_ESCAPED-1.bdl
+       [bundle "bundle2"]
+               uri = $BUNDLE_URI_BUNDLE_URI_ESCAPED-2.bdl
+       [bundle "bundle3"]
+               uri = $BUNDLE_URI_BUNDLE_URI_ESCAPED-3.bdl
+       EOF
+
+       test-tool bundle-uri \
+               ls-remote \
+               "$BUNDLE_URI_REPO_URI" \
+               >actual &&
+       test_cmp_config_output expect actual
+'
index 502b4bcf9ea0ad06cddaad50b999dd8024af28eb..8ea31d187a91ee4df52203cbd72a1fac7dd4140e 100755 (executable)
@@ -815,7 +815,8 @@ test_expect_success 'test_oid provides sane info by default' '
        grep "^00*\$" actual &&
        rawsz="$(test_oid rawsz)" &&
        hexsz="$(test_oid hexsz)" &&
-       test "$hexsz" -eq $(wc -c <actual) &&
+       # +1 accounts for the trailing newline
+       test $(( $hexsz + 1)) -eq $(wc -c <actual) &&
        test $(( $rawsz * 2)) -eq "$hexsz"
 '
 
@@ -826,7 +827,7 @@ test_expect_success 'test_oid can look up data for SHA-1' '
        grep "^00*\$" actual &&
        rawsz="$(test_oid rawsz)" &&
        hexsz="$(test_oid hexsz)" &&
-       test $(wc -c <actual) -eq 40 &&
+       test $(wc -c <actual) -eq 41 &&
        test "$rawsz" -eq 20 &&
        test "$hexsz" -eq 40
 '
@@ -838,7 +839,7 @@ test_expect_success 'test_oid can look up data for SHA-256' '
        grep "^00*\$" actual &&
        rawsz="$(test_oid rawsz)" &&
        hexsz="$(test_oid hexsz)" &&
-       test $(wc -c <actual) -eq 64 &&
+       test $(wc -c <actual) -eq 65 &&
        test "$rawsz" -eq 32 &&
        test "$hexsz" -eq 64
 '
index e56f4b9ac59572288f69cbd3554ad27df736c6b4..eeb8539c1bcb2df86a5aff1fa2c78fc555598d14 100755 (executable)
@@ -5,6 +5,12 @@ test_description='basic sanity checks for git var'
 TEST_PASSES_SANITIZE_LEAK=true
 . ./test-lib.sh
 
+sane_unset_all_editors () {
+       sane_unset GIT_EDITOR &&
+       sane_unset VISUAL &&
+       sane_unset EDITOR
+}
+
 test_expect_success 'get GIT_AUTHOR_IDENT' '
        test_tick &&
        echo "$GIT_AUTHOR_NAME <$GIT_AUTHOR_EMAIL> $GIT_AUTHOR_DATE" >expect &&
@@ -47,6 +53,100 @@ test_expect_success 'get GIT_DEFAULT_BRANCH with configuration' '
        )
 '
 
+test_expect_success 'get GIT_EDITOR without configuration' '
+       (
+               sane_unset_all_editors &&
+               test_expect_code 1 git var GIT_EDITOR >out &&
+               test_must_be_empty out
+       )
+'
+
+test_expect_success 'get GIT_EDITOR with configuration' '
+       test_config core.editor foo &&
+       (
+               sane_unset_all_editors &&
+               echo foo >expect &&
+               git var GIT_EDITOR >actual &&
+               test_cmp expect actual
+       )
+'
+
+test_expect_success 'get GIT_EDITOR with environment variable GIT_EDITOR' '
+       (
+               sane_unset_all_editors &&
+               echo bar >expect &&
+               GIT_EDITOR=bar git var GIT_EDITOR >actual &&
+               test_cmp expect actual
+       )
+'
+
+test_expect_success 'get GIT_EDITOR with environment variable EDITOR' '
+       (
+               sane_unset_all_editors &&
+               echo bar >expect &&
+               EDITOR=bar git var GIT_EDITOR >actual &&
+               test_cmp expect actual
+       )
+'
+
+test_expect_success 'get GIT_EDITOR with configuration and environment variable GIT_EDITOR' '
+       test_config core.editor foo &&
+       (
+               sane_unset_all_editors &&
+               echo bar >expect &&
+               GIT_EDITOR=bar git var GIT_EDITOR >actual &&
+               test_cmp expect actual
+       )
+'
+
+test_expect_success 'get GIT_EDITOR with configuration and environment variable EDITOR' '
+       test_config core.editor foo &&
+       (
+               sane_unset_all_editors &&
+               echo foo >expect &&
+               EDITOR=bar git var GIT_EDITOR >actual &&
+               test_cmp expect actual
+       )
+'
+
+test_expect_success 'get GIT_SEQUENCE_EDITOR without configuration' '
+       (
+               sane_unset GIT_SEQUENCE_EDITOR &&
+               git var GIT_EDITOR >expect &&
+               git var GIT_SEQUENCE_EDITOR >actual &&
+               test_cmp expect actual
+       )
+'
+
+test_expect_success 'get GIT_SEQUENCE_EDITOR with configuration' '
+       test_config sequence.editor foo &&
+       (
+               sane_unset GIT_SEQUENCE_EDITOR &&
+               echo foo >expect &&
+               git var GIT_SEQUENCE_EDITOR >actual &&
+               test_cmp expect actual
+       )
+'
+
+test_expect_success 'get GIT_SEQUENCE_EDITOR with environment variable' '
+       (
+               sane_unset GIT_SEQUENCE_EDITOR &&
+               echo bar >expect &&
+               GIT_SEQUENCE_EDITOR=bar git var GIT_SEQUENCE_EDITOR >actual &&
+               test_cmp expect actual
+       )
+'
+
+test_expect_success 'get GIT_SEQUENCE_EDITOR with configuration and environment variable' '
+       test_config sequence.editor foo &&
+       (
+               sane_unset GIT_SEQUENCE_EDITOR &&
+               echo bar >expect &&
+               GIT_SEQUENCE_EDITOR=bar git var GIT_SEQUENCE_EDITOR >actual &&
+               test_cmp expect actual
+       )
+'
+
 # For git var -l, we check only a representative variable;
 # testing the whole output would make our test too brittle with
 # respect to unrelated changes in the test suite's environment.
index abecd75e4e430b6a1182690ae54a0e5d4e76bc32..46abbeed683992672f9dcaa6f9e2b256f1ea02eb 100755 (executable)
@@ -8,8 +8,8 @@ export GIT_TEST_DEFAULT_INITIAL_BRANCH_NAME
 . ./test-lib.sh
 . "$TEST_DIRECTORY"/lib-terminal.sh
 
-TEST_ROOT="$PWD"
-PATH=$TEST_ROOT:$PATH
+PATH=$PWD:$PATH
+TEST_ROOT="$(pwd)"
 
 write_script <<\EOF "$TEST_ROOT/rot13.sh"
 tr \
index c6e0d655630638853b6831350618bc10b15de6e2..3648d439a87eabb37b581cf5e731f9048085703b 100755 (executable)
@@ -2,6 +2,7 @@
 
 test_description='git for-each-repo builtin'
 
+TEST_PASSES_SANITIZE_LEAK=true
 . ./test-lib.sh
 
 test_expect_success 'run based on configured value' '
index 8d59905ef099bf0ab46d0681568dd17d7ed85d98..574de3419800e5f392038de8f35374d355fc9bcf 100755 (executable)
@@ -6,6 +6,7 @@ test_description='check that the most basic functions work
 Verify wrappers and compatibility functions.
 '
 
+TEST_PASSES_SANITIZE_LEAK=true
 . ./test-lib.sh
 
 test_expect_success 'character classes (isspace, isalpha etc.)' '
index 516a6112fdcf1907fb3827cc23e7ad0b24b39099..3fb1b0c162ded2a5d78767dbd62de97fc59e5d29 100755 (executable)
@@ -370,7 +370,7 @@ test_expect_success 'read-tree supports the super-prefix' '
        cat <<-EOF >expect &&
                error: Updating '\''fictional/a'\'' would lose untracked files in it
        EOF
-       test_must_fail git --super-prefix fictional/ read-tree -u -m "$treeH" "$treeM" 2>actual &&
+       test_must_fail git read-tree --super-prefix fictional/ -u -m "$treeH" "$treeM" 2>actual &&
        test_cmp expect actual
 '
 
index 742f0fa909fd6e8a7fe2b376b3e5083a30d71c73..595b24c0adb3081230fe65884fd436e1f12896d4 100755 (executable)
@@ -12,6 +12,7 @@ test_description='sparse checkout tests
 '
 
 TEST_CREATE_REPO_NO_TEMPLATE=1
+TEST_PASSES_SANITIZE_LEAK=true
 . ./test-lib.sh
 . "$TEST_DIRECTORY"/lib-read-tree.sh
 
index da539716359614f6d91cd1db896b3981e1805d6d..cca4380e4319a024416c4bbe2e097f63e409af48 100755 (executable)
@@ -3,7 +3,7 @@
 test_description='git read-tree in partial clones'
 
 TEST_NO_CREATE_REPO=1
-
+TEST_PASSES_SANITIZE_LEAK=true
 . ./test-lib.sh
 
 test_expect_success 'read-tree in partial clone prefetches in one batch' '
index b563d6c263ec1cee0d4bfc93cba1e85e4833f4e6..627267be1531056c4eff9e5a0dc2bc1d8d1eab38 100755 (executable)
@@ -238,7 +238,7 @@ test_expect_success 'cone mode: match patterns' '
 test_expect_success 'cone mode: warn on bad pattern' '
        test_when_finished mv sparse-checkout repo/.git/info/ &&
        cp repo/.git/info/sparse-checkout . &&
-       echo "!/deep/deeper/*" >>repo/.git/info/sparse-checkout &&
+       echo "!/deep/deeper/*/" >>repo/.git/info/sparse-checkout &&
        git -C repo read-tree -mu HEAD 2>err &&
        test_i18ngrep "unrecognized negative pattern" err
 '
@@ -667,6 +667,15 @@ test_expect_success 'pattern-checks: starting "*"' '
        check_read_tree_errors repo "a deep" "disabling cone pattern matching"
 '
 
+test_expect_success 'pattern-checks: non directory pattern' '
+       cat >repo/.git/info/sparse-checkout <<-\EOF &&
+       /deep/deeper1/a
+       EOF
+       check_read_tree_errors repo deep "disabling cone pattern matching" &&
+       check_files repo/deep deeper1 &&
+       check_files repo/deep/deeper1 a
+'
+
 test_expect_success 'pattern-checks: contained glob characters' '
        for c in "[a]" "\\" "?" "*"
        do
index 0acabb6d11b48aca54e73025965f4c4e6b561d34..7cf80bf66a6d2e1adea5c269c91923652c29e232 100755 (executable)
@@ -27,7 +27,7 @@ test_expect_success 'setup' '
 '
 
 test_expect_success 'gitdir selection on normal repos' '
-       echo $(test_oid version) >expect &&
+       test_oid version >expect &&
        git config core.repositoryformatversion >actual &&
        git -C test config core.repositoryformatversion >actual2 &&
        test_cmp expect actual &&
index 13c2b43bbae8343c2402952bba8ac9635001b453..b5606d93b52bf44366a51f3d4b6dd19e7a8816ab 100755 (executable)
@@ -1,6 +1,8 @@
 #!/bin/sh
 
 test_description='Test git update-ref error handling'
+
+TEST_PASSES_SANITIZE_LEAK=true
 . ./test-lib.sh
 
 # Create some references, perhaps run pack-refs --all, then try to
index be12fb63506e2a8c242e876526dbe1d60e105f49..f23c0152a858265e4a8c6b0c068e00823aadb444 100755 (executable)
@@ -2,6 +2,7 @@
 
 test_description='avoid rewriting packed-refs unnecessarily'
 
+TEST_PASSES_SANITIZE_LEAK=true
 . ./test-lib.sh
 
 # Add an identifying mark to the packed-refs file header line. This
index 934688a1ee82e29f2552bb2444eb06626fe254f8..d2a4822d46fd05c45273f48b0caf48cadb2a07f1 100755 (executable)
@@ -4,6 +4,7 @@ test_description='Test reflog interaction with detached HEAD'
 GIT_TEST_DEFAULT_INITIAL_BRANCH_NAME=main
 export GIT_TEST_DEFAULT_INITIAL_BRANCH_NAME
 
+TEST_PASSES_SANITIZE_LEAK=true
 . ./test-lib.sh
 
 reset_state () {
index 81de584ea29bca8cfcbe20769c69072202a53efb..37ee5091b5caa7c087663cbb51f701084227c414 100755 (executable)
@@ -195,7 +195,7 @@ test_expect_success 'rev-parse --is-shallow-repository in non-shallow repo' '
 '
 
 test_expect_success 'rev-parse --show-object-format in repo' '
-       echo "$(test_oid algo)" >expect &&
+       test_oid algo >expect &&
        git rev-parse --show-object-format >actual &&
        test_cmp expect actual &&
        git rev-parse --show-object-format=storage >actual &&
index b75558040ffd6ee9174ed37bd2ca746212b57013..ae6528aecea7d3b36cfe51c5f19c7098f05429e3 100755 (executable)
@@ -1,6 +1,8 @@
 #!/bin/sh
 
 test_description='test separate work tree'
+
+TEST_PASSES_SANITIZE_LEAK=true
 . ./test-lib.sh
 
 test_expect_success 'setup' '
index 553a3f601ba7c76efa193d4a33823197c05281e7..c799f5b6acad35633c6efa87a1a5e10aa67b8fd2 100755 (executable)
@@ -221,7 +221,8 @@ test_expect_success 'setup' '
        rm -rf /.git &&
        echo "Initialized empty Git repository in /.git/" > expected &&
        git init > result &&
-       test_cmp expected result
+       test_cmp expected result &&
+       git config --global --add safe.directory /
 '
 
 test_vars 'auto gitdir, root' ".git" "/" ""
@@ -242,7 +243,7 @@ say "auto bare gitdir"
 # DESTROYYYYY!!!!!
 test_expect_success 'setup' '
        rm -rf /refs /objects /info /hooks &&
-       rm -f /expected /ls.expected /me /result &&
+       rm -f /HEAD /expected /ls.expected /me /result &&
        cd / &&
        echo "Initialized empty Git repository in /" > expected &&
        git init --bare > result &&
@@ -255,4 +256,9 @@ test_expect_success 'go to /foo' 'cd /foo'
 
 test_vars 'auto gitdir, root' "/" "" ""
 
+test_expect_success 'cleanup root' '
+       rm -rf /.git /refs /objects /info /hooks /branches /foo &&
+       rm -f /HEAD /config /description /expected /ls.expected /me /result
+'
+
 test_done
index 1f6c4ed0428a5c966c76359c1f4fa2e536f37fcf..4b6372f4c3e946c26a490698335a821cff5701f2 100755 (executable)
@@ -5,6 +5,7 @@ test_description='checkout can switch to last branch and merge base'
 GIT_TEST_DEFAULT_INITIAL_BRANCH_NAME=main
 export GIT_TEST_DEFAULT_INITIAL_BRANCH_NAME
 
+TEST_PASSES_SANITIZE_LEAK=true
 . ./test-lib.sh
 
 test_expect_success 'setup' '
index 771c3c3c50e60583e3ba8dffbb62f47a7619f35b..8581ad3437907862e022e1bea8339182d68ef873 100755 (executable)
@@ -3,6 +3,7 @@
 test_description='checkout'
 
 TEST_CREATE_REPO_NO_TEMPLATE=1
+TEST_PASSES_SANITIZE_LEAK=true
 . ./test-lib.sh
 
 # Arguments: [!] <branch> <oid> [<checkout options>]
index 8f13341cf8ee0d82e0adeeb28449e31d104a5243..3832c3de8135524b4294b7ece693194b3bf506c1 100755 (executable)
@@ -2,6 +2,7 @@
 
 test_description='checkout --no-overlay <tree-ish> -- <pathspec>'
 
+TEST_PASSES_SANITIZE_LEAK=true
 . ./test-lib.sh
 
 test_expect_success 'setup' '
index 963f3462b750b2a4fc7d009df6cf1fa1949090d6..14218b342437788fef40c539c45fb41575ef0071 100755 (executable)
@@ -18,6 +18,7 @@ This test runs git ls-files --others with the following working tree:
       git repository with a commit and an untracked file
 '
 
+TEST_PASSES_SANITIZE_LEAK=true
 . ./test-lib.sh
 
 test_expect_success 'setup: directories' '
index 580e158f9918bcaf86224ce8743894da67266096..054178703d732403ce32eda9871d8d85c47dddaa 100755 (executable)
@@ -41,6 +41,8 @@ Also for modification test, the cache and working tree have:
 We should report path0, path1, path2/file2, path3/file3, path7 and path8
 modified without reporting path9 and path10.  submod1 is also modified.
 '
+
+TEST_PASSES_SANITIZE_LEAK=true
 . ./test-lib.sh
 
 test_expect_success 'git update-index --add to add various paths.' '
index f1f09abdd9b254eed1efa6b0d06369f3c8606cf8..3884694165525b9e4a3d8527785f40758817be53 100755 (executable)
@@ -2,6 +2,7 @@
 
 test_description='fetching and pushing project with subproject'
 
+TEST_PASSES_SANITIZE_LEAK=true
 . ./test-lib.sh
 
 test_expect_success setup '
index 52f76f7b57f3ba9861bc9220602a5ca9fc45aeea..c4a72ae44623a0bab3ce06c1a5e850e9e8405054 100755 (executable)
@@ -8,6 +8,8 @@ test_description='git ls-files test (--with-tree).
 This test runs git ls-files --with-tree and in particular in
 a scenario known to trigger a crash with some versions of git.
 '
+
+TEST_PASSES_SANITIZE_LEAK=true
 . ./test-lib.sh
 
 test_expect_success 'setup' '
index 793bf4d269e6a1f101481ae70280dd5db19e75a9..3399344f25dc859ad743b3914a3f5c37e92bffbb 100755 (executable)
@@ -57,6 +57,16 @@ test_expect_success 'create branch with pseudo-qualified name' '
        expect_branch refs/heads/refs/heads/qualified two
 '
 
+test_expect_success 'force-copy a branch to itself via @{-1} is no-op' '
+       git branch -t copiable main &&
+       git checkout copiable &&
+       git checkout - &&
+       git branch -C @{-1} copiable &&
+       git config --get-all branch.copiable.merge >actual &&
+       echo refs/heads/main >expect &&
+       test_cmp expect actual
+'
+
 test_expect_success 'delete branch via @{-1}' '
        git branch previous-del &&
 
index 83ffb39d9ffdfd869ecb24e5adbbe11944a73c53..acaf5558dbe2e445efe2989e137b2f9ef5140aaf 100755 (executable)
@@ -2,6 +2,7 @@
 
 test_description='git rebase interactive environment'
 
+TEST_PASSES_SANITIZE_LEAK=true
 . ./test-lib.sh
 
 test_expect_success 'setup' '
index 9fab0d779bb6cf32dbc0ece0a69ad2c2821a598e..e8456831e8ba1a3ff2009ff6339459f6c00055da 100755 (executable)
@@ -5,6 +5,7 @@ test_description='git rebase with its hook(s)'
 GIT_TEST_DEFAULT_INITIAL_BRANCH_NAME=main
 export GIT_TEST_DEFAULT_INITIAL_BRANCH_NAME
 
+TEST_PASSES_SANITIZE_LEAK=true
 . ./test-lib.sh
 
 test_expect_success setup '
index f6993b7e14d91617b337bfaefa717ec8591fb28f..e1b1e947647462f7cbc9fb2492b2cfe127056f1e 100755 (executable)
@@ -5,6 +5,7 @@ test_description='git rebase --signoff
 This test runs git rebase --signoff and make sure that it works.
 '
 
+TEST_PASSES_SANITIZE_LEAK=true
 . ./test-lib.sh
 
 # A simple file to commit
index abd66f360213e18ad80b1c7bb2396b06d3e77b28..8e0d03969a2fec13fba015158b3380ec2ee37c42 100755 (executable)
@@ -2,6 +2,7 @@
 
 test_description='rebase should reread the todo file if an exec modifies it'
 
+TEST_PASSES_SANITIZE_LEAK=true
 . ./test-lib.sh
 . "$TEST_DIRECTORY"/lib-rebase.sh
 
index 05df964670f49d011bf35b414177a35f812a4c4e..c8172b0852233b6f370eab47c6cc4877db54d8f7 100755 (executable)
@@ -2,6 +2,7 @@
 
 test_description='git rebase across mode change'
 
+TEST_PASSES_SANITIZE_LEAK=true
 . ./test-lib.sh
 
 test_expect_success 'setup' '
index 4c661d4d54a779f8bc056c62e5d9e864401420c7..67fd2345affd78507a6cb714b48565ed190cb5af 100755 (executable)
@@ -12,7 +12,7 @@ create_crlf_ref () {
        cat >.crlf-orig-$branch.txt &&
        cat .crlf-orig-$branch.txt | append_cr >.crlf-message-$branch.txt &&
        grep 'Subject' .crlf-orig-$branch.txt | tr '\n' ' ' | sed 's/[ ]*$//' | tr -d '\n' >.crlf-subject-$branch.txt &&
-       grep 'Body' .crlf-message-$branch.txt >.crlf-body-$branch.txt || true &&
+       grep 'Body' .crlf-orig-$branch.txt | append_cr >.crlf-body-$branch.txt &&
        LIB_CRLF_BRANCHES="${LIB_CRLF_BRANCHES} ${branch}" &&
        test_tick &&
        hash=$(git commit-tree HEAD^{tree} -p HEAD -F .crlf-message-${branch}.txt) &&
index de1da4673da9626f0c155fd4c67597a403870ece..012f155e10ae9f9e1fcbd3472cd9a789bae919ec 100755 (executable)
@@ -2281,7 +2281,7 @@ test_expect_success 'format-patch --attach cover-letter only is non-multipart' '
        test_line_count = 1 output
 '
 
-test_expect_success 'format-patch --pretty=mboxrd' '
+test_expect_success '-c format.mboxrd format-patch' '
        sp=" " &&
        cat >msg <<-INPUT_END &&
        mboxrd should escape the body
@@ -2316,7 +2316,9 @@ test_expect_success 'format-patch --pretty=mboxrd' '
        INPUT_END
 
        C=$(git commit-tree HEAD^^{tree} -p HEAD <msg) &&
-       git format-patch --pretty=mboxrd --stdout -1 $C~1..$C >patch &&
+       git -c format.mboxrd format-patch --stdout -1 $C~1..$C >patch &&
+       git format-patch --pretty=mboxrd --stdout -1 $C~1..$C >compat &&
+       test_cmp patch compat &&
        git grep -h --no-index -A11 \
                "^>From could trip up a loose mbox parser" patch >actual &&
        test_cmp expect actual
index f3e20dd5bba202ba5d5589e897d753672096b866..b298f220e01fe6de17a5dfe608550a5a57dcf9c6 100755 (executable)
@@ -1638,7 +1638,7 @@ test_expect_success 'no effect on diff from --color-moved with --word-diff' '
        test_cmp expect actual
 '
 
-test_expect_success !SANITIZE_LEAK 'no effect on show from --color-moved with --word-diff' '
+test_expect_success 'no effect on show from --color-moved with --word-diff' '
        git show --color-moved --word-diff >actual &&
        git show --word-diff >expect &&
        test_cmp expect actual
@@ -2024,7 +2024,7 @@ test_expect_success '--color-moved rewinds for MIN_ALNUM_COUNT' '
        test_cmp expected actual
 '
 
-test_expect_success !SANITIZE_LEAK 'move detection with submodules' '
+test_expect_success 'move detection with submodules' '
        test_create_repo bananas &&
        echo ripe >bananas/recipe &&
        git -C bananas add recipe &&
index 7cb99092938d7dd64e8ab6f4e147fc5f70c0a310..787605ce3fd076b101d1c825fca174020ae7ff46 100755 (executable)
@@ -52,8 +52,8 @@ test_expect_success setup '
 '
 
 test_expect_success 'cross renames to be detected for regular files' '
-
-       git diff-tree five six -r --name-status -B -M | sort >actual &&
+       git diff-tree five six -r --name-status -B -M >out &&
+       sort out >actual &&
        {
                echo "R100      foo     bar" &&
                echo "R100      bar     foo"
@@ -63,8 +63,8 @@ test_expect_success 'cross renames to be detected for regular files' '
 '
 
 test_expect_success 'cross renames to be detected for typechange' '
-
-       git diff-tree one two -r --name-status -B -M | sort >actual &&
+       git diff-tree one two -r --name-status -B -M >out &&
+       sort out >actual &&
        {
                echo "R100      foo     bar" &&
                echo "R100      bar     foo"
@@ -74,8 +74,8 @@ test_expect_success 'cross renames to be detected for typechange' '
 '
 
 test_expect_success 'moves and renames' '
-
-       git diff-tree three four -r --name-status -B -M | sort >actual &&
+       git diff-tree three four -r --name-status -B -M >out &&
+       sort out >actual &&
        {
                # see -B -M (#6) in t4008
                echo "C100      foo     bar" &&
index 29e49d22902dd7e7566f2662174475bf35c6ebed..9f6043dabaccdcbbaecccf2b2f0248ca132c946e 100755 (executable)
@@ -34,12 +34,12 @@ test_expect_success 'setup' '
        100644 blob $(test_oid hash2)   foo
        EOF
 
-       echo "$(test_oid val1)" > foo &&
+       test_oid val1 > foo &&
        git add foo &&
        git commit -m "initial" &&
        git cat-file -p HEAD: > actual &&
        test_cmp expect_initial actual &&
-       echo "$(test_oid val2)" > foo &&
+       test_oid val2 > foo &&
        git commit -a -m "update" &&
        git cat-file -p HEAD: > actual &&
        test_cmp expect_update actual
index fab351b48a1f52b450f3911aa6edafae70015613..9b46c4c1befece1f3e231b9672291b7a538c3f6f 100755 (executable)
@@ -1,6 +1,8 @@
 #!/bin/sh
 
 test_description='diff --relative tests'
+
+TEST_PASSES_SANITIZE_LEAK=true
 . ./test-lib.sh
 
 test_expect_success 'setup' '
@@ -162,6 +164,35 @@ check_diff_relative_option subdir file2 true --no-relative --relative
 check_diff_relative_option . file2 false --no-relative --relative=subdir
 check_diff_relative_option . file2 true --no-relative --relative=subdir
 
+test_expect_success 'external diff with --relative' '
+       test_when_finished "git reset --hard" &&
+       echo changed >file1 &&
+       echo changed >subdir/file2 &&
+
+       write_script mydiff <<-\EOF &&
+       # hacky pretend diff; the goal here is just to make sure we got
+       # passed sensible input that we _could_ diff, without relying on
+       # the specific output of a system diff tool.
+       echo "diff a/$1 b/$1" &&
+       echo "--- a/$1" &&
+       echo "+++ b/$1" &&
+       echo "@@ -1 +0,0 @@" &&
+       sed "s/^/-/" "$2" &&
+       sed "s/^/+/" "$5"
+       EOF
+
+       cat >expect <<-\EOF &&
+       diff a/file2 b/file2
+       --- a/file2
+       +++ b/file2
+       @@ -1 +0,0 @@
+       -other content
+       +changed
+       EOF
+       GIT_EXTERNAL_DIFF=./mydiff git diff --relative=subdir >actual &&
+       test_cmp expect actual
+'
+
 test_expect_success 'setup diff --relative unmerged' '
        test_commit zero file0 &&
        test_commit base subdir/file0 &&
index 0ae0cd3a524da312d7f1cc7b1d4b0f6ed7b2b529..ffaf69335f7b224f76ece7228435a4ce057e5c83 100755 (executable)
@@ -86,4 +86,14 @@ test_expect_success 'diff-files -3' '
        test_cmp diff-files-3.expect diff-files-3.actual
 '
 
+test_expect_success 'diff --stat' '
+       for path in $paths
+       do
+               echo " $path | Unmerged" || return 1
+       done >diff-stat.expect &&
+       echo " 0 files changed" >>diff-stat.expect &&
+       git diff --cached --stat >diff-stat.actual &&
+       test_cmp diff-stat.expect diff-stat.actual
+'
+
 test_done
index b5c281edaa7037097f3c8fad28c1bb1994607a20..3ee27e277dca1cd95484ec9ef3b1251e38b22dde 100755 (executable)
@@ -8,6 +8,7 @@ test_description='test --stat output of various commands'
 GIT_TEST_DEFAULT_INITIAL_BRANCH_NAME=main
 export GIT_TEST_DEFAULT_INITIAL_BRANCH_NAME
 
+TEST_PASSES_SANITIZE_LEAK=true
 . ./test-lib.sh
 . "$TEST_DIRECTORY"/lib-terminal.sh
 
index 3feadf0e3595b2950dd6b5698ba0f25969de8b93..4e9fa0403d3631fdeafd0e90e3ae80894930c2e9 100755 (executable)
@@ -2,6 +2,7 @@
 
 test_description='diff --no-index'
 
+TEST_PASSES_SANITIZE_LEAK=true
 . ./test-lib.sh
 
 test_expect_success 'setup' '
index 28f42a4046e08ed61fcf2f300b826c33136e9e98..f60f5cbd65f049e1d3148bae352f3c6032454f2e 100755 (executable)
@@ -2,6 +2,7 @@
 
 test_description='behavior of diff when reading objects in a partial clone'
 
+TEST_PASSES_SANITIZE_LEAK=true
 . ./test-lib.sh
 
 test_expect_success 'git show batches blobs' '
index 1618a6dbc7c718677cf669b5f6fc96f2682992fb..e9a87d761d168323ada1e58ecd9df719cc7511b3 100755 (executable)
@@ -2,6 +2,7 @@
 
 test_description='patching from inconvenient places'
 
+TEST_PASSES_SANITIZE_LEAK=true
 . ./test-lib.sh
 
 test_expect_success 'setup' '
index 6bc3fb97a754bf242f8baf1ba867aa9f7fa93d0d..d3502c6fddf5a6a69bde769b1f808c9431d1e38e 100755 (executable)
@@ -2,6 +2,7 @@
 
 test_description='git apply with weird postimage filenames'
 
+TEST_PASSES_SANITIZE_LEAK=true
 . ./test-lib.sh
 
 test_expect_success 'setup' '
index cdad4b688078adc24da922bce14e130982f4484e..78cf1c880e69832ecd796ec11af8eff6ce19bd80 100755 (executable)
@@ -345,6 +345,21 @@ test_expect_success 'am with failing applypatch-msg hook' '
        test_cmp_rev first HEAD
 '
 
+test_expect_success 'am with failing applypatch-msg hook (no verify)' '
+       rm -fr .git/rebase-apply &&
+       git reset --hard &&
+       git checkout first &&
+       test_hook applypatch-msg <<-\EOF &&
+       echo hook-message >"$1"
+       exit 1
+       EOF
+       git am --no-verify patch1 &&
+       test_path_is_missing .git/rebase-apply &&
+       git diff --exit-code second &&
+       git log -1 --format=format:%B >actual &&
+       test_cmp msg actual
+'
+
 test_expect_success 'am with pre-applypatch hook' '
        rm -fr .git/rebase-apply &&
        git reset --hard &&
@@ -374,6 +389,23 @@ test_expect_success 'am with failing pre-applypatch hook' '
        test_cmp_rev first HEAD
 '
 
+test_expect_success 'am with failing pre-applypatch hook (no verify)' '
+       rm -fr .git/rebase-apply &&
+       git reset --hard &&
+       git checkout first &&
+       touch empty-file &&
+       test_hook pre-applypatch <<-\EOF &&
+       rm empty-file
+       exit 1
+       EOF
+       git am --no-verify patch1 &&
+       test_path_is_missing .git/rebase-apply &&
+       test_path_is_file empty-file &&
+       git diff --exit-code second &&
+       git log -1 --format=format:%B >actual &&
+       test_cmp msg actual
+'
+
 test_expect_success 'am with post-applypatch hook' '
        rm -fr .git/rebase-apply &&
        git reset --hard &&
@@ -1033,7 +1065,7 @@ test_expect_success 'am --patch-format=mboxrd handles mboxrd' '
        >From extra escape for reversibility
        INPUT_END
        git commit -F msg &&
-       git format-patch --pretty=mboxrd --stdout -1 >mboxrd1 &&
+       git -c format.mboxrd format-patch --stdout -1 >mboxrd1 &&
        grep "^>From could trip up a loose mbox parser" mboxrd1 &&
        git checkout -f first &&
        git am --patch-format=mboxrd mboxrd1 &&
index cd1cab3e54b9170d5751279bd68e02dbc3cbcad2..fa7f987284b117607fa051c91dd439749da546d4 100755 (executable)
@@ -1022,4 +1022,69 @@ test_expect_success '--mailmap enables mailmap in cat-file for annotated tag obj
        test_cmp expect actual
 '
 
+test_expect_success 'git cat-file -s returns correct size with --use-mailmap' '
+       test_when_finished "rm .mailmap" &&
+       cat >.mailmap <<-\EOF &&
+       C O Mitter <committer@example.com> Orig <orig@example.com>
+       EOF
+       git cat-file commit HEAD >commit.out &&
+       echo $(wc -c <commit.out) >expect &&
+       git cat-file --use-mailmap commit HEAD >commit.out &&
+       echo $(wc -c <commit.out) >>expect &&
+       git cat-file -s HEAD >actual &&
+       git cat-file --use-mailmap -s HEAD >>actual &&
+       test_cmp expect actual
+'
+
+test_expect_success 'git cat-file -s returns correct size with --use-mailmap for tag objects' '
+       test_when_finished "rm .mailmap" &&
+       cat >.mailmap <<-\EOF &&
+       Orig <orig@example.com> C O Mitter <committer@example.com>
+       EOF
+       git tag -a -m "annotated tag" v3 &&
+       git cat-file tag v3 >tag.out &&
+       echo $(wc -c <tag.out) >expect &&
+       git cat-file --use-mailmap tag v3 >tag.out &&
+       echo $(wc -c <tag.out) >>expect &&
+       git cat-file -s v3 >actual &&
+       git cat-file --use-mailmap -s v3 >>actual &&
+       test_cmp expect actual
+'
+
+test_expect_success 'git cat-file --batch-check returns correct size with --use-mailmap' '
+       test_when_finished "rm .mailmap" &&
+       cat >.mailmap <<-\EOF &&
+       C O Mitter <committer@example.com> Orig <orig@example.com>
+       EOF
+       git cat-file commit HEAD >commit.out &&
+       commit_size=$(wc -c <commit.out) &&
+       commit_sha=$(git rev-parse HEAD) &&
+       echo $commit_sha commit $commit_size >expect &&
+       git cat-file --use-mailmap commit HEAD >commit.out &&
+       commit_size=$(wc -c <commit.out) &&
+       echo $commit_sha commit $commit_size >>expect &&
+       echo "HEAD" >in &&
+       git cat-file --batch-check <in >actual &&
+       git cat-file --use-mailmap --batch-check <in >>actual &&
+       test_cmp expect actual
+'
+
+test_expect_success 'git cat-file --batch-command returns correct size with --use-mailmap' '
+       test_when_finished "rm .mailmap" &&
+       cat >.mailmap <<-\EOF &&
+       C O Mitter <committer@example.com> Orig <orig@example.com>
+       EOF
+       git cat-file commit HEAD >commit.out &&
+       commit_size=$(wc -c <commit.out) &&
+       commit_sha=$(git rev-parse HEAD) &&
+       echo $commit_sha commit $commit_size >expect &&
+       git cat-file --use-mailmap commit HEAD >commit.out &&
+       commit_size=$(wc -c <commit.out) &&
+       echo $commit_sha commit $commit_size >>expect &&
+       echo "info HEAD" >in &&
+       git cat-file --batch-command <in >actual &&
+       git cat-file --use-mailmap --batch-command <in >>actual &&
+       test_cmp expect actual
+'
+
 test_done
index e448ef2928a8261e7c991ee6eb0e96bf299d84ee..0404491d6ee9fa275764497f917d90a7334d6776 100755 (executable)
@@ -156,7 +156,7 @@ test_expect_success 'NUL termination with --reflog --pretty=oneline' '
        for r in $revs
        do
                git show -s --pretty=oneline "$r" >raw &&
-               cat raw | lf_to_nul || exit 1
+               cat raw | lf_to_nul || return 1
        done >expect &&
        # the trailing NUL is already produced so we do not need to
        # output another one
index ac9e4d0928593cd37b4a79e27f5fe99b6e4a3faf..c6540e822fbdc56f068ae1b8d991175303bff038 100755 (executable)
@@ -315,4 +315,26 @@ test_expect_success 'line-log with --before' '
        test_cmp expect actual
 '
 
+test_expect_success 'setup tests for zero-width regular expressions' '
+       cat >expect <<-EOF
+       Modify func1() in file.c
+       Add func1() and func2() in file.c
+       EOF
+'
+
+test_expect_success 'zero-width regex $ matches any function name' '
+       git log --format="%s" --no-patch "-L:$:file.c" >actual &&
+       test_cmp expect actual
+'
+
+test_expect_success 'zero-width regex ^ matches any function name' '
+       git log --format="%s" --no-patch "-L:^:file.c" >actual &&
+       test_cmp expect actual
+'
+
+test_expect_success 'zero-width regex .* matches any function name' '
+       git log --format="%s" --no-patch "-L:.*:file.c" >actual &&
+       test_cmp expect actual
+'
+
 test_done
index 53a4af324495dda3738388af31dcecf31f8ee1ee..590fce95e90a8bd751daf13c0696d0a39b8f7d1b 100755 (executable)
@@ -2,6 +2,7 @@
 
 test_description='log/show --expand-tabs'
 
+TEST_PASSES_SANITIZE_LEAK=true
 . ./test-lib.sh
 
 HT="   "
index cac85591b52b3b3d7d0a7aa7f5dbc0c32d8ebdda..a8983a0edcb99b6dd1166567ee658e41918df428 100755 (executable)
@@ -860,4 +860,66 @@ test_expect_success '--stdin with both a successful and a conflicted merge' '
        test_cmp expect actual
 '
 
+
+test_expect_success '--merge-base is incompatible with --stdin' '
+       test_must_fail git merge-tree --merge-base=side1 --stdin 2>expect &&
+
+       grep "^fatal: --merge-base is incompatible with --stdin" expect
+'
+
+# specify merge-base as parent of branch2
+# git merge-tree --write-tree --merge-base=c2 c1 c3
+#   Commit c1: add file1
+#   Commit c2: add file2 after c1
+#   Commit c3: add file3 after c2
+#   Expected: add file3, and file2 does NOT appear
+
+test_expect_success 'specify merge-base as parent of branch2' '
+       # Setup
+       test_when_finished "rm -rf base-b2-p" &&
+       git init base-b2-p &&
+       test_commit -C base-b2-p c1 file1 &&
+       test_commit -C base-b2-p c2 file2 &&
+       test_commit -C base-b2-p c3 file3 &&
+
+       # Testing
+       TREE_OID=$(git -C base-b2-p merge-tree --write-tree --merge-base=c2 c1 c3) &&
+
+       q_to_tab <<-EOF >expect &&
+       100644 blob $(git -C base-b2-p rev-parse c1:file1)Qfile1
+       100644 blob $(git -C base-b2-p rev-parse c3:file3)Qfile3
+       EOF
+
+       git -C base-b2-p ls-tree $TREE_OID >actual &&
+       test_cmp expect actual
+'
+
+# Since the earlier tests have verified that individual merge-tree calls
+# are doing the right thing, this test case is only used to verify that
+# we can also trigger merges via --stdin, and that when we do we get
+# the same answer as running a bunch of separate merges.
+
+test_expect_success 'check the input format when --stdin is passed' '
+       test_when_finished "rm -rf repo" &&
+       git init repo &&
+       test_commit -C repo c1 &&
+       test_commit -C repo c2 &&
+       test_commit -C repo c3 &&
+       printf "c1 c3\nc2 -- c1 c3\nc2 c3" | git -C repo merge-tree --stdin >actual &&
+
+       printf "1\0" >expect &&
+       git -C repo merge-tree --write-tree -z c1 c3 >>expect &&
+       printf "\0" >>expect &&
+
+       printf "1\0" >>expect &&
+       git -C repo merge-tree --write-tree -z --merge-base=c2 c1 c3 >>expect &&
+       printf "\0" >>expect &&
+
+       printf "1\0" >>expect &&
+       git -C repo merge-tree --write-tree -z c2 c3 >>expect &&
+       printf "\0" >>expect &&
+
+       test_cmp expect actual
+'
+
 test_done
index 6d693eef82f74a72ad4f29c399e297427c892d12..7d8dee41b0ded364d9b364ab46296c20b2c8eee1 100755 (executable)
@@ -428,8 +428,9 @@ test_bitmap_cases () {
                        test_line_count = 2 packs &&
                        test_line_count = 2 bitmaps &&
 
-                       git rev-list --use-bitmap-index HEAD 2>err &&
-                       grep "ignoring extra bitmap file" err
+                       GIT_TRACE2_EVENT=$(pwd)/trace2.txt git rev-list --use-bitmap-index HEAD &&
+                       grep "opened bitmap" trace2.txt &&
+                       grep "ignoring extra bitmap" trace2.txt
                )
        '
 }
index cc4cfaa9d3712684f57f34372a4f40d0abf11f7e..ceaa6700a27230cdea34bd65aeff2db34c53ad64 100755 (executable)
@@ -59,7 +59,7 @@ test_expect_success 'setup' '
 test_expect_success 'set up base packfile and variables' '
        # the hash of this content starts with ff, which
        # makes some later computations much simpler
-       echo $(test_oid oidfff) >file &&
+       test_oid oidfff >file &&
        git add file &&
        git commit -m base &&
        git repack -ad &&
index 73a241743aa50101c032b53566de7ba2e7a3b410..82734b9a3c4420a63b91901d165ae06f4b823ea7 100755 (executable)
@@ -63,13 +63,16 @@ TEST_PASSES_SANITIZE_LEAK=true
 # Note that the two variants of "file" must be similar enough to convince git
 # to create the delta.
 make_pack () {
-       {
-               printf '%s\n' "-$(git rev-parse $2)"
-               printf '%s dummy\n' "$(git rev-parse $1:dummy)"
-               printf '%s file\n' "$(git rev-parse $1:file)"
-       } |
-       git pack-objects --stdout |
-       git index-pack --stdin --fix-thin
+       ln1=$(git rev-parse "$2") &&
+       ln2=$(git rev-parse "$1:dummy") &&
+       ln3=$(git rev-parse "$1:file") &&
+       cat >list <<-EOF
+       -$ln1
+       $ln2 dummy
+       $ln3 file
+       EOF
+       git pack-objects --stdout <list >pack &&
+       git index-pack --stdin --fix-thin <pack
 }
 
 test_expect_success 'setup' '
index bb633c9b099ed034c275389cb8e9e9da16efe704..5b707d911b5818f29ed75650256b096bd680168a 100755 (executable)
@@ -24,8 +24,9 @@ parse_verify_pack_blob_oid () {
 }
 
 test_expect_success 'verify blob count in normal packfile' '
-       git -C r1 ls-files -s file.1 file.2 file.3 file.4 file.5 |
-       test_parse_ls_files_stage_oids |
+       git -C r1 ls-files -s file.1 file.2 file.3 file.4 file.5 \
+               >ls_files_result &&
+       test_parse_ls_files_stage_oids <ls_files_result |
        sort >expected &&
 
        git -C r1 pack-objects --revs --stdout >all.pack <<-EOF &&
@@ -123,8 +124,8 @@ test_expect_success 'setup r2' '
 '
 
 test_expect_success 'verify blob count in normal packfile' '
-       git -C r2 ls-files -s large.1000 large.10000 |
-       test_parse_ls_files_stage_oids |
+       git -C r2 ls-files -s large.1000 large.10000 >ls_files_result &&
+       test_parse_ls_files_stage_oids <ls_files_result |
        sort >expected &&
 
        git -C r2 pack-objects --revs --stdout >all.pack <<-EOF &&
@@ -161,8 +162,8 @@ test_expect_success 'verify blob:limit=1000' '
 '
 
 test_expect_success 'verify blob:limit=1001' '
-       git -C r2 ls-files -s large.1000 |
-       test_parse_ls_files_stage_oids |
+       git -C r2 ls-files -s large.1000 >ls_files_result &&
+       test_parse_ls_files_stage_oids <ls_files_result |
        sort >expected &&
 
        git -C r2 pack-objects --revs --stdout --filter=blob:limit=1001 >filter.pack <<-EOF &&
@@ -179,8 +180,8 @@ test_expect_success 'verify blob:limit=1001' '
 '
 
 test_expect_success 'verify blob:limit=10001' '
-       git -C r2 ls-files -s large.1000 large.10000 |
-       test_parse_ls_files_stage_oids |
+       git -C r2 ls-files -s large.1000 large.10000 >ls_files_result &&
+       test_parse_ls_files_stage_oids <ls_files_result |
        sort >expected &&
 
        git -C r2 pack-objects --revs --stdout --filter=blob:limit=10001 >filter.pack <<-EOF &&
@@ -197,8 +198,8 @@ test_expect_success 'verify blob:limit=10001' '
 '
 
 test_expect_success 'verify blob:limit=1k' '
-       git -C r2 ls-files -s large.1000 |
-       test_parse_ls_files_stage_oids |
+       git -C r2 ls-files -s large.1000 >ls_files_result &&
+       test_parse_ls_files_stage_oids <ls_files_result |
        sort >expected &&
 
        git -C r2 pack-objects --revs --stdout --filter=blob:limit=1k >filter.pack <<-EOF &&
@@ -215,8 +216,8 @@ test_expect_success 'verify blob:limit=1k' '
 '
 
 test_expect_success 'verify explicitly specifying oversized blob in input' '
-       git -C r2 ls-files -s large.1000 large.10000 |
-       test_parse_ls_files_stage_oids |
+       git -C r2 ls-files -s large.1000 large.10000 >ls_files_result &&
+       test_parse_ls_files_stage_oids <ls_files_result |
        sort >expected &&
 
        echo HEAD >objects &&
@@ -233,8 +234,8 @@ test_expect_success 'verify explicitly specifying oversized blob in input' '
 '
 
 test_expect_success 'verify blob:limit=1m' '
-       git -C r2 ls-files -s large.1000 large.10000 |
-       test_parse_ls_files_stage_oids |
+       git -C r2 ls-files -s large.1000 large.10000 >ls_files_result &&
+       test_parse_ls_files_stage_oids <ls_files_result |
        sort >expected &&
 
        git -C r2 pack-objects --revs --stdout --filter=blob:limit=1m >filter.pack <<-EOF &&
@@ -264,6 +265,44 @@ test_expect_success 'verify normal and blob:limit packfiles have same commits/tr
        test_cmp expected observed
 '
 
+test_expect_success 'verify small limit and big limit results in small limit' '
+       git -C r2 ls-files -s large.1000 >ls_files_result &&
+       test_parse_ls_files_stage_oids <ls_files_result |
+       sort >expected &&
+
+       git -C r2 pack-objects --revs --stdout --filter=blob:limit=1001 \
+               --filter=blob:limit=10001 >filter.pack <<-EOF &&
+       HEAD
+       EOF
+       git -C r2 index-pack ../filter.pack &&
+
+       git -C r2 verify-pack -v ../filter.pack >verify_result &&
+       grep blob verify_result |
+       parse_verify_pack_blob_oid |
+       sort >observed &&
+
+       test_cmp expected observed
+'
+
+test_expect_success 'verify big limit and small limit results in small limit' '
+       git -C r2 ls-files -s large.1000 >ls_files_result &&
+       test_parse_ls_files_stage_oids <ls_files_result |
+       sort >expected &&
+
+       git -C r2 pack-objects --revs --stdout --filter=blob:limit=10001 \
+               --filter=blob:limit=1001 >filter.pack <<-EOF &&
+       HEAD
+       EOF
+       git -C r2 index-pack ../filter.pack &&
+
+       git -C r2 verify-pack -v ../filter.pack >verify_result &&
+       grep blob verify_result |
+       parse_verify_pack_blob_oid |
+       sort >observed &&
+
+       test_cmp expected observed
+'
+
 # Test sparse:path=<path> filter.
 # !!!!
 # NOTE: sparse:path filter support has been dropped for security reasons,
@@ -289,8 +328,9 @@ test_expect_success 'setup r3' '
 '
 
 test_expect_success 'verify blob count in normal packfile' '
-       git -C r3 ls-files -s sparse1 sparse2 dir1/sparse1 dir1/sparse2 |
-       test_parse_ls_files_stage_oids |
+       git -C r3 ls-files -s sparse1 sparse2 dir1/sparse1 dir1/sparse2 \
+               >ls_files_result &&
+       test_parse_ls_files_stage_oids <ls_files_result |
        sort >expected &&
 
        git -C r3 pack-objects --revs --stdout >all.pack <<-EOF &&
@@ -341,8 +381,9 @@ test_expect_success 'setup r4' '
 '
 
 test_expect_success 'verify blob count in normal packfile' '
-       git -C r4 ls-files -s pattern sparse1 sparse2 dir1/sparse1 dir1/sparse2 |
-       test_parse_ls_files_stage_oids |
+       git -C r4 ls-files -s pattern sparse1 sparse2 dir1/sparse1 dir1/sparse2 \
+               >ls_files_result &&
+       test_parse_ls_files_stage_oids <ls_files_result |
        sort >expected &&
 
        git -C r4 pack-objects --revs --stdout >all.pack <<-EOF &&
@@ -359,8 +400,8 @@ test_expect_success 'verify blob count in normal packfile' '
 '
 
 test_expect_success 'verify sparse:oid=OID' '
-       git -C r4 ls-files -s dir1/sparse1 dir1/sparse2 |
-       test_parse_ls_files_stage_oids |
+       git -C r4 ls-files -s dir1/sparse1 dir1/sparse2 >ls_files_result &&
+       test_parse_ls_files_stage_oids <ls_files_result |
        sort >expected &&
 
        git -C r4 ls-files -s pattern >staged &&
@@ -379,8 +420,8 @@ test_expect_success 'verify sparse:oid=OID' '
 '
 
 test_expect_success 'verify sparse:oid=oid-ish' '
-       git -C r4 ls-files -s dir1/sparse1 dir1/sparse2 |
-       test_parse_ls_files_stage_oids |
+       git -C r4 ls-files -s dir1/sparse1 dir1/sparse2 >ls_files_result &&
+       test_parse_ls_files_stage_oids <ls_files_result |
        sort >expected &&
 
        git -C r4 pack-objects --revs --stdout --filter=sparse:oid=main:pattern >filter.pack <<-EOF &&
@@ -400,8 +441,9 @@ test_expect_success 'verify sparse:oid=oid-ish' '
 # This models previously omitted objects that we did not receive.
 
 test_expect_success 'setup r1 - delete loose blobs' '
-       git -C r1 ls-files -s file.1 file.2 file.3 file.4 file.5 |
-       test_parse_ls_files_stage_oids |
+       git -C r1 ls-files -s file.1 file.2 file.3 file.4 file.5 \
+               >ls_files_result &&
+       test_parse_ls_files_stage_oids <ls_files_result |
        sort >expected &&
 
        for id in `cat expected | sed "s|..|&/|"`
index 54f54f8d2ebde39c846e1c193e93e13e8b509618..1a9e14bbccd3c572d86f5cb15afef37abd8d3d67 100755 (executable)
@@ -1,6 +1,8 @@
 #!/bin/sh
 
 test_description='test custom script in place of pack-objects'
+
+TEST_PASSES_SANITIZE_LEAK=true
 . ./test-lib.sh
 
 test_expect_success 'create some history to fetch' '
index 2ac7b5859e7eefdea13613ba9ae1d238313833dd..06991e8e8aabc28c1a4e522324e5c3eb505cc37e 100755 (executable)
@@ -1,6 +1,8 @@
 #!/bin/sh
 
 test_description='test noop fetch negotiator'
+
+TEST_PASSES_SANITIZE_LEAK=true
 . ./test-lib.sh
 
 test_expect_success 'noop negotiator does not emit any "have"' '
index b2524a24c29796960bfd4dc59ec9a9650e53175e..1928ea1dd7cb9b42e7b6f6db6216c1545bb77a9e 100755 (executable)
@@ -772,6 +772,65 @@ test_expect_success 'reject cloning shallow repository using HTTP' '
        git clone --no-reject-shallow $HTTPD_URL/smart/repo.git repo
 '
 
+test_expect_success 'auto-discover bundle URI from HTTP clone' '
+       test_when_finished rm -rf trace.txt repo2 "$HTTPD_DOCUMENT_ROOT_PATH/repo2.git" &&
+       git -C src bundle create "$HTTPD_DOCUMENT_ROOT_PATH/everything.bundle" --all &&
+       git clone --bare --no-local src "$HTTPD_DOCUMENT_ROOT_PATH/repo2.git" &&
+
+       git -C "$HTTPD_DOCUMENT_ROOT_PATH/repo2.git" config \
+               uploadpack.advertiseBundleURIs true &&
+       git -C "$HTTPD_DOCUMENT_ROOT_PATH/repo2.git" config \
+               bundle.version 1 &&
+       git -C "$HTTPD_DOCUMENT_ROOT_PATH/repo2.git" config \
+               bundle.mode all &&
+       git -C "$HTTPD_DOCUMENT_ROOT_PATH/repo2.git" config \
+               bundle.everything.uri "$HTTPD_URL/everything.bundle" &&
+
+       GIT_TRACE2_EVENT="$(pwd)/trace.txt" \
+               git -c protocol.version=2 \
+                   -c transfer.bundleURI=true clone \
+               $HTTPD_URL/smart/repo2.git repo2 &&
+       cat >pattern <<-EOF &&
+       "event":"child_start".*"argv":\["git-remote-https","$HTTPD_URL/everything.bundle"\]
+       EOF
+       grep -f pattern trace.txt
+'
+
+test_expect_success 'auto-discover multiple bundles from HTTP clone' '
+       test_when_finished rm -rf trace.txt repo3 "$HTTPD_DOCUMENT_ROOT_PATH/repo3.git" &&
+
+       test_commit -C src new &&
+       git -C src bundle create "$HTTPD_DOCUMENT_ROOT_PATH/new.bundle" HEAD~1..HEAD &&
+       git clone --bare --no-local src "$HTTPD_DOCUMENT_ROOT_PATH/repo3.git" &&
+
+       git -C "$HTTPD_DOCUMENT_ROOT_PATH/repo3.git" config \
+               uploadpack.advertiseBundleURIs true &&
+       git -C "$HTTPD_DOCUMENT_ROOT_PATH/repo3.git" config \
+               bundle.version 1 &&
+       git -C "$HTTPD_DOCUMENT_ROOT_PATH/repo3.git" config \
+               bundle.mode all &&
+
+       git -C "$HTTPD_DOCUMENT_ROOT_PATH/repo3.git" config \
+               bundle.everything.uri "$HTTPD_URL/everything.bundle" &&
+       git -C "$HTTPD_DOCUMENT_ROOT_PATH/repo3.git" config \
+               bundle.new.uri "$HTTPD_URL/new.bundle" &&
+
+       GIT_TRACE2_EVENT="$(pwd)/trace.txt" \
+               git -c protocol.version=2 \
+                   -c transfer.bundleURI=true clone \
+               $HTTPD_URL/smart/repo3.git repo3 &&
+
+       # We should fetch _both_ bundles
+       cat >pattern <<-EOF &&
+       "event":"child_start".*"argv":\["git-remote-https","$HTTPD_URL/everything.bundle"\]
+       EOF
+       grep -f pattern trace.txt &&
+       cat >pattern <<-EOF &&
+       "event":"child_start".*"argv":\["git-remote-https","$HTTPD_URL/new.bundle"\]
+       EOF
+       grep -f pattern trace.txt
+'
+
 # DO NOT add non-httpd-specific tests here, because the last part of this
 # test script is only executed when httpd is available and enabled.
 
index a7ec21eda5aabf16144f79a7f348e6f6be9b7507..022ed3d87c37159f758b349f6604b781a29b9327 100755 (executable)
@@ -4,6 +4,7 @@ test_description='test cloning a repository with detached HEAD'
 GIT_TEST_DEFAULT_INITIAL_BRANCH_NAME=main
 export GIT_TEST_DEFAULT_INITIAL_BRANCH_NAME
 
+TEST_PASSES_SANITIZE_LEAK=true
 . ./test-lib.sh
 
 head_is_detached() {
index 4b3877216ee4649a0a31986962740e0f37b5ad85..727caff443368365ca281373ad730de8f3987ab1 100755 (executable)
@@ -4,6 +4,7 @@ test_description='tests for git clone -c key=value'
 GIT_TEST_DEFAULT_INITIAL_BRANCH_NAME=main
 export GIT_TEST_DEFAULT_INITIAL_BRANCH_NAME
 
+TEST_PASSES_SANITIZE_LEAK=true
 . ./test-lib.sh
 
 test_expect_success 'clone -c sets config in cloned repo' '
index 0c85ef834ab90e6fddec2ec510929df1d62acb1d..c2a2bb453eeabcc8725bffae2c537ec99b8a7d87 100755 (executable)
@@ -2,6 +2,7 @@
 
 test_description='Test shallow cloning of repos with submodules'
 
+TEST_PASSES_SANITIZE_LEAK=true
 . ./test-lib.sh
 
 pwd=$(pwd)
index 037941b95d2019f3c65b1b7fa39c03bfc343cd33..f519d2a87a7d84e482284998cbca338a0dfba419 100755 (executable)
@@ -644,6 +644,49 @@ test_expect_success 'repack does not loosen promisor objects' '
        grep "loosen_unused_packed_objects/loosened:0" trace
 '
 
+test_expect_success 'lazy-fetch in submodule succeeds' '
+       # setup
+       test_config_global protocol.file.allow always &&
+
+       test_when_finished "rm -rf src-sub" &&
+       git init src-sub &&
+       git -C src-sub config uploadpack.allowfilter 1 &&
+       git -C src-sub config uploadpack.allowanysha1inwant 1 &&
+
+       # This blob must be missing in the subsequent commit.
+       echo foo >src-sub/file &&
+       git -C src-sub add file &&
+       git -C src-sub commit -m "submodule one" &&
+       SUB_ONE=$(git -C src-sub rev-parse HEAD) &&
+
+       echo bar >src-sub/file &&
+       git -C src-sub add file &&
+       git -C src-sub commit -m "submodule two" &&
+       SUB_TWO=$(git -C src-sub rev-parse HEAD) &&
+
+       test_when_finished "rm -rf src-super" &&
+       git init src-super &&
+       git -C src-super config uploadpack.allowfilter 1 &&
+       git -C src-super config uploadpack.allowanysha1inwant 1 &&
+       git -C src-super submodule add ../src-sub src-sub &&
+
+       git -C src-super/src-sub checkout $SUB_ONE &&
+       git -C src-super add src-sub &&
+       git -C src-super commit -m "superproject one" &&
+
+       git -C src-super/src-sub checkout $SUB_TWO &&
+       git -C src-super add src-sub &&
+       git -C src-super commit -m "superproject two" &&
+
+       # the fetch
+       test_when_finished "rm -rf client" &&
+       git clone --filter=blob:none --also-filter-submodules \
+               --recurse-submodules "file://$(pwd)/src-super" client &&
+
+       # Trigger lazy-fetch from the superproject
+       git -C client restore --recurse-submodules --source=HEAD^ :/
+'
+
 . "$TEST_DIRECTORY"/lib-httpd.sh
 start_httpd
 
index 688433824934d89354669d920acd1fb06a5da725..5a4d7936a72aac48fde876076393956bb83d42e9 100755 (executable)
@@ -5,6 +5,7 @@ test_description='Test cloning repos with submodules using remote-tracking branc
 GIT_TEST_DEFAULT_INITIAL_BRANCH_NAME=main
 export GIT_TEST_DEFAULT_INITIAL_BRANCH_NAME
 
+TEST_PASSES_SANITIZE_LEAK=true
 . ./test-lib.sh
 
 pwd=$(pwd)
index 3353216f09e52450a28fd10d68df9478dcfc129b..f905db0a3fdf406dd564114713cb0236fda42ce6 100755 (executable)
@@ -1,6 +1,8 @@
 #!/bin/sh
 
 test_description='test handling of --alternate-refs traversal'
+
+TEST_PASSES_SANITIZE_LEAK=true
 . ./test-lib.sh
 
 # Avoid test_commit because we want a specific and known set of refs:
index 1896f671cb37f916bc09e50f0f5a45df81da1327..f21e5e9d33d199368df3aeb5ebae64e6b20cc5e3 100755 (executable)
@@ -13,7 +13,7 @@ test_expect_success 'test capability advertisement' '
        wrong_algo sha1:sha256
        wrong_algo sha256:sha1
        EOF
-       cat >expect <<-EOF &&
+       cat >expect.base <<-EOF &&
        version 2
        agent=git/$(git version | cut -d" " -f3)
        ls-refs=unborn
@@ -21,8 +21,11 @@ test_expect_success 'test capability advertisement' '
        server-option
        object-format=$(test_oid algo)
        object-info
+       EOF
+       cat >expect.trailer <<-EOF &&
        0000
        EOF
+       cat expect.base expect.trailer >expect &&
 
        GIT_TEST_SIDEBAND_ALL=0 test-tool serve-v2 \
                --advertise-capabilities >out &&
@@ -342,4 +345,39 @@ test_expect_success 'basics of object-info' '
        test_cmp expect actual
 '
 
+test_expect_success 'test capability advertisement with uploadpack.advertiseBundleURIs' '
+       test_config uploadpack.advertiseBundleURIs true &&
+
+       cat >expect.extra <<-EOF &&
+       bundle-uri
+       EOF
+       cat expect.base \
+           expect.extra \
+           expect.trailer >expect &&
+
+       GIT_TEST_SIDEBAND_ALL=0 test-tool serve-v2 \
+               --advertise-capabilities >out &&
+       test-tool pkt-line unpack <out >actual &&
+       test_cmp expect actual
+'
+
+test_expect_success 'basics of bundle-uri: dies if not enabled' '
+       test-tool pkt-line pack >in <<-EOF &&
+       command=bundle-uri
+       0000
+       EOF
+
+       cat >err.expect <<-\EOF &&
+       fatal: invalid command '"'"'bundle-uri'"'"'
+       EOF
+
+       cat >expect <<-\EOF &&
+       ERR serve: invalid command '"'"'bundle-uri'"'"'
+       EOF
+
+       test_must_fail test-tool serve-v2 --stateless-rpc <in >out 2>err.actual &&
+       test_cmp err.expect err.actual &&
+       test_must_be_empty out
+'
+
 test_done
diff --git a/t/t5730-protocol-v2-bundle-uri-file.sh b/t/t5730-protocol-v2-bundle-uri-file.sh
new file mode 100755 (executable)
index 0000000..37bdb72
--- /dev/null
@@ -0,0 +1,17 @@
+#!/bin/sh
+
+test_description="Test bundle-uri with protocol v2 and 'file://' transport"
+
+TEST_NO_CREATE_REPO=1
+
+GIT_TEST_DEFAULT_INITIAL_BRANCH_NAME=main
+export GIT_TEST_DEFAULT_INITIAL_BRANCH_NAME
+
+. ./test-lib.sh
+
+# Test protocol v2 with 'file://' transport
+#
+BUNDLE_URI_PROTOCOL=file
+. "$TEST_DIRECTORY"/lib-bundle-uri-protocol.sh
+
+test_done
diff --git a/t/t5731-protocol-v2-bundle-uri-git.sh b/t/t5731-protocol-v2-bundle-uri-git.sh
new file mode 100755 (executable)
index 0000000..8add1b3
--- /dev/null
@@ -0,0 +1,17 @@
+#!/bin/sh
+
+test_description="Test bundle-uri with protocol v2 and 'git://' transport"
+
+TEST_NO_CREATE_REPO=1
+
+GIT_TEST_DEFAULT_INITIAL_BRANCH_NAME=main
+export GIT_TEST_DEFAULT_INITIAL_BRANCH_NAME
+
+. ./test-lib.sh
+
+# Test protocol v2 with 'git://' transport
+#
+BUNDLE_URI_PROTOCOL=git
+. "$TEST_DIRECTORY"/lib-bundle-uri-protocol.sh
+
+test_done
diff --git a/t/t5732-protocol-v2-bundle-uri-http.sh b/t/t5732-protocol-v2-bundle-uri-http.sh
new file mode 100755 (executable)
index 0000000..129daa0
--- /dev/null
@@ -0,0 +1,17 @@
+#!/bin/sh
+
+test_description="Test bundle-uri with protocol v2 and 'http://' transport"
+
+TEST_NO_CREATE_REPO=1
+
+GIT_TEST_DEFAULT_INITIAL_BRANCH_NAME=main
+export GIT_TEST_DEFAULT_INITIAL_BRANCH_NAME
+
+. ./test-lib.sh
+
+# Test protocol v2 with 'http://' transport
+#
+BUNDLE_URI_PROTOCOL=http
+. "$TEST_DIRECTORY"/lib-bundle-uri-protocol.sh
+
+test_done
index c2fe3f9c5a5c40543e8c2c0b65949c0dbfc1b594..7b4f930e532a90f7aec76e1047b4bec79f855f21 100755 (executable)
@@ -30,6 +30,58 @@ test_expect_success 'bundle_uri_parse_line() just URIs' '
        test_cmp_config_output expect actual
 '
 
+test_expect_success 'bundle_uri_parse_line(): relative URIs' '
+       cat >in <<-\EOF &&
+       bundle.one.uri=bundle.bdl
+       bundle.two.uri=../bundle.bdl
+       bundle.three.uri=sub/dir/bundle.bdl
+       EOF
+
+       cat >expect <<-\EOF &&
+       [bundle]
+               version = 1
+               mode = all
+       [bundle "one"]
+               uri = <uri>/bundle.bdl
+       [bundle "two"]
+               uri = bundle.bdl
+       [bundle "three"]
+               uri = <uri>/sub/dir/bundle.bdl
+       EOF
+
+       test-tool bundle-uri parse-key-values in >actual 2>err &&
+       test_must_be_empty err &&
+       test_cmp_config_output expect actual
+'
+
+test_expect_success 'bundle_uri_parse_line(): relative URIs and parent paths' '
+       cat >in <<-\EOF &&
+       bundle.one.uri=bundle.bdl
+       bundle.two.uri=../bundle.bdl
+       bundle.three.uri=../../bundle.bdl
+       EOF
+
+       cat >expect <<-\EOF &&
+       [bundle]
+               version = 1
+               mode = all
+       [bundle "one"]
+               uri = <uri>/bundle.bdl
+       [bundle "two"]
+               uri = bundle.bdl
+       [bundle "three"]
+               uri = <uri>/../bundle.bdl
+       EOF
+
+       # TODO: We would prefer if parsing a bundle list would not cause
+       # a die() and instead would give a warning and allow the rest of
+       # a Git command to continue. This test_must_fail is necessary for
+       # now until the interface for relative_url() allows for reporting
+       # an error instead of die()ing.
+       test_must_fail test-tool bundle-uri parse-key-values in >actual 2>err &&
+       grep "fatal: cannot strip one component off url" err
+'
+
 test_expect_success 'bundle_uri_parse_line() parsing edge cases: empty key or value' '
        cat >in <<-\EOF &&
        =bogus-value
@@ -136,6 +188,36 @@ test_expect_success 'parse config format: just URIs' '
        test_cmp_config_output expect actual
 '
 
+test_expect_success 'parse config format: relative URIs' '
+       cat >in <<-\EOF &&
+       [bundle]
+               version = 1
+               mode = all
+       [bundle "one"]
+               uri = bundle.bdl
+       [bundle "two"]
+               uri = ../bundle.bdl
+       [bundle "three"]
+               uri = sub/dir/bundle.bdl
+       EOF
+
+       cat >expect <<-\EOF &&
+       [bundle]
+               version = 1
+               mode = all
+       [bundle "one"]
+               uri = <uri>/bundle.bdl
+       [bundle "two"]
+               uri = bundle.bdl
+       [bundle "three"]
+               uri = <uri>/sub/dir/bundle.bdl
+       EOF
+
+       test-tool bundle-uri parse-config in >actual 2>err &&
+       test_must_be_empty err &&
+       test_cmp_config_output expect actual
+'
+
 test_expect_success 'parse config format edge cases: empty key or value' '
        cat >in1 <<-\EOF &&
        = bogus-value
index 833205125abf5a2673ae0cbede62653fbfd151c8..3a1cf30b1d737e3bc5cb34039822b7d45ab99445 100755 (executable)
@@ -11,6 +11,13 @@ export GIT_TEST_DEFAULT_INITIAL_BRANCH_NAME
 . ./test-lib.sh
 . "$TEST_DIRECTORY"/lib-bundle.sh
 
+for cmd in create verify list-heads unbundle
+do
+       test_expect_success "usage: git bundle $cmd needs an argument" '
+               test_expect_code 129 git bundle $cmd
+       '
+done
+
 # Create a commit or tag and set the variable with the object ID.
 test_commit_setvar () {
        notick=
index 6dbbe62eb2203c4c12619463e01e438d0bfa93bc..98a72ff78a78905df62869a9a1177af5324db01c 100755 (executable)
@@ -34,6 +34,36 @@ HASH2=
 HASH3=
 HASH4=
 
+test_bisect_usage () {
+       local code="$1" &&
+       shift &&
+       cat >expect &&
+       test_expect_code $code "$@" >out 2>actual &&
+       test_must_be_empty out &&
+       test_cmp expect actual
+}
+
+test_expect_success 'bisect usage' "
+       test_bisect_usage 1 git bisect reset extra1 extra2 <<-\EOF &&
+       error: 'git bisect reset' requires either no argument or a commit
+       EOF
+       test_bisect_usage 1 git bisect terms extra1 extra2 <<-\EOF &&
+       error: 'git bisect terms' requires 0 or 1 argument
+       EOF
+       test_bisect_usage 1 git bisect next extra1 <<-\EOF &&
+       error: 'git bisect next' requires 0 arguments
+       EOF
+       test_bisect_usage 1 git bisect log extra1 <<-\EOF &&
+       error: We are not bisecting.
+       EOF
+       test_bisect_usage 1 git bisect replay <<-\EOF &&
+       error: no logfile given
+       EOF
+       test_bisect_usage 1 git bisect run <<-\EOF
+       error: 'git bisect run' failed: no command provided.
+       EOF
+"
+
 test_expect_success 'set up basic repo with 1 file (hello) and 4 commits' '
      add_line_into_file "1: Hello World" hello &&
      HASH1=$(git rev-parse --verify HEAD) &&
@@ -252,6 +282,124 @@ test_expect_success 'bisect skip: with commit both bad and skipped' '
        grep $HASH4 my_bisect_log.txt
 '
 
+test_bisect_run_args () {
+       test_when_finished "rm -f run.sh actual" &&
+       >actual &&
+       cat >expect.args &&
+       cat <&6 >expect.out &&
+       cat <&7 >expect.err &&
+       write_script run.sh <<-\EOF &&
+       while test $# != 0
+       do
+               echo "<$1>" &&
+               shift
+       done >actual.args
+       EOF
+
+       test_when_finished "git bisect reset" &&
+       git bisect start &&
+       git bisect good $HASH1 &&
+       git bisect bad $HASH4 &&
+       git bisect run ./run.sh $@ >actual.out.raw 2>actual.err &&
+       # Prune just the log output
+       sed -n \
+               -e '/^Author:/d' \
+               -e '/^Date:/d' \
+               -e '/^$/d' \
+               -e '/^commit /d' \
+               -e '/^ /d' \
+               -e 'p' \
+               <actual.out.raw >actual.out &&
+       test_cmp expect.out actual.out &&
+       test_cmp expect.err actual.err &&
+       test_cmp expect.args actual.args
+}
+
+test_expect_success 'git bisect run: args, stdout and stderr with no arguments' "
+       test_bisect_run_args <<-'EOF_ARGS' 6<<-EOF_OUT 7<<-'EOF_ERR'
+       EOF_ARGS
+       running './run.sh'
+       $HASH4 is the first bad commit
+       bisect found first bad commit
+       EOF_OUT
+       EOF_ERR
+"
+
+test_expect_success 'git bisect run: args, stdout and stderr: "--" argument' "
+       test_bisect_run_args -- <<-'EOF_ARGS' 6<<-EOF_OUT 7<<-'EOF_ERR'
+       <-->
+       EOF_ARGS
+       running './run.sh' '--'
+       $HASH4 is the first bad commit
+       bisect found first bad commit
+       EOF_OUT
+       EOF_ERR
+"
+
+test_expect_success 'git bisect run: args, stdout and stderr: "--log foo --no-log bar" arguments' "
+       test_bisect_run_args --log foo --no-log bar <<-'EOF_ARGS' 6<<-EOF_OUT 7<<-'EOF_ERR'
+       <--log>
+       <foo>
+       <--no-log>
+       <bar>
+       EOF_ARGS
+       running './run.sh' '--log' 'foo' '--no-log' 'bar'
+       $HASH4 is the first bad commit
+       bisect found first bad commit
+       EOF_OUT
+       EOF_ERR
+"
+
+test_expect_success 'git bisect run: args, stdout and stderr: "--bisect-start" argument' "
+       test_bisect_run_args --bisect-start <<-'EOF_ARGS' 6<<-EOF_OUT 7<<-'EOF_ERR'
+       <--bisect-start>
+       EOF_ARGS
+       running './run.sh' '--bisect-start'
+       $HASH4 is the first bad commit
+       bisect found first bad commit
+       EOF_OUT
+       EOF_ERR
+"
+
+test_expect_success 'git bisect run: negative exit code' "
+       write_script fail.sh <<-'EOF' &&
+       exit 255
+       EOF
+       cat <<-'EOF' >expect &&
+       bisect run failed: exit code -1 from './fail.sh' is < 0 or >= 128
+       EOF
+       test_when_finished 'git bisect reset' &&
+       git bisect start &&
+       git bisect good $HASH1 &&
+       git bisect bad $HASH4 &&
+       ! git bisect run ./fail.sh 2>err &&
+       sed -En 's/.*(bisect.*code) (-?[0-9]+) (from.*)/\1 -1 \3/p' err >actual &&
+       test_cmp expect actual
+"
+
+test_expect_success 'git bisect run: unable to verify on good' "
+       write_script fail.sh <<-'EOF' &&
+       head=\$(git rev-parse --verify HEAD)
+       good=\$(git rev-parse --verify $HASH1)
+       if test "\$head" = "\$good"
+       then
+               exit 255
+       else
+               exit 127
+       fi
+       EOF
+       cat <<-'EOF' >expect &&
+       unable to verify './fail.sh' on good revision
+       EOF
+       test_when_finished 'git bisect reset' &&
+       git bisect start &&
+       git bisect good $HASH1 &&
+       git bisect bad $HASH4 &&
+       ! git bisect run ./fail.sh 2>err &&
+       sed -n 's/.*\(unable to verify.*\)/\1/p' err >actual &&
+       test_cmp expect actual
+"
+
 # We want to automatically find the commit that
 # added "Another" into hello.
 test_expect_success '"git bisect run" simple case' '
index ed449abe5520e59ddb0ad27def5711007f620650..1a8b64cce18239284a28260734a0348f95a2d7b6 100755 (executable)
@@ -1,6 +1,8 @@
 #!/bin/sh
 
 test_description='basic git merge-index / git-merge-one-file tests'
+
+TEST_PASSES_SANITIZE_LEAK=true
 . ./test-lib.sh
 
 test_expect_success 'setup diverging branches' '
index fa38b874416220d3d3c39ee48a99ad53b94dc2df..2ae1fc721b1c477e8830d15eeda323c7e4b9bb7a 100755 (executable)
@@ -1242,6 +1242,24 @@ test_expect_success 'basic atom: rest must fail' '
        test_must_fail git for-each-ref --format="%(rest)" refs/heads/main
 '
 
+test_expect_success 'HEAD atom does not take arguments' '
+       test_must_fail git for-each-ref --format="%(HEAD:foo)" 2>err &&
+       echo "fatal: %(HEAD) does not take arguments" >expect &&
+       test_cmp expect err
+'
+
+test_expect_success 'subject atom rejects unknown arguments' '
+       test_must_fail git for-each-ref --format="%(subject:foo)" 2>err &&
+       echo "fatal: unrecognized %(subject) argument: foo" >expect &&
+       test_cmp expect err
+'
+
+test_expect_success 'refname atom rejects unknown arguments' '
+       test_must_fail git for-each-ref --format="%(refname:foo)" 2>err &&
+       echo "fatal: unrecognized %(refname) argument: foo" >expect &&
+       test_cmp expect err
+'
+
 test_expect_success 'trailer parsing not fooled by --- line' '
        git commit --allow-empty -F - <<-\EOF &&
        this is the subject
index 40edf9dab53402db9b5fad6dd060768537f9ba9a..bfda1f46ad25f49c7d51ce2e5853b1818314010f 100755 (executable)
@@ -2,6 +2,7 @@
 
 test_description='for-each-ref errors for broken refs'
 
+TEST_PASSES_SANITIZE_LEAK=true
 . ./test-lib.sh
 
 ZEROS=$ZERO_OID
index 9d5e992878fad10f20c404a0dd676261345b2973..1962310408b12209aef235b9d6e09b8049de5f60 100755 (executable)
@@ -8,6 +8,8 @@
 
 
 test_description='Test criss-cross merge'
+
+TEST_PASSES_SANITIZE_LEAK=true
 . ./test-lib.sh
 
 test_expect_success 'prepare repository' '
index 8650a88c40a2a58182ee39f36cdc59315ae1a35c..5e4e4dd6d9e6b967f8c1d94fe7ce1a79c787a177 100755 (executable)
@@ -8,6 +8,7 @@ test_description='per path merge controlled by merge attribute'
 GIT_TEST_DEFAULT_INITIAL_BRANCH_NAME=main
 export GIT_TEST_DEFAULT_INITIAL_BRANCH_NAME
 
+TEST_PASSES_SANITIZE_LEAK=true
 . ./test-lib.sh
 
 test_expect_success setup '
index e8a28717cece3248c2d9996e797dcfdb9af128eb..0753fc95f45efb642543f9f23191d3430d4d6cde 100755 (executable)
@@ -5,6 +5,7 @@ test_description='ask merge-recursive to merge binary files'
 GIT_TEST_DEFAULT_INITIAL_BRANCH_NAME=main
 export GIT_TEST_DEFAULT_INITIAL_BRANCH_NAME
 
+TEST_PASSES_SANITIZE_LEAK=true
 . ./test-lib.sh
 
 test_expect_success setup '
index 2655e295f5ae4536eaf86dcf95e8062f61cc2948..ae00492c768217a46d73fde7cfbc330dcc502616 100755 (executable)
@@ -4,6 +4,7 @@ test_description='merging when a directory was replaced with a symlink'
 GIT_TEST_DEFAULT_INITIAL_BRANCH_NAME=main
 export GIT_TEST_DEFAULT_INITIAL_BRANCH_NAME
 
+TEST_PASSES_SANITIZE_LEAK=true
 . ./test-lib.sh
 
 test_expect_success 'create a commit where dir a/b changed to symlink' '
index fde4aa3cd1ab66a13c61234155da0c2bacb851a9..78628fb248ab6143ef83d11e819f15c30b52f278 100755 (executable)
@@ -3,6 +3,7 @@
 test_description='merge with sparse files'
 
 TEST_CREATE_REPO_NO_TEMPLATE=1
+TEST_PASSES_SANITIZE_LEAK=true
 . ./test-lib.sh
 
 # test_file $filename $content
index a60153f9f32318c5ee31d2b16ec506a7a59ddd6d..18bbd9975ebfda330d0727c600bd0cec4b8b5181 100755 (executable)
@@ -63,7 +63,7 @@ test_expect_success '"mixed" reset is not allowed in bare' '
        test_must_fail git reset --mixed HEAD^
 '
 
-test_expect_success !SANITIZE_LEAK '"soft" reset is allowed in bare' '
+test_expect_success '"soft" reset is allowed in bare' '
        git reset --soft HEAD^ &&
        git show --pretty=format:%s >out &&
        echo one >expect &&
index 2859695c6d208341f5ba1de01e1cfe4131388a2b..f77832185765585e2bda1677f8cbbe13841127f7 100755 (executable)
@@ -10,6 +10,7 @@ TEST_PASSES_SANITIZE_LEAK=true
 . ./test-lib.sh
 
 test_expect_success 'setup a real submodule' '
+       cwd="$(pwd)" &&
        git init sub1 &&
        test_commit -C sub1 first &&
        git submodule add ./sub1 &&
@@ -18,13 +19,21 @@ test_expect_success 'setup a real submodule' '
 '
 
 test_expect_success 'absorb the git dir' '
+       >expect &&
+       >actual &&
        >expect.1 &&
        >expect.2 &&
        >actual.1 &&
        >actual.2 &&
        git status >expect.1 &&
        git -C sub1 rev-parse HEAD >expect.2 &&
-       git submodule absorbgitdirs &&
+       cat >expect <<-EOF &&
+       Migrating git directory of '\''sub1'\'' from
+       '\''$cwd/sub1/.git'\'' to
+       '\''$cwd/.git/modules/sub1'\''
+       EOF
+       git submodule absorbgitdirs 2>actual &&
+       test_cmp expect actual &&
        git fsck &&
        test -f sub1/.git &&
        test -d .git/modules/sub1 &&
@@ -37,7 +46,8 @@ test_expect_success 'absorb the git dir' '
 test_expect_success 'absorbing does not fail for deinitialized submodules' '
        test_when_finished "git submodule update --init" &&
        git submodule deinit --all &&
-       git submodule absorbgitdirs &&
+       git submodule absorbgitdirs 2>err &&
+       test_must_be_empty err &&
        test -d .git/modules/sub1 &&
        test -d sub1 &&
        ! test -e sub1/.git
@@ -56,7 +66,13 @@ test_expect_success 'setup nested submodule' '
 test_expect_success 'absorb the git dir in a nested submodule' '
        git status >expect.1 &&
        git -C sub1/nested rev-parse HEAD >expect.2 &&
-       git submodule absorbgitdirs &&
+       cat >expect <<-EOF &&
+       Migrating git directory of '\''sub1/nested'\'' from
+       '\''$cwd/sub1/nested/.git'\'' to
+       '\''$cwd/.git/modules/sub1/modules/nested'\''
+       EOF
+       git submodule absorbgitdirs 2>actual &&
+       test_cmp expect actual &&
        test -f sub1/nested/.git &&
        test -d .git/modules/sub1/modules/nested &&
        git status >actual.1 &&
@@ -87,7 +103,13 @@ test_expect_success 're-setup nested submodule' '
 test_expect_success 'absorb the git dir in a nested submodule' '
        git status >expect.1 &&
        git -C sub1/nested rev-parse HEAD >expect.2 &&
-       git submodule absorbgitdirs &&
+       cat >expect <<-EOF &&
+       Migrating git directory of '\''sub1'\'' from
+       '\''$cwd/sub1/.git'\'' to
+       '\''$cwd/.git/modules/sub1'\''
+       EOF
+       git submodule absorbgitdirs 2>actual &&
+       test_cmp expect actual &&
        test -f sub1/.git &&
        test -f sub1/nested/.git &&
        test -d .git/modules/sub1/modules/nested &&
@@ -97,6 +119,27 @@ test_expect_success 'absorb the git dir in a nested submodule' '
        test_cmp expect.2 actual.2
 '
 
+test_expect_success 'absorb the git dir outside of primary worktree' '
+       test_when_finished "rm -rf repo-bare.git" &&
+       git clone --bare . repo-bare.git &&
+       test_when_finished "rm -rf repo-wt" &&
+       git -C repo-bare.git worktree add ../repo-wt &&
+
+       test_when_finished "rm -f .gitconfig" &&
+       test_config_global protocol.file.allow always &&
+       git -C repo-wt submodule update --init &&
+       git init repo-wt/sub2 &&
+       test_commit -C repo-wt/sub2 A &&
+       git -C repo-wt submodule add ./sub2 sub2 &&
+       cat >expect <<-EOF &&
+       Migrating git directory of '\''sub2'\'' from
+       '\''$cwd/repo-wt/sub2/.git'\'' to
+       '\''$cwd/repo-bare.git/worktrees/repo-wt/modules/sub2'\''
+       EOF
+       git -C repo-wt submodule absorbgitdirs 2>actual &&
+       test_cmp expect actual
+'
+
 test_expect_success 'setup a gitlink with missing .gitmodules entry' '
        git init sub2 &&
        test_commit -C sub2 first &&
@@ -107,7 +150,11 @@ test_expect_success 'setup a gitlink with missing .gitmodules entry' '
 test_expect_success 'absorbing the git dir fails for incomplete submodules' '
        git status >expect.1 &&
        git -C sub2 rev-parse HEAD >expect.2 &&
-       test_must_fail git submodule absorbgitdirs &&
+       cat >expect <<-\EOF &&
+       fatal: could not lookup name for submodule '\''sub2'\''
+       EOF
+       test_must_fail git submodule absorbgitdirs 2>actual &&
+       test_cmp expect actual &&
        git -C sub2 fsck &&
        test -d sub2/.git &&
        git status >actual &&
@@ -127,8 +174,11 @@ test_expect_success 'setup a submodule with multiple worktrees' '
 '
 
 test_expect_success 'absorbing fails for a submodule with multiple worktrees' '
-       test_must_fail git submodule absorbgitdirs sub3 2>error &&
-       test_i18ngrep "not supported" error
+       cat >expect <<-\EOF &&
+       fatal: could not lookup name for submodule '\''sub2'\''
+       EOF
+       test_must_fail git submodule absorbgitdirs 2>actual &&
+       test_cmp expect actual
 '
 
 test_done
index a39de8c112674f070f8dc19745e27980b7b52c9c..07ca46fb0d539c62c7dcbe066b8c95e550d0b3a3 100755 (executable)
@@ -5,6 +5,7 @@ test_description='commit-msg hook'
 GIT_TEST_DEFAULT_INITIAL_BRANCH_NAME=main
 export GIT_TEST_DEFAULT_INITIAL_BRANCH_NAME
 
+TEST_PASSES_SANITIZE_LEAK=true
 . ./test-lib.sh
 
 test_expect_success 'with no hook' '
index 2b7ef6c41a455423c0c62087a440e51bc4c72486..aed07c5b622e9a244313e781ceba0e6898024d21 100755 (executable)
@@ -1676,4 +1676,74 @@ test_expect_success 'racy timestamps will be fixed for dirty worktree' '
        ! test_is_magic_mtime .git/index
 '
 
+test_expect_success 'setup slow status advice' '
+       GIT_TEST_DEFAULT_INITIAL_BRANCH_NAME=main git init slowstatus &&
+       (
+               cd slowstatus &&
+               cat >.gitignore <<-\EOF &&
+               /actual
+               /expected
+               /out
+               EOF
+               git add .gitignore &&
+               git commit -m "Add .gitignore" &&
+               git config advice.statusuoption true
+       )
+'
+
+test_expect_success 'slow status advice when core.untrackedCache and fsmonitor are unset' '
+       (
+               cd slowstatus &&
+               git config core.untrackedCache false &&
+               git config core.fsmonitor false &&
+               GIT_TEST_UF_DELAY_WARNING=1 git status >actual &&
+               cat >expected <<-\EOF &&
+               On branch main
+
+               It took 3.25 seconds to enumerate untracked files.
+               See '\''git help status'\'' for information on how to improve this.
+
+               nothing to commit, working tree clean
+               EOF
+               test_cmp expected actual
+       )
+'
+
+test_expect_success 'slow status advice when core.untrackedCache true, but not fsmonitor' '
+       (
+               cd slowstatus &&
+               git config core.untrackedCache true &&
+               git config core.fsmonitor false &&
+               GIT_TEST_UF_DELAY_WARNING=1 git status >actual &&
+               cat >expected <<-\EOF &&
+               On branch main
+
+               It took 3.25 seconds to enumerate untracked files.
+               See '\''git help status'\'' for information on how to improve this.
+
+               nothing to commit, working tree clean
+               EOF
+               test_cmp expected actual
+       )
+'
+
+test_expect_success 'slow status advice when core.untrackedCache true, and fsmonitor' '
+       (
+               cd slowstatus &&
+               git config core.untrackedCache true &&
+               git config core.fsmonitor true &&
+               GIT_TEST_UF_DELAY_WARNING=1 git status >actual &&
+               cat >expected <<-\EOF &&
+               On branch main
+
+               It took 3.25 seconds to enumerate untracked files,
+               but the results were cached, and subsequent runs may be faster.
+               See '\''git help status'\'' for information on how to improve this.
+
+               nothing to commit, working tree clean
+               EOF
+               test_cmp expected actual
+       )
+'
+
 test_done
index 163ae8046850e729ef5e329f6589faf5381e9cdc..efc6496e2b27f3ec74431d625e64e52c384e0383 100755 (executable)
@@ -9,6 +9,7 @@ test_description='per-repo forced setting of email address'
 GIT_TEST_DEFAULT_INITIAL_BRANCH_NAME=main
 export GIT_TEST_DEFAULT_INITIAL_BRANCH_NAME
 
+TEST_PASSES_SANITIZE_LEAK=true
 . ./test-lib.sh
 
 test_expect_success 'setup a likely user.useConfigOnly use case' '
index dc57526e6f1f7af741a4cd6c1ce4800563be5c37..184b2589893b78ef1e77521a6a9e7f0e5b514de9 100755 (executable)
@@ -2,6 +2,7 @@
 
 test_description='ignored hook warning'
 
+TEST_PASSES_SANITIZE_LEAK=true
 . ./test-lib.sh
 
 test_expect_success setup '
index 4abc74db2bb8ede3879ab32ea771d688b8211eac..76d0220daa03f667e4d2950fc3f89a0ade83bfc4 100755 (executable)
@@ -866,27 +866,9 @@ test_expect_success 'submodule always visited' '
 # the submodule, and someone does a `git submodule absorbgitdirs`
 # in the super, Git will recursively invoke `git submodule--helper`
 # to do the work and this may try to read the index.  This will
-# try to start the daemon in the submodule *and* pass (either
-# directly or via inheritance) the `--super-prefix` arg to the
-# `git fsmonitor--daemon start` command inside the submodule.
-# This causes a warning because fsmonitor--daemon does take that
-# global arg (see the table in git.c)
-#
-# This causes a warning when trying to start the daemon that is
-# somewhat confusing.  It does not seem to hurt anything because
-# the fsmonitor code maps the query failure into a trivial response
-# and does the work anyway.
-#
-# It would be nice to silence the warning, however.
-
-have_t2_error_event () {
-       log=$1
-       msg="fsmonitor--daemon doesnQt support --super-prefix" &&
-
-       tr '\047' Q <$1 | grep -e "$msg"
-}
+# try to start the daemon in the submodule.
 
-test_expect_success "stray submodule super-prefix warning" '
+test_expect_success "submodule absorbgitdirs implicitly starts daemon" '
        test_when_finished "rm -rf super; \
                            rm -rf sub;   \
                            rm super-sub.trace" &&
@@ -904,10 +886,20 @@ test_expect_success "stray submodule super-prefix warning" '
 
        test_path_is_dir super/dir_1/dir_2/sub/.git &&
 
+       cwd="$(cd super && pwd)" &&
+       cat >expect <<-EOF &&
+       Migrating git directory of '\''dir_1/dir_2/sub'\'' from
+       '\''$cwd/dir_1/dir_2/sub/.git'\'' to
+       '\''$cwd/.git/modules/dir_1/dir_2/sub'\''
+       EOF
        GIT_TRACE2_EVENT="$PWD/super-sub.trace" \
-               git -C super submodule absorbgitdirs &&
+               git -C super submodule absorbgitdirs >out 2>actual &&
+       test_cmp expect actual &&
+       test_must_be_empty out &&
 
-       ! have_t2_error_event super-sub.trace
+       # Confirm that the trace2 log contains a record of the
+       # daemon starting.
+       test_subcommand git fsmonitor--daemon start <super-sub.trace
 '
 
 # On a case-insensitive file system, confirm that the daemon
index 7c3f6ed99431839abc42533d520cd342f25dafd1..060e145957f3ad94933b37ebe6148ffbdf168d37 100755 (executable)
@@ -105,7 +105,7 @@ verify_mergeheads () {
        test_write_lines "$@" >mergehead.expected &&
        while read sha1 rest
        do
-               git rev-parse $sha1
+               git rev-parse $sha1 || return 1
        done <.git/MERGE_HEAD >mergehead.actual &&
        test_cmp mergehead.expected mergehead.actual
 }
index 5d56c3854647b2f2bc2fdf4221b1f76f5d0b0629..62d935d31c2e1dc9cbca3779407b6b86bc5ce4c5 100755 (executable)
@@ -4,6 +4,7 @@ test_description='git merge
 
 Testing the resolve strategy.'
 
+TEST_PASSES_SANITIZE_LEAK=true
 . ./test-lib.sh
 
 test_expect_success 'setup' '
index fee258d4f0c2c0d4b230891eeb93d522d27a4979..cf96a35e8e7b20975628cce6a5882c18c32fbeaa 100755 (executable)
@@ -8,6 +8,7 @@ This test runs git merge --signoff and makes sure that it works.
 GIT_TEST_DEFAULT_INITIAL_BRANCH_NAME=main
 export GIT_TEST_DEFAULT_INITIAL_BRANCH_NAME
 
+TEST_PASSES_SANITIZE_LEAK=true
 . ./test-lib.sh
 
 # Setup test files
index f00deaf3815f3b49c42da4021335e011bbcb3c32..14a704d0a8c311ff776f65313178073e70b0a38f 100755 (executable)
@@ -1,6 +1,8 @@
 #!/bin/sh
 
 test_description='help.autocorrect finding a match'
+
+TEST_PASSES_SANITIZE_LEAK=true
 . ./test-lib.sh
 
 test_expect_success 'setup' '
@@ -60,4 +62,10 @@ test_expect_success 'autocorrect can be declined altogether' '
        test_line_count = 1 actual
 '
 
+test_expect_success 'autocorrect works in work tree created from bare repo' '
+       git clone --bare . bare.git &&
+       git -C bare.git worktree add ../worktree &&
+       git -C worktree -c help.autocorrect=immediate stauts
+'
+
 test_done
index 419f055721dbc32122e9d595d691c77bb8a24212..743fbe1fe46929583aacacc60ba959c38d8a2f09 100755 (executable)
@@ -5,7 +5,6 @@
 
 test_description='git svn dcommit can commit renames of files with ugly names'
 
-TEST_FAILS_SANITIZE_LEAK=true
 . ./lib-git-svn.sh
 
 test_expect_success 'load repository with strange names' '
index 8201c3e808a931d07bc7e135f87bcc0499f65e20..088d1c57a88a209d57bdc95e45368f40af87c7a6 100755 (executable)
@@ -28,7 +28,7 @@ test_cmp_info () {
        rm -f tmp.expect tmp.actual
 }
 
-quoted_svnrepo="$(echo $svnrepo | sed 's/ /%20/')"
+quoted_svnrepo="$(echo $svnrepo | test_uri_escape)"
 
 test_expect_success 'setup repository and import' '
        mkdir info &&
index 79c26ed69c16cad42a32a555bb33635a154441e7..09606f1b3cfeb4244de4fbc12e4e96ddb7fd7de5 100755 (executable)
@@ -4,7 +4,6 @@
 
 test_description='git svn creates empty directories'
 
-TEST_FAILS_SANITIZE_LEAK=true
 . ./lib-git-svn.sh
 
 test_expect_success 'initialize repo' '
index 6cc76a07b39a90d590759bbb7033da3953969092..aebb28995e50a0a589e8fff33997d9816de1fdb2 100755 (executable)
@@ -5,7 +5,6 @@
 
 test_description='git svn propset tests'
 
-TEST_FAILS_SANITIZE_LEAK=true
 . ./lib-git-svn.sh
 
 test_expect_success 'setup propset via import' '
index 9cf7a1427ab081f2084b147b7dc4b1e3aa0f1005..36c6b1a12ffd95a1d6fe4ee4efd18a47f1581078 100755 (executable)
@@ -9,7 +9,6 @@ This test uses git to clone a Subversion repository that contains empty
 directories, and checks that corresponding directories are created in the
 local Git repository with placeholder files.'
 
-TEST_FAILS_SANITIZE_LEAK=true
 . ./lib-git-svn.sh
 
 GIT_REPO=git-svn-repo
index 43de868b8005d330c10ec5d81fcacdc325aff858..d6c0478d98b85412307a422b142386c180ad030a 100755 (executable)
@@ -2255,6 +2255,36 @@ test_expect_success 'checkout completes ref names' '
        EOF
 '
 
+test_expect_success 'checkout does not match ref names of a different case' '
+       test_completion "git checkout M" ""
+'
+
+test_expect_success 'checkout matches case insensitively with GIT_COMPLETION_IGNORE_CASE' '
+       (
+               GIT_COMPLETION_IGNORE_CASE=1 &&
+               test_completion "git checkout M" <<-\EOF
+               main Z
+               mybranch Z
+               mytag Z
+               EOF
+       )
+'
+
+test_expect_success 'checkout completes pseudo refs' '
+       test_completion "git checkout H" <<-\EOF
+       HEAD Z
+       EOF
+'
+
+test_expect_success 'checkout completes pseudo refs case insensitively with GIT_COMPLETION_IGNORE_CASE' '
+       (
+               GIT_COMPLETION_IGNORE_CASE=1 &&
+               test_completion "git checkout h" <<-\EOF
+               HEAD Z
+               EOF
+       )
+'
+
 test_expect_success 'git -C <path> checkout uses the right repo' '
        test_completion "git -C subdir -C subsubdir -C .. -C ../otherrepo checkout b" <<-\EOF
        branch-in-other Z
index 60308843f8f4f0063dd5958004c3146e0993ac08..1b8d4344a56dd7e8ed7d3e03db038bb1fb5110cf 100644 (file)
@@ -1450,72 +1450,6 @@ test_skip_or_die () {
        error "$2"
 }
 
-# The following mingw_* functions obey POSIX shell syntax, but are actually
-# bash scripts, and are meant to be used only with bash on Windows.
-
-# A test_cmp function that treats LF and CRLF equal and avoids to fork
-# diff when possible.
-mingw_test_cmp () {
-       # Read text into shell variables and compare them. If the results
-       # are different, use regular diff to report the difference.
-       local test_cmp_a= test_cmp_b=
-
-       # When text came from stdin (one argument is '-') we must feed it
-       # to diff.
-       local stdin_for_diff=
-
-       # Since it is difficult to detect the difference between an
-       # empty input file and a failure to read the files, we go straight
-       # to diff if one of the inputs is empty.
-       if test -s "$1" && test -s "$2"
-       then
-               # regular case: both files non-empty
-               mingw_read_file_strip_cr_ test_cmp_a <"$1"
-               mingw_read_file_strip_cr_ test_cmp_b <"$2"
-       elif test -s "$1" && test "$2" = -
-       then
-               # read 2nd file from stdin
-               mingw_read_file_strip_cr_ test_cmp_a <"$1"
-               mingw_read_file_strip_cr_ test_cmp_b
-               stdin_for_diff='<<<"$test_cmp_b"'
-       elif test "$1" = - && test -s "$2"
-       then
-               # read 1st file from stdin
-               mingw_read_file_strip_cr_ test_cmp_a
-               mingw_read_file_strip_cr_ test_cmp_b <"$2"
-               stdin_for_diff='<<<"$test_cmp_a"'
-       fi
-       test -n "$test_cmp_a" &&
-       test -n "$test_cmp_b" &&
-       test "$test_cmp_a" = "$test_cmp_b" ||
-       eval "diff -u \"\$@\" $stdin_for_diff"
-}
-
-# $1 is the name of the shell variable to fill in
-mingw_read_file_strip_cr_ () {
-       # Read line-wise using LF as the line separator
-       # and use IFS to strip CR.
-       local line
-       while :
-       do
-               if IFS=$'\r' read -r -d $'\n' line
-               then
-                       # good
-                       line=$line$'\n'
-               else
-                       # we get here at EOF, but also if the last line
-                       # was not terminated by LF; in the latter case,
-                       # some text was read
-                       if test -z "$line"
-                       then
-                               # EOF, really
-                               break
-                       fi
-               fi
-               eval "$1=\$$1\$line"
-       done
-}
-
 # Like "env FOO=BAR some-program", but run inside a subshell, which means
 # it also works for shell functions (though those functions cannot impact
 # the environment outside of the test_env invocation).
@@ -1682,7 +1616,7 @@ test_oid () {
        then
                BUG "undefined key '$1'"
        fi &&
-       eval "printf '%s' \"\${$var}\""
+       eval "printf '%s\n' \"\${$var}\""
 }
 
 # Insert a slash into an object ID so it can be used to reference a location
@@ -1751,6 +1685,13 @@ test_path_is_hidden () {
        return 1
 }
 
+# Poor man's URI escaping. Good enough for the test suite whose trash
+# directory has a space in it. See 93c3fcbe4d4 (git-svn: attempt to
+# mimic SVN 1.7 URL canonicalization, 2012-07-28) for prior art.
+test_uri_escape() {
+       sed 's/ /%20/g'
+}
+
 # Check that the given command was invoked as part of the
 # trace2-format trace on stdin.
 #
index 6db377f68b82c41b297538855391d335275ff6da..4fab1c1984c5146783840cf683bf93be51d7cdd8 100644 (file)
@@ -1721,7 +1721,7 @@ case $uname_s in
        test_set_prereq SED_STRIPS_CR
        test_set_prereq GREP_STRIPS_CR
        test_set_prereq WINDOWS
-       GIT_TEST_CMP=mingw_test_cmp
+       GIT_TEST_CMP="GIT_DIR=/dev/null git diff --no-index --ignore-cr-at-eol --"
        ;;
 *CYGWIN*)
        test_set_prereq POSIXPERM
index e95267a4ab54dc12318435d157d1fdd37821f567..3ea7c2bb5ad8c462ae0f54c9466af210bc27d0c2 100644 (file)
@@ -1267,9 +1267,22 @@ static struct ref *get_refs_list_using_list(struct transport *transport,
        return ret;
 }
 
+static int get_bundle_uri(struct transport *transport)
+{
+       get_helper(transport);
+
+       if (process_connect(transport, 0)) {
+               do_take_over(transport);
+               return transport->vtable->get_bundle_uri(transport);
+       }
+
+       return -1;
+}
+
 static struct transport_vtable vtable = {
        .set_option     = set_helper_option,
        .get_refs_list  = get_refs_list,
+       .get_bundle_uri = get_bundle_uri,
        .fetch_refs     = fetch_refs,
        .push_refs      = push_refs,
        .connect        = connect_helper,
index c4ca0b733acbdbe7fc7a5906490e396240ec7b06..90ea749e5cfbf7de994a8747186ad5fb3ae7c653 100644 (file)
@@ -26,6 +26,13 @@ struct transport_vtable {
        struct ref *(*get_refs_list)(struct transport *transport, int for_push,
                                     struct transport_ls_refs_options *transport_options);
 
+       /**
+        * Populates the remote side's bundle-uri under protocol v2,
+        * if the "bundle-uri" capability was advertised. Returns 0 if
+        * OK, negative values on error.
+        */
+       int (*get_bundle_uri)(struct transport *transport);
+
        /**
         * Fetch the objects for the given refs. Note that this gets
         * an array, and should ignore the list structure.
index e7b97194c103a3a7fad6d4174e40c96db9ff2a4c..77a61a9d7bb06a5cfc7992c6fc4f9f3aa8a8ada5 100644 (file)
@@ -22,6 +22,7 @@
 #include "protocol.h"
 #include "object-store.h"
 #include "color.h"
+#include "bundle-uri.h"
 
 static int transport_use_color = -1;
 static char transport_colors[][COLOR_MAXLEN] = {
@@ -197,7 +198,7 @@ struct git_transport_data {
        struct git_transport_options options;
        struct child_process *conn;
        int fd[2];
-       unsigned got_remote_heads : 1;
+       unsigned finished_handshake : 1;
        enum protocol_version version;
        struct oid_array extra_have;
        struct oid_array shallow;
@@ -344,7 +345,7 @@ static struct ref *handshake(struct transport *transport, int for_push,
        case protocol_unknown_version:
                BUG("unknown protocol version");
        }
-       data->got_remote_heads = 1;
+       data->finished_handshake = 1;
        transport->hash_algo = reader.hash_algo;
 
        if (reader.line_peeked)
@@ -359,6 +360,39 @@ static struct ref *get_refs_via_connect(struct transport *transport, int for_pus
        return handshake(transport, for_push, options, 1);
 }
 
+static int get_bundle_uri(struct transport *transport)
+{
+       struct git_transport_data *data = transport->data;
+       struct packet_reader reader;
+       int stateless_rpc = transport->stateless_rpc;
+
+       if (!transport->bundles) {
+               CALLOC_ARRAY(transport->bundles, 1);
+               init_bundle_list(transport->bundles);
+       }
+
+       if (!data->finished_handshake) {
+               struct ref *refs = handshake(transport, 0, NULL, 0);
+
+               if (refs)
+                       free_refs(refs);
+       }
+
+       /*
+        * "Support" protocol v0 and v2 without bundle-uri support by
+        * silently degrading to a NOOP.
+        */
+       if (!server_supports_v2("bundle-uri"))
+               return 0;
+
+       packet_reader_init(&reader, data->fd[0], NULL, 0,
+                          PACKET_READ_CHOMP_NEWLINE |
+                          PACKET_READ_GENTLE_ON_EOF);
+
+       return get_remote_bundle_uri(data->fd[1], &reader,
+                                    transport->bundles, stateless_rpc);
+}
+
 static int fetch_refs_via_pack(struct transport *transport,
                               int nr_heads, struct ref **to_fetch)
 {
@@ -394,7 +428,7 @@ static int fetch_refs_via_pack(struct transport *transport,
        args.negotiation_tips = data->options.negotiation_tips;
        args.reject_shallow_remote = transport->smart_options->reject_shallow;
 
-       if (!data->got_remote_heads) {
+       if (!data->finished_handshake) {
                int i;
                int must_list_refs = 0;
                for (i = 0; i < nr_heads; i++) {
@@ -434,7 +468,7 @@ static int fetch_refs_via_pack(struct transport *transport,
                          to_fetch, nr_heads, &data->shallow,
                          &transport->pack_lockfiles, data->version);
 
-       data->got_remote_heads = 0;
+       data->finished_handshake = 0;
        data->options.self_contained_and_connected =
                args.self_contained_and_connected;
        data->options.connectivity_checked = args.connectivity_checked;
@@ -819,7 +853,7 @@ static int git_transport_push(struct transport *transport, struct ref *remote_re
        if (transport_color_config() < 0)
                return -1;
 
-       if (!data->got_remote_heads)
+       if (!data->finished_handshake)
                get_refs_via_connect(transport, 1, NULL);
 
        memset(&args, 0, sizeof(args));
@@ -867,7 +901,7 @@ static int git_transport_push(struct transport *transport, struct ref *remote_re
        else
                ret = finish_connect(data->conn);
        data->conn = NULL;
-       data->got_remote_heads = 0;
+       data->finished_handshake = 0;
 
        return ret;
 }
@@ -887,7 +921,7 @@ static int disconnect_git(struct transport *transport)
 {
        struct git_transport_data *data = transport->data;
        if (data->conn) {
-               if (data->got_remote_heads && !transport->stateless_rpc)
+               if (data->finished_handshake && !transport->stateless_rpc)
                        packet_flush(data->fd[1]);
                close(data->fd[0]);
                if (data->fd[1] >= 0)
@@ -902,6 +936,7 @@ static int disconnect_git(struct transport *transport)
 
 static struct transport_vtable taken_over_vtable = {
        .get_refs_list  = get_refs_via_connect,
+       .get_bundle_uri = get_bundle_uri,
        .fetch_refs     = fetch_refs_via_pack,
        .push_refs      = git_transport_push,
        .disconnect     = disconnect_git
@@ -921,7 +956,7 @@ void transport_take_over(struct transport *transport,
        data->conn = child;
        data->fd[0] = data->conn->out;
        data->fd[1] = data->conn->in;
-       data->got_remote_heads = 0;
+       data->finished_handshake = 0;
        transport->data = data;
 
        transport->vtable = &taken_over_vtable;
@@ -1054,6 +1089,7 @@ static struct transport_vtable bundle_vtable = {
 
 static struct transport_vtable builtin_smart_vtable = {
        .get_refs_list  = get_refs_via_connect,
+       .get_bundle_uri = get_bundle_uri,
        .fetch_refs     = fetch_refs_via_pack,
        .push_refs      = git_transport_push,
        .connect        = connect_git,
@@ -1068,6 +1104,9 @@ struct transport *transport_get(struct remote *remote, const char *url)
        ret->progress = isatty(2);
        string_list_init_dup(&ret->pack_lockfiles);
 
+       CALLOC_ARRAY(ret->bundles, 1);
+       init_bundle_list(ret->bundles);
+
        if (!remote)
                BUG("No remote provided to transport_get()");
 
@@ -1118,7 +1157,7 @@ struct transport *transport_get(struct remote *remote, const char *url)
                ret->smart_options = &(data->options);
 
                data->conn = NULL;
-               data->got_remote_heads = 0;
+               data->finished_handshake = 0;
        } else {
                /* Unknown protocol in URL. Pass to external handler. */
                int len = external_specification_len(url);
@@ -1482,6 +1521,34 @@ int transport_fetch_refs(struct transport *transport, struct ref *refs)
        return rc;
 }
 
+int transport_get_remote_bundle_uri(struct transport *transport)
+{
+       int value = 0;
+       const struct transport_vtable *vtable = transport->vtable;
+
+       /* Check config only once. */
+       if (transport->got_remote_bundle_uri)
+               return 0;
+       transport->got_remote_bundle_uri = 1;
+
+       /*
+        * Don't request bundle-uri from the server unless configured to
+        * do so by the transfer.bundleURI=true config option.
+        */
+       if (git_config_get_bool("transfer.bundleuri", &value) || !value)
+               return 0;
+
+       if (!transport->bundles->baseURI)
+               transport->bundles->baseURI = xstrdup(transport->url);
+
+       if (!vtable->get_bundle_uri)
+               return error(_("bundle-uri operation not supported by protocol"));
+
+       if (vtable->get_bundle_uri(transport) < 0)
+               return error(_("could not retrieve server-advertised bundle-uri list"));
+       return 0;
+}
+
 void transport_unlock_pack(struct transport *transport, unsigned int flags)
 {
        int in_signal_handler = !!(flags & TRANSPORT_UNLOCK_PACK_IN_SIGNAL_HANDLER);
@@ -1512,6 +1579,8 @@ int transport_disconnect(struct transport *transport)
                ret = transport->vtable->disconnect(transport);
        if (transport->got_remote_refs)
                free_refs((void *)transport->remote_refs);
+       clear_bundle_list(transport->bundles);
+       free(transport->bundles);
        free(transport);
        return ret;
 }
index b5bf7b3e70418a9ded196bf124b1e445bc8076cb..85150f504fbfb9501109d297223f566cc36fe0e5 100644 (file)
@@ -62,6 +62,7 @@ enum transport_family {
        TRANSPORT_FAMILY_IPV6
 };
 
+struct bundle_list;
 struct transport {
        const struct transport_vtable *vtable;
 
@@ -76,6 +77,18 @@ struct transport {
         */
        unsigned got_remote_refs : 1;
 
+       /**
+        * Indicates whether we already called get_bundle_uri_list(); set by
+        * transport.c::transport_get_remote_bundle_uri().
+        */
+       unsigned got_remote_bundle_uri : 1;
+
+       /*
+        * The results of "command=bundle-uri", if both sides support
+        * the "bundle-uri" capability.
+        */
+       struct bundle_list *bundles;
+
        /*
         * Transports that call take-over destroys the data specific to
         * the transport type while doing so, and cannot be reused.
@@ -281,6 +294,12 @@ void transport_ls_refs_options_release(struct transport_ls_refs_options *opts);
 const struct ref *transport_get_remote_refs(struct transport *transport,
                                            struct transport_ls_refs_options *transport_options);
 
+/**
+ * Retrieve bundle URI(s) from a remote. Populates "struct
+ * transport"'s "bundle_uri" and "got_remote_bundle_uri".
+ */
+int transport_get_remote_bundle_uri(struct transport *transport);
+
 /*
  * Fetch the hash algorithm used by a remote.
  *
index 8a762aa0772eb4f7e107da6807092fe0a2be1a61..ea09023b0158e6945e8750f435ce7addd2b2bda2 100644 (file)
@@ -19,7 +19,6 @@
 #include "promisor-remote.h"
 #include "entry.h"
 #include "parallel-checkout.h"
-#include "sparse-index.h"
 
 /*
  * Error messages expected by scripts out of plumbing commands such as
@@ -71,7 +70,7 @@ static const char *unpack_plumbing_errors[NB_UNPACK_TREES_WARNING_TYPES] = {
          ? ((o)->msgs[(type)])      \
          : (unpack_plumbing_errors[(type)]) )
 
-static const char *super_prefixed(const char *path)
+static const char *super_prefixed(const char *path, const char *super_prefix)
 {
        /*
         * It is necessary and sufficient to have two static buffers
@@ -83,7 +82,6 @@ static const char *super_prefixed(const char *path)
        static unsigned idx = ARRAY_SIZE(buf) - 1;
 
        if (super_prefix_len < 0) {
-               const char *super_prefix = get_super_prefix();
                if (!super_prefix) {
                        super_prefix_len = 0;
                } else {
@@ -236,7 +234,8 @@ static int add_rejected_path(struct unpack_trees_options *o,
                return -1;
 
        if (!o->show_all_errors)
-               return error(ERRORMSG(o, e), super_prefixed(path));
+               return error(ERRORMSG(o, e), super_prefixed(path,
+                                                           o->super_prefix));
 
        /*
         * Otherwise, insert in a list for future display by
@@ -263,7 +262,8 @@ static void display_error_msgs(struct unpack_trees_options *o)
                        error_displayed = 1;
                        for (i = 0; i < rejects->nr; i++)
                                strbuf_addf(&path, "\t%s\n", rejects->items[i].string);
-                       error(ERRORMSG(o, e), super_prefixed(path.buf));
+                       error(ERRORMSG(o, e), super_prefixed(path.buf,
+                                                            o->super_prefix));
                        strbuf_release(&path);
                }
                string_list_clear(rejects, 0);
@@ -290,7 +290,8 @@ static void display_warning_msgs(struct unpack_trees_options *o)
                        warning_displayed = 1;
                        for (i = 0; i < rejects->nr; i++)
                                strbuf_addf(&path, "\t%s\n", rejects->items[i].string);
-                       warning(ERRORMSG(o, e), super_prefixed(path.buf));
+                       warning(ERRORMSG(o, e), super_prefixed(path.buf,
+                                                              o->super_prefix));
                        strbuf_release(&path);
                }
                string_list_clear(rejects, 0);
@@ -312,7 +313,8 @@ static int check_submodule_move_head(const struct cache_entry *ce,
        if (o->reset)
                flags |= SUBMODULE_MOVE_HEAD_FORCE;
 
-       if (submodule_move_head(ce->name, old_id, new_id, flags))
+       if (submodule_move_head(ce->name, o->super_prefix, old_id, new_id,
+                               flags))
                return add_rejected_path(o, ERROR_WOULD_LOSE_SUBMODULE, ce->name);
        return 0;
 }
@@ -415,6 +417,7 @@ static int check_updates(struct unpack_trees_options *o,
        int i, pc_workers, pc_threshold;
 
        trace_performance_enter();
+       state.super_prefix = o->super_prefix;
        state.force = 1;
        state.quiet = 1;
        state.refresh_cache = 1;
@@ -445,7 +448,7 @@ static int check_updates(struct unpack_trees_options *o,
 
                if (ce->ce_flags & CE_WT_REMOVE) {
                        display_progress(progress, ++cnt);
-                       unlink_entry(ce);
+                       unlink_entry(ce, o->super_prefix);
                }
        }
 
@@ -2959,8 +2962,8 @@ int bind_merge(const struct cache_entry * const *src,
        if (a && old)
                return o->quiet ? -1 :
                        error(ERRORMSG(o, ERROR_BIND_OVERLAP),
-                             super_prefixed(a->name),
-                             super_prefixed(old->name));
+                             super_prefixed(a->name, o->super_prefix),
+                             super_prefixed(old->name, o->super_prefix));
        if (!a)
                return keep_entry(old, o);
        else
@@ -3021,7 +3024,7 @@ int stash_worktree_untracked_merge(const struct cache_entry * const *src,
 
        if (worktree && untracked)
                return error(_("worktree and untracked commit have duplicate entries: %s"),
-                            super_prefixed(worktree->name));
+                            super_prefixed(worktree->name, o->super_prefix));
 
        return merged_entry(worktree ? worktree : untracked, NULL, o);
 }
index 6ab0d74c84dc5ca3004e8b3b160dde98d1805e5d..3a7b3e5f0077938f1e28de5f41a691191c098744 100644 (file)
@@ -75,6 +75,7 @@ struct unpack_trees_options {
                     skip_cache_tree_update;
        enum unpack_trees_reset_type reset;
        const char *prefix;
+       const char *super_prefix;
        int cache_bottom;
        struct pathspec *pathspec;
        merge_fn_t fn;
index 151d9a52784b1f4a149913d38739c91e389d95d6..e25356a06124dae49a486884ccdd2872cb429a29 100644 (file)
@@ -315,7 +315,8 @@ struct find_by_namelen_data {
 };
 
 static int userdiff_find_by_namelen_cb(struct userdiff_driver *driver,
-                                      enum userdiff_driver_type type, void *priv)
+                                      enum userdiff_driver_type type UNUSED,
+                                      void *priv)
 {
        struct find_by_namelen_data *cb_data = priv;
 
diff --git a/ws.c b/ws.c
index 6e69877f25791632d98bf7b109a2eaebd04c96af..46a77bcad6670eacb793c9e07334f61a4b93f242 100644 (file)
--- a/ws.c
+++ b/ws.c
@@ -252,7 +252,7 @@ unsigned ws_check(const char *line, int len, unsigned ws_rule)
        return ws_check_emit_1(line, len, ws_rule, NULL, NULL, NULL, NULL);
 }
 
-int ws_blank_line(const char *line, int len, unsigned ws_rule)
+int ws_blank_line(const char *line, int len)
 {
        /*
         * We _might_ want to treat CR differently from other
index 5813174896cc9ae5fa9287c8bcc432dc8cf2ef47..3162241a570cb60393607f06a8a12963b71afb78 100644 (file)
 #include "worktree.h"
 #include "lockfile.h"
 #include "sequencer.h"
+#include "fsmonitor-settings.h"
 
 #define AB_DELAY_WARNING_IN_MS (2 * 1000)
+#define UF_DELAY_WARNING_IN_MS (2 * 1000)
 
 static const char cut_line[] =
 "------------------------ >8 ------------------------\n";
@@ -438,7 +440,7 @@ static char short_submodule_status(struct wt_status_change_data *d)
 }
 
 static void wt_status_collect_changed_cb(struct diff_queue_struct *q,
-                                        struct diff_options *options,
+                                        struct diff_options *options UNUSED,
                                         void *data)
 {
        struct wt_status *s = data;
@@ -525,7 +527,7 @@ static int unmerged_mask(struct index_state *istate, const char *path)
 }
 
 static void wt_status_collect_updated_cb(struct diff_queue_struct *q,
-                                        struct diff_options *options,
+                                        struct diff_options *options UNUSED,
                                         void *data)
 {
        struct wt_status *s = data;
@@ -1205,6 +1207,13 @@ static void wt_longstatus_print_tracking(struct wt_status *s)
        strbuf_release(&sb);
 }
 
+static int uf_was_slow(struct wt_status *s)
+{
+       if (getenv("GIT_TEST_UF_DELAY_WARNING"))
+               s->untracked_in_ms = 3250;
+       return UF_DELAY_WARNING_IN_MS < s->untracked_in_ms;
+}
+
 static void show_merge_in_progress(struct wt_status *s,
                                   const char *color)
 {
@@ -1814,6 +1823,7 @@ static void wt_longstatus_print(struct wt_status *s)
 {
        const char *branch_color = color(WT_STATUS_ONBRANCH, s);
        const char *branch_status_color = color(WT_STATUS_HEADER, s);
+       enum fsmonitor_mode fsm_mode = fsm_settings__get_mode(s->repo);
 
        if (s->branch) {
                const char *on_what = _("On branch ");
@@ -1870,13 +1880,21 @@ static void wt_longstatus_print(struct wt_status *s)
                wt_longstatus_print_other(s, &s->untracked, _("Untracked files"), "add");
                if (s->show_ignored_mode)
                        wt_longstatus_print_other(s, &s->ignored, _("Ignored files"), "add -f");
-               if (advice_enabled(ADVICE_STATUS_U_OPTION) && 2000 < s->untracked_in_ms) {
+               if (advice_enabled(ADVICE_STATUS_U_OPTION) && uf_was_slow(s)) {
                        status_printf_ln(s, GIT_COLOR_NORMAL, "%s", "");
+                       if (fsm_mode > FSMONITOR_MODE_DISABLED) {
+                               status_printf_ln(s, GIT_COLOR_NORMAL,
+                                               _("It took %.2f seconds to enumerate untracked files,\n"
+                                               "but the results were cached, and subsequent runs may be faster."),
+                                               s->untracked_in_ms / 1000.0);
+                       } else {
+                               status_printf_ln(s, GIT_COLOR_NORMAL,
+                                               _("It took %.2f seconds to enumerate untracked files."),
+                                               s->untracked_in_ms / 1000.0);
+                       }
                        status_printf_ln(s, GIT_COLOR_NORMAL,
-                                        _("It took %.2f seconds to enumerate untracked files. 'status -uno'\n"
-                                          "may speed it up, but you have to be careful not to forget to add\n"
-                                          "new files yourself (see 'git help status')."),
-                                        s->untracked_in_ms / 1000.0);
+                                       _("See 'git help status' for information on how to improve this."));
+                       status_printf_ln(s, GIT_COLOR_NORMAL, "%s", "");
                }
        } else if (s->committable)
                status_printf_ln(s, GIT_COLOR_NORMAL, _("Untracked files not listed%s"),
index 32652ded2d7c5cad87f120a13c3c6c84d6def61f..344c2dfc3eb443a721cc45b6ba3be56e9f186c97 100644 (file)
@@ -973,7 +973,7 @@ void xdl_free_script(xdchange_t *xscr) {
        }
 }
 
-static int xdl_call_hunk_func(xdfenv_t *xe, xdchange_t *xscr, xdemitcb_t *ecb,
+static int xdl_call_hunk_func(xdfenv_t *xe UNUSED, xdchange_t *xscr, xdemitcb_t *ecb,
                              xdemitconf_t const *xecfg)
 {
        xdchange_t *xch, *xche;
index c4ccd68d4760bc08735d6ca3b7dcdd3440abaf16..75f0fe498661c43078fe19f8b5d10926700763de 100644 (file)
@@ -95,7 +95,7 @@ xdchange_t *xdl_get_hunk(xdchange_t **xscr, xdemitconf_t const *xecfg)
 }
 
 
-static long def_ff(const char *rec, long len, char *buf, long sz, void *priv)
+static long def_ff(const char *rec, long len, char *buf, long sz)
 {
        if (len > 0 &&
                        (isalpha((unsigned char)*rec) || /* identifier? */
@@ -117,7 +117,7 @@ static long match_func_rec(xdfile_t *xdf, xdemitconf_t const *xecfg, long ri,
        const char *rec;
        long len = xdl_get_rec(xdf, ri, &rec);
        if (!xecfg->find_func)
-               return def_ff(rec, len, buf, sz, xecfg->find_func_priv);
+               return def_ff(rec, len, buf, sz);
        return xecfg->find_func(rec, len, buf, sz, xecfg->find_func_priv);
 }