2005-06-21 18:47:37 +02:00
|
|
|
#!/bin/sh
|
2005-07-08 19:57:21 +02:00
|
|
|
|
2006-01-12 23:04:36 +01:00
|
|
|
USAGE='[-f] [-b <new_branch>] [-m] [<branch>] [<paths>...]'
|
2005-12-23 07:37:50 +01:00
|
|
|
SUBDIRECTORY_OK=Sometimes
|
2005-12-13 23:30:31 +01:00
|
|
|
. git-sh-setup
|
2005-10-29 23:46:41 +02:00
|
|
|
|
2005-06-21 18:47:37 +02:00
|
|
|
old=$(git-rev-parse HEAD)
|
2006-05-19 11:17:16 +02:00
|
|
|
old_name=HEAD
|
2005-06-21 20:03:11 +02:00
|
|
|
new=
|
2006-05-19 11:17:16 +02:00
|
|
|
new_name=
|
2005-06-21 18:59:26 +02:00
|
|
|
force=
|
2005-06-21 20:03:11 +02:00
|
|
|
branch=
|
2005-07-12 05:44:20 +02:00
|
|
|
newbranch=
|
2006-05-19 11:17:16 +02:00
|
|
|
newbranch_log=
|
2006-01-12 23:04:36 +01:00
|
|
|
merge=
|
2005-06-21 20:03:11 +02:00
|
|
|
while [ "$#" != "0" ]; do
|
|
|
|
arg="$1"
|
|
|
|
shift
|
|
|
|
case "$arg" in
|
2005-07-12 05:44:20 +02:00
|
|
|
"-b")
|
|
|
|
newbranch="$1"
|
|
|
|
shift
|
|
|
|
[ -z "$newbranch" ] &&
|
|
|
|
die "git checkout: -b needs a branch name"
|
Enable the packed refs file format
This actually "turns on" the packed ref file format, now that the
infrastructure to do so sanely exists (ie notably the change to make the
reference reading logic take refnames rather than pathnames to the loose
objects that no longer necessarily even exist).
In particular, when the ref lookup hits a refname that has no loose file
associated with it, it falls back on the packed-ref information. Also, the
ref-locking code, while still using a loose file for the locking itself
(and _creating_ a loose file for the new ref) no longer requires that the
old ref be in such an unpacked state.
Finally, this does a minimal hack to git-checkout.sh to rather than check
the ref-file directly, do a "git-rev-parse" on the "heads/$refname".
That's not really wonderful - we should rather really have a special
routine to verify the names as proper branch head names, but it is a
workable solution for now.
With this, I can literally do something like
git pack-refs
find .git/refs -type f -print0 | xargs -0 rm -f --
and the end result is a largely working repository (ie I've done two
commits - which creates _one_ unpacked ref file - done things like run
"gitk" and "git log" etc, and it all looks ok).
There are probably things missing, but I'm hoping that the missing things
are now of the "small and obvious" kind, and that somebody else might want
to start looking at this too. Hint hint ;)
Signed-off-by: Linus Torvalds <torvalds@osdl.org>
Signed-off-by: Junio C Hamano <junkio@cox.net>
2006-09-14 19:14:47 +02:00
|
|
|
git-rev-parse --symbolic "heads/$newbranch" >&/dev/null &&
|
2005-07-12 05:44:20 +02:00
|
|
|
die "git checkout: branch $newbranch already exists"
|
2005-10-14 03:57:39 +02:00
|
|
|
git-check-ref-format "heads/$newbranch" ||
|
2006-02-15 20:22:11 +01:00
|
|
|
die "git checkout: we do not like '$newbranch' as a branch name."
|
2005-07-12 05:44:20 +02:00
|
|
|
;;
|
2006-05-19 11:17:16 +02:00
|
|
|
"-l")
|
|
|
|
newbranch_log=1
|
|
|
|
;;
|
2005-06-21 18:47:37 +02:00
|
|
|
"-f")
|
2005-06-21 20:03:11 +02:00
|
|
|
force=1
|
2005-06-21 18:47:37 +02:00
|
|
|
;;
|
2006-01-12 23:04:36 +01:00
|
|
|
-m)
|
|
|
|
merge=1
|
|
|
|
;;
|
2005-10-18 10:29:27 +02:00
|
|
|
--)
|
|
|
|
break
|
|
|
|
;;
|
2005-10-29 23:46:41 +02:00
|
|
|
-*)
|
|
|
|
usage
|
|
|
|
;;
|
2005-06-21 18:47:37 +02:00
|
|
|
*)
|
2005-10-18 10:29:27 +02:00
|
|
|
if rev=$(git-rev-parse --verify "$arg^0" 2>/dev/null)
|
|
|
|
then
|
|
|
|
if [ -z "$rev" ]; then
|
|
|
|
echo "unknown flag $arg"
|
|
|
|
exit 1
|
|
|
|
fi
|
|
|
|
new="$rev"
|
2006-05-19 11:17:16 +02:00
|
|
|
new_name="$arg^0"
|
Enable the packed refs file format
This actually "turns on" the packed ref file format, now that the
infrastructure to do so sanely exists (ie notably the change to make the
reference reading logic take refnames rather than pathnames to the loose
objects that no longer necessarily even exist).
In particular, when the ref lookup hits a refname that has no loose file
associated with it, it falls back on the packed-ref information. Also, the
ref-locking code, while still using a loose file for the locking itself
(and _creating_ a loose file for the new ref) no longer requires that the
old ref be in such an unpacked state.
Finally, this does a minimal hack to git-checkout.sh to rather than check
the ref-file directly, do a "git-rev-parse" on the "heads/$refname".
That's not really wonderful - we should rather really have a special
routine to verify the names as proper branch head names, but it is a
workable solution for now.
With this, I can literally do something like
git pack-refs
find .git/refs -type f -print0 | xargs -0 rm -f --
and the end result is a largely working repository (ie I've done two
commits - which creates _one_ unpacked ref file - done things like run
"gitk" and "git log" etc, and it all looks ok).
There are probably things missing, but I'm hoping that the missing things
are now of the "small and obvious" kind, and that somebody else might want
to start looking at this too. Hint hint ;)
Signed-off-by: Linus Torvalds <torvalds@osdl.org>
Signed-off-by: Junio C Hamano <junkio@cox.net>
2006-09-14 19:14:47 +02:00
|
|
|
if git-rev-parse "heads/$arg^0" >&/dev/null; then
|
2005-10-18 10:29:27 +02:00
|
|
|
branch="$arg"
|
|
|
|
fi
|
|
|
|
elif rev=$(git-rev-parse --verify "$arg^{tree}" 2>/dev/null)
|
|
|
|
then
|
|
|
|
# checking out selected paths from a tree-ish.
|
|
|
|
new="$rev"
|
2006-05-19 11:17:16 +02:00
|
|
|
new_name="$arg^{tree}"
|
2005-10-18 10:29:27 +02:00
|
|
|
branch=
|
|
|
|
else
|
|
|
|
new=
|
2006-05-19 11:17:16 +02:00
|
|
|
new_name=
|
2005-10-18 10:29:27 +02:00
|
|
|
branch=
|
|
|
|
set x "$arg" "$@"
|
|
|
|
shift
|
2005-06-21 20:03:11 +02:00
|
|
|
fi
|
2006-07-31 12:07:12 +02:00
|
|
|
case "$1" in
|
|
|
|
--)
|
|
|
|
shift ;;
|
|
|
|
esac
|
2005-10-18 10:29:27 +02:00
|
|
|
break
|
2005-06-21 20:03:11 +02:00
|
|
|
;;
|
2005-06-21 18:47:37 +02:00
|
|
|
esac
|
|
|
|
done
|
|
|
|
|
2005-10-18 10:29:27 +02:00
|
|
|
# The behaviour of the command with and without explicit path
|
|
|
|
# parameters is quite different.
|
|
|
|
#
|
|
|
|
# Without paths, we are checking out everything in the work tree,
|
|
|
|
# possibly switching branches. This is the traditional behaviour.
|
2005-07-12 05:44:20 +02:00
|
|
|
#
|
2005-10-18 10:29:27 +02:00
|
|
|
# With paths, we are _never_ switching branch, but checking out
|
|
|
|
# the named paths from either index (when no rev is given),
|
|
|
|
# or the named tree-ish (when rev is given).
|
|
|
|
|
|
|
|
if test "$#" -ge 1
|
|
|
|
then
|
2006-02-15 20:22:11 +01:00
|
|
|
hint=
|
|
|
|
if test "$#" -eq 1
|
|
|
|
then
|
|
|
|
hint="
|
|
|
|
Did you intend to checkout '$@' which can not be resolved as commit?"
|
|
|
|
fi
|
2006-01-12 23:04:36 +01:00
|
|
|
if test '' != "$newbranch$force$merge"
|
2005-10-18 10:29:27 +02:00
|
|
|
then
|
2006-02-15 20:22:11 +01:00
|
|
|
die "git checkout: updating paths is incompatible with switching branches/forcing$hint"
|
2005-10-18 10:29:27 +02:00
|
|
|
fi
|
|
|
|
if test '' != "$new"
|
|
|
|
then
|
|
|
|
# from a specific tree-ish; note that this is for
|
|
|
|
# rescuing paths and is never meant to remove what
|
|
|
|
# is not in the named tree-ish.
|
2005-12-23 23:10:16 +01:00
|
|
|
git-ls-tree --full-name -r "$new" "$@" |
|
2005-10-18 10:29:27 +02:00
|
|
|
git-update-index --index-info || exit $?
|
|
|
|
fi
|
|
|
|
git-checkout-index -f -u -- "$@"
|
|
|
|
exit $?
|
|
|
|
else
|
|
|
|
# Make sure we did not fall back on $arg^{tree} codepath
|
|
|
|
# since we are not checking out from an arbitrary tree-ish,
|
|
|
|
# but switching branches.
|
|
|
|
if test '' != "$new"
|
|
|
|
then
|
|
|
|
git-rev-parse --verify "$new^{commit}" >/dev/null 2>&1 ||
|
|
|
|
die "Cannot switch branch to a non-commit."
|
|
|
|
fi
|
|
|
|
fi
|
|
|
|
|
2005-12-23 07:37:50 +01:00
|
|
|
# We are switching branches and checking out trees, so
|
|
|
|
# we *NEED* to be at the toplevel.
|
|
|
|
cdup=$(git-rev-parse --show-cdup)
|
|
|
|
if test ! -z "$cdup"
|
|
|
|
then
|
|
|
|
cd "$cdup"
|
|
|
|
fi
|
|
|
|
|
2006-05-19 11:17:16 +02:00
|
|
|
[ -z "$new" ] && new=$old && new_name="$old_name"
|
2005-10-18 10:29:27 +02:00
|
|
|
|
2005-07-12 05:44:20 +02:00
|
|
|
# If we don't have an old branch that we're switching to,
|
|
|
|
# and we don't have a new branch name for the target we
|
|
|
|
# are switching to, then we'd better just be checking out
|
|
|
|
# what we already had
|
2005-10-18 10:29:27 +02:00
|
|
|
|
2005-07-12 05:44:20 +02:00
|
|
|
[ -z "$branch$newbranch" ] &&
|
|
|
|
[ "$new" != "$old" ] &&
|
2006-02-15 20:22:11 +01:00
|
|
|
die "git checkout: to checkout the requested commit you need to specify
|
|
|
|
a name for a new branch which is created and switched to"
|
2005-07-12 05:44:20 +02:00
|
|
|
|
2005-06-21 18:59:26 +02:00
|
|
|
if [ "$force" ]
|
2005-06-21 18:47:37 +02:00
|
|
|
then
|
2006-06-20 11:47:14 +02:00
|
|
|
git-read-tree --reset -u $new
|
2005-06-21 18:47:37 +02:00
|
|
|
else
|
2005-09-11 23:12:08 +02:00
|
|
|
git-update-index --refresh >/dev/null
|
2006-01-12 23:04:36 +01:00
|
|
|
merge_error=$(git-read-tree -m -u $old $new 2>&1) || (
|
|
|
|
case "$merge" in
|
|
|
|
'')
|
|
|
|
echo >&2 "$merge_error"
|
|
|
|
exit 1 ;;
|
2006-01-12 08:07:27 +01:00
|
|
|
esac
|
|
|
|
|
2006-01-12 23:04:36 +01:00
|
|
|
# Match the index to the working tree, and do a three-way.
|
|
|
|
git diff-files --name-only | git update-index --remove --stdin &&
|
2006-01-12 08:07:27 +01:00
|
|
|
work=`git write-tree` &&
|
2006-06-28 20:47:28 +02:00
|
|
|
git read-tree --reset -u $new &&
|
2006-05-10 04:23:23 +02:00
|
|
|
git read-tree -m -u --aggressive $old $new $work || exit
|
2006-01-12 23:04:36 +01:00
|
|
|
|
2006-01-12 08:07:27 +01:00
|
|
|
if result=`git write-tree 2>/dev/null`
|
|
|
|
then
|
2006-01-12 23:04:36 +01:00
|
|
|
echo >&2 "Trivially automerged."
|
|
|
|
else
|
|
|
|
git merge-index -o git-merge-one-file -a
|
2006-01-12 08:07:27 +01:00
|
|
|
fi
|
2006-01-12 23:04:36 +01:00
|
|
|
|
|
|
|
# Do not register the cleanly merged paths in the index yet.
|
|
|
|
# this is not a real merge before committing, but just carrying
|
|
|
|
# the working tree changes along.
|
|
|
|
unmerged=`git ls-files -u`
|
|
|
|
git read-tree --reset $new
|
|
|
|
case "$unmerged" in
|
|
|
|
'') ;;
|
|
|
|
*)
|
|
|
|
(
|
|
|
|
z40=0000000000000000000000000000000000000000
|
|
|
|
echo "$unmerged" |
|
|
|
|
sed -e 's/^[0-7]* [0-9a-f]* /'"0 $z40 /"
|
|
|
|
echo "$unmerged"
|
|
|
|
) | git update-index --index-info
|
|
|
|
;;
|
|
|
|
esac
|
|
|
|
exit 0
|
2006-01-12 08:07:27 +01:00
|
|
|
)
|
2006-01-14 21:31:18 +01:00
|
|
|
saved_err=$?
|
2006-02-15 01:05:57 +01:00
|
|
|
if test "$saved_err" = 0
|
|
|
|
then
|
|
|
|
test "$new" = "$old" || git diff-index --name-status "$new"
|
|
|
|
fi
|
2006-01-14 21:31:18 +01:00
|
|
|
(exit $saved_err)
|
2005-06-22 00:40:00 +02:00
|
|
|
fi
|
|
|
|
|
|
|
|
#
|
2005-12-17 08:12:33 +01:00
|
|
|
# Switch the HEAD pointer to the new branch if we
|
2005-06-22 00:40:00 +02:00
|
|
|
# checked out a branch head, and remove any potential
|
|
|
|
# old MERGE_HEAD's (subsequent commits will clearly not
|
|
|
|
# be based on them, since we re-set the index)
|
|
|
|
#
|
|
|
|
if [ "$?" -eq 0 ]; then
|
2005-07-12 05:44:20 +02:00
|
|
|
if [ "$newbranch" ]; then
|
2006-05-19 11:17:16 +02:00
|
|
|
if [ "$newbranch_log" ]; then
|
2006-05-25 05:34:04 +02:00
|
|
|
mkdir -p $(dirname "$GIT_DIR/logs/refs/heads/$newbranch")
|
|
|
|
touch "$GIT_DIR/logs/refs/heads/$newbranch"
|
2006-05-19 11:17:16 +02:00
|
|
|
fi
|
|
|
|
git-update-ref -m "checkout: Created from $new_name" "refs/heads/$newbranch" $new || exit
|
2005-07-12 05:44:20 +02:00
|
|
|
branch="$newbranch"
|
|
|
|
fi
|
2005-09-30 23:26:57 +02:00
|
|
|
[ "$branch" ] &&
|
|
|
|
GIT_DIR="$GIT_DIR" git-symbolic-ref HEAD "refs/heads/$branch"
|
2005-06-22 00:40:00 +02:00
|
|
|
rm -f "$GIT_DIR/MERGE_HEAD"
|
2005-08-23 23:03:14 +02:00
|
|
|
else
|
|
|
|
exit 1
|
2005-06-22 00:40:00 +02:00
|
|
|
fi
|