Skip to content

Commit

Permalink
bisect: introduce support for --no-checkout option.
Browse files Browse the repository at this point in the history
If --no-checkout is specified, then the bisection process uses:

	git update-ref --no-deref HEAD <trial>

at each trial instead of:

	git checkout <trial>

Improved-by: Christian Couder <[email protected]>
Signed-off-by: Jon Seymour <[email protected]>
Signed-off-by: Junio C Hamano <[email protected]>
  • Loading branch information
jonseymour authored and gitster committed Aug 4, 2011
1 parent d3dfeed commit fee92fc
Show file tree
Hide file tree
Showing 3 changed files with 28 additions and 14 deletions.
33 changes: 22 additions & 11 deletions bisect.c
Original file line number Diff line number Diff line change
Expand Up @@ -24,6 +24,7 @@ struct argv_array {

static const char *argv_checkout[] = {"checkout", "-q", NULL, "--", NULL};
static const char *argv_show_branch[] = {"show-branch", NULL, NULL};
static const char *argv_update_ref[] = {"update-ref", "--no-deref", "BISECT_HEAD", NULL, NULL};

/* bits #0-15 in revision.h */

Expand Down Expand Up @@ -707,16 +708,23 @@ static void mark_expected_rev(char *bisect_rev_hex)
die("closing file %s: %s", filename, strerror(errno));
}

static int bisect_checkout(char *bisect_rev_hex)
static int bisect_checkout(char *bisect_rev_hex, int no_checkout)
{
int res;

mark_expected_rev(bisect_rev_hex);

argv_checkout[2] = bisect_rev_hex;
res = run_command_v_opt(argv_checkout, RUN_GIT_CMD);
if (res)
exit(res);
if (no_checkout) {
argv_update_ref[3] = bisect_rev_hex;
if (run_command_v_opt(argv_update_ref, RUN_GIT_CMD))
die("update-ref --no-deref HEAD failed on %s",
bisect_rev_hex);
} else {
res = run_command_v_opt(argv_checkout, RUN_GIT_CMD);
if (res)
exit(res);
}

argv_show_branch[1] = bisect_rev_hex;
return run_command_v_opt(argv_show_branch, RUN_GIT_CMD);
Expand Down Expand Up @@ -788,7 +796,7 @@ static void handle_skipped_merge_base(const unsigned char *mb)
* - If one is "skipped", we can't know but we should warn.
* - If we don't know, we should check it out and ask the user to test.
*/
static void check_merge_bases(void)
static void check_merge_bases(int no_checkout)
{
struct commit_list *result;
int rev_nr;
Expand All @@ -806,7 +814,7 @@ static void check_merge_bases(void)
handle_skipped_merge_base(mb);
} else {
printf("Bisecting: a merge base must be tested\n");
exit(bisect_checkout(sha1_to_hex(mb)));
exit(bisect_checkout(sha1_to_hex(mb), no_checkout));
}
}

Expand Down Expand Up @@ -849,7 +857,7 @@ static int check_ancestors(const char *prefix)
* If a merge base must be tested by the user, its source code will be
* checked out to be tested by the user and we will exit.
*/
static void check_good_are_ancestors_of_bad(const char *prefix)
static void check_good_are_ancestors_of_bad(const char *prefix, int no_checkout)
{
const char *filename = git_path("BISECT_ANCESTORS_OK");
struct stat st;
Expand All @@ -868,7 +876,7 @@ static void check_good_are_ancestors_of_bad(const char *prefix)

/* Check if all good revs are ancestor of the bad rev. */
if (check_ancestors(prefix))
check_merge_bases();
check_merge_bases(no_checkout);

/* Create file BISECT_ANCESTORS_OK. */
fd = open(filename, O_CREAT | O_TRUNC | O_WRONLY, 0600);
Expand Down Expand Up @@ -908,8 +916,11 @@ static void show_diff_tree(const char *prefix, struct commit *commit)
* We use the convention that exiting with an exit code 10 means that
* the bisection process finished successfully.
* In this case the calling shell script should exit 0.
*
* If no_checkout is non-zero, the bisection process does not
* checkout the trial commit but instead simply updates BISECT_HEAD.
*/
int bisect_next_all(const char *prefix)
int bisect_next_all(const char *prefix, int no_checkout)
{
struct rev_info revs;
struct commit_list *tried;
Expand All @@ -920,7 +931,7 @@ int bisect_next_all(const char *prefix)
if (read_bisect_refs())
die("reading bisect refs failed");

check_good_are_ancestors_of_bad(prefix);
check_good_are_ancestors_of_bad(prefix, no_checkout);

bisect_rev_setup(&revs, prefix, "%s", "^%s", 1);
revs.limited = 1;
Expand Down Expand Up @@ -966,6 +977,6 @@ int bisect_next_all(const char *prefix)
"(roughly %d step%s)\n", nr, (nr == 1 ? "" : "s"),
steps, (steps == 1 ? "" : "s"));

return bisect_checkout(bisect_rev_hex);
return bisect_checkout(bisect_rev_hex, no_checkout);
}

2 changes: 1 addition & 1 deletion bisect.h
Original file line number Diff line number Diff line change
Expand Up @@ -27,7 +27,7 @@ struct rev_list_info {
const char *header_prefix;
};

extern int bisect_next_all(const char *prefix);
extern int bisect_next_all(const char *prefix, int no_checkout);

extern int estimate_bisect_steps(int all);

Expand Down
7 changes: 5 additions & 2 deletions builtin/bisect--helper.c
Original file line number Diff line number Diff line change
Expand Up @@ -4,16 +4,19 @@
#include "bisect.h"

static const char * const git_bisect_helper_usage[] = {
"git bisect--helper --next-all",
"git bisect--helper --next-all [--no-checkout]",
NULL
};

int cmd_bisect__helper(int argc, const char **argv, const char *prefix)
{
int next_all = 0;
int no_checkout = 0;
struct option options[] = {
OPT_BOOLEAN(0, "next-all", &next_all,
"perform 'git bisect next'"),
OPT_BOOLEAN(0, "no-checkout", &no_checkout,
"update BISECT_HEAD instead of checking out the current commit"),
OPT_END()
};

Expand All @@ -24,5 +27,5 @@ int cmd_bisect__helper(int argc, const char **argv, const char *prefix)
usage_with_options(git_bisect_helper_usage, options);

/* next-all */
return bisect_next_all(prefix);
return bisect_next_all(prefix, no_checkout);
}

0 comments on commit fee92fc

Please sign in to comment.