You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
158 lines
3.5 KiB
158 lines
3.5 KiB
#!/bin/sh |
|
. git-sh-setup-script || dir "Not a git archive" |
|
|
|
usage() { |
|
echo >&2 'usage: git bisect [start | bad | good | next | reset] |
|
git bisect start reset bisect state and start bisection. |
|
git bisect bad [<rev>] mark <rev> a known-bad revision. |
|
git bisect good [<rev>...] mark <rev>... known-good revisions. |
|
git bisect next find next bisection to test and check it out. |
|
git bisect reset [<branch>] finish bisection search and go back to branch.' |
|
exit 1 |
|
} |
|
|
|
bisect_autostart() { |
|
test -d "$GIT_DIR/refs/bisect" || { |
|
echo >&2 'You need to start by "git bisect start"' |
|
if test -t 0 |
|
then |
|
echo >&2 -n 'Do you want me to do it for you [Y/n]? ' |
|
read yesno |
|
case "$yesno" in |
|
[Nn]*) |
|
exit ;; |
|
esac |
|
bisect_start |
|
else |
|
exit 1 |
|
fi |
|
} |
|
} |
|
|
|
bisect_start() { |
|
case "$#" in 0) ;; *) usage ;; esac |
|
# |
|
# Verify HEAD. If we were bisecting before this, reset to the |
|
# top-of-line master first! |
|
# |
|
head=$(readlink $GIT_DIR/HEAD) || die "Bad HEAD - I need a symlink" |
|
case "$head" in |
|
refs/heads/bisect*) |
|
git checkout master || exit |
|
;; |
|
refs/heads/*) |
|
;; |
|
*) |
|
die "Bad HEAD - strange symlink" |
|
;; |
|
esac |
|
|
|
# |
|
# Get rid of any old bisect state |
|
# |
|
rm -f "$GIT_DIR/refs/heads/bisect" |
|
rm -rf "$GIT_DIR/refs/bisect/" |
|
mkdir "$GIT_DIR/refs/bisect" |
|
} |
|
|
|
bisect_bad() { |
|
bisect_autostart |
|
case "$#" in 0 | 1) ;; *) usage ;; esac |
|
rev=$(git-rev-parse --verify --default HEAD "$@") || exit |
|
echo "$rev" > "$GIT_DIR/refs/bisect/bad" |
|
bisect_auto_next |
|
} |
|
|
|
bisect_good() { |
|
bisect_autostart |
|
case "$#" in |
|
0) revs=$(git-rev-parse --verify HEAD) || exit ;; |
|
*) revs=$(git-rev-parse --revs-only --no-flags "$@") || exit ;; |
|
esac |
|
for rev in $revs |
|
do |
|
echo "$rev" >"$GIT_DIR/refs/bisect/good-$rev" |
|
done |
|
bisect_auto_next |
|
} |
|
|
|
bisect_next_check() { |
|
next_ok=no |
|
test -f "$GIT_DIR/refs/bisect/bad" && |
|
case "$(cd "$GIT_DIR" && echo refs/bisect/good-*)" in |
|
refs/bisect/good-\*) ;; |
|
*) next_ok=yes ;; |
|
esac |
|
case "$next_ok,$1" in |
|
no,) false ;; |
|
no,fail) |
|
echo >&2 'You need to give me at least one good and one bad revisions.' |
|
exit 1 ;; |
|
*) |
|
true ;; |
|
esac |
|
} |
|
|
|
bisect_auto_next() { |
|
bisect_next_check && bisect_next |
|
} |
|
|
|
bisect_next() { |
|
case "$#" in 0) ;; *) usage ;; esac |
|
bisect_autostart |
|
bisect_next_check fail |
|
bad=$(git-rev-parse --verify refs/bisect/bad) && |
|
good=$(git-rev-parse --sq --revs-only --not \ |
|
$(cd "$GIT_DIR" && ls refs/bisect/good-*)) && |
|
rev=$(eval "git-rev-list --bisect $good $bad") || exit |
|
nr=$(eval "git-rev-list $rev $good" | wc -l) || exit |
|
if [ "$nr" -le "1" ]; then |
|
echo "$rev is first bad commit" |
|
git-diff-tree --pretty $rev |
|
exit 0 |
|
fi |
|
echo "Bisecting: $nr revisions left to test after this" |
|
echo "$rev" > "$GIT_DIR/refs/heads/new-bisect" |
|
git checkout new-bisect || exit |
|
mv "$GIT_DIR/refs/heads/new-bisect" "$GIT_DIR/refs/heads/bisect" && |
|
ln -sf refs/heads/bisect "$GIT_DIR/HEAD" |
|
} |
|
|
|
bisect_reset() { |
|
case "$#" in |
|
0) branch=master ;; |
|
1) test -f "$GIT_DIR/refs/heads/$1" || { |
|
echo >&2 "$1 does not seem to be a valid branch" |
|
exit 1 |
|
} |
|
branch="$1" ;; |
|
*) |
|
usage ;; |
|
esac |
|
git checkout "$branch" && |
|
rm -fr "$GIT_DIR/refs/bisect" |
|
rm -f "$GIT_DIR/refs/reads/bisect" |
|
} |
|
|
|
case "$#" in |
|
0) |
|
usage ;; |
|
*) |
|
cmd="$1" |
|
shift |
|
case "$cmd" in |
|
start) |
|
bisect_start "$@" ;; |
|
bad) |
|
bisect_bad "$@" ;; |
|
good) |
|
bisect_good "$@" ;; |
|
next) |
|
# Not sure we want "next" at the UI level anymore. |
|
bisect_next "$@" ;; |
|
reset) |
|
bisect_reset "$@" ;; |
|
*) |
|
usage ;; |
|
esac |
|
esac
|
|
|