Browse Source
Preparation for a new merge strategy. * en/merge-ort-api-null-impl: merge,rebase,revert: select ort or recursive by config or environment fast-rebase: demonstrate merge-ort's API via new test-tool command merge-ort-wrappers: new convience wrappers to mimic the old merge API merge-ort: barebones API of new merge strategy with empty implementationmaint
Junio C Hamano
4 years ago
13 changed files with 517 additions and 15 deletions
@ -0,0 +1,62 @@
@@ -0,0 +1,62 @@
|
||||
#include "cache.h" |
||||
#include "merge-ort.h" |
||||
#include "merge-ort-wrappers.h" |
||||
|
||||
#include "commit.h" |
||||
|
||||
static int unclean(struct merge_options *opt, struct tree *head) |
||||
{ |
||||
/* Sanity check on repo state; index must match head */ |
||||
struct strbuf sb = STRBUF_INIT; |
||||
|
||||
if (head && repo_index_has_changes(opt->repo, head, &sb)) { |
||||
fprintf(stderr, _("Your local changes to the following files would be overwritten by merge:\n %s"), |
||||
sb.buf); |
||||
strbuf_release(&sb); |
||||
return -1; |
||||
} |
||||
|
||||
return 0; |
||||
} |
||||
|
||||
int merge_ort_nonrecursive(struct merge_options *opt, |
||||
struct tree *head, |
||||
struct tree *merge, |
||||
struct tree *merge_base) |
||||
{ |
||||
struct merge_result result; |
||||
|
||||
if (unclean(opt, head)) |
||||
return -1; |
||||
|
||||
if (oideq(&merge_base->object.oid, &merge->object.oid)) { |
||||
printf(_("Already up to date!")); |
||||
return 1; |
||||
} |
||||
|
||||
memset(&result, 0, sizeof(result)); |
||||
merge_incore_nonrecursive(opt, merge_base, head, merge, &result); |
||||
merge_switch_to_result(opt, head, &result, 1, 1); |
||||
|
||||
return result.clean; |
||||
} |
||||
|
||||
int merge_ort_recursive(struct merge_options *opt, |
||||
struct commit *side1, |
||||
struct commit *side2, |
||||
struct commit_list *merge_bases, |
||||
struct commit **result) |
||||
{ |
||||
struct tree *head = repo_get_commit_tree(opt->repo, side1); |
||||
struct merge_result tmp; |
||||
|
||||
if (unclean(opt, head)) |
||||
return -1; |
||||
|
||||
memset(&tmp, 0, sizeof(tmp)); |
||||
merge_incore_recursive(opt, merge_bases, side1, side2, &tmp); |
||||
merge_switch_to_result(opt, head, &tmp, 1, 1); |
||||
*result = NULL; |
||||
|
||||
return tmp.clean; |
||||
} |
@ -0,0 +1,25 @@
@@ -0,0 +1,25 @@
|
||||
#ifndef MERGE_ORT_WRAPPERS_H |
||||
#define MERGE_ORT_WRAPPERS_H |
||||
|
||||
#include "merge-recursive.h" |
||||
|
||||
/* |
||||
* rename-detecting three-way merge, no recursion. |
||||
* Wrapper mimicking the old merge_trees() function. |
||||
*/ |
||||
int merge_ort_nonrecursive(struct merge_options *opt, |
||||
struct tree *head, |
||||
struct tree *merge, |
||||
struct tree *common); |
||||
|
||||
/* |
||||
* rename-detecting three-way merge with recursive ancestor consolidation. |
||||
* Wrapper mimicking the old merge_recursive() function. |
||||
*/ |
||||
int merge_ort_recursive(struct merge_options *opt, |
||||
struct commit *h1, |
||||
struct commit *h2, |
||||
struct commit_list *ancestors, |
||||
struct commit **result); |
||||
|
||||
#endif |
@ -0,0 +1,52 @@
@@ -0,0 +1,52 @@
|
||||
/* |
||||
* "Ostensibly Recursive's Twin" merge strategy, or "ort" for short. Meant |
||||
* as a drop-in replacement for the "recursive" merge strategy, allowing one |
||||
* to replace |
||||
* |
||||
* git merge [-s recursive] |
||||
* |
||||
* with |
||||
* |
||||
* git merge -s ort |
||||
* |
||||
* Note: git's parser allows the space between '-s' and its argument to be |
||||
* missing. (Should I have backronymed "ham", "alsa", "kip", "nap, "alvo", |
||||
* "cale", "peedy", or "ins" instead of "ort"?) |
||||
*/ |
||||
|
||||
#include "cache.h" |
||||
#include "merge-ort.h" |
||||
|
||||
void merge_switch_to_result(struct merge_options *opt, |
||||
struct tree *head, |
||||
struct merge_result *result, |
||||
int update_worktree_and_index, |
||||
int display_update_msgs) |
||||
{ |
||||
die("Not yet implemented"); |
||||
merge_finalize(opt, result); |
||||
} |
||||
|
||||
void merge_finalize(struct merge_options *opt, |
||||
struct merge_result *result) |
||||
{ |
||||
die("Not yet implemented"); |
||||
} |
||||
|
||||
void merge_incore_nonrecursive(struct merge_options *opt, |
||||
struct tree *merge_base, |
||||
struct tree *side1, |
||||
struct tree *side2, |
||||
struct merge_result *result) |
||||
{ |
||||
die("Not yet implemented"); |
||||
} |
||||
|
||||
void merge_incore_recursive(struct merge_options *opt, |
||||
struct commit_list *merge_bases, |
||||
struct commit *side1, |
||||
struct commit *side2, |
||||
struct merge_result *result) |
||||
{ |
||||
die("Not yet implemented"); |
||||
} |
@ -0,0 +1,58 @@
@@ -0,0 +1,58 @@
|
||||
#ifndef MERGE_ORT_H |
||||
#define MERGE_ORT_H |
||||
|
||||
#include "merge-recursive.h" |
||||
|
||||
struct commit; |
||||
struct tree; |
||||
|
||||
struct merge_result { |
||||
/* Whether the merge is clean */ |
||||
int clean; |
||||
|
||||
/* |
||||
* Result of merge. If !clean, represents what would go in worktree |
||||
* (thus possibly including files containing conflict markers). |
||||
*/ |
||||
struct tree *tree; |
||||
|
||||
/* |
||||
* Additional metadata used by merge_switch_to_result() or future calls |
||||
* to merge_incore_*(). Includes data needed to update the index (if |
||||
* !clean) and to print "CONFLICT" messages. Not for external use. |
||||
*/ |
||||
void *priv; |
||||
}; |
||||
|
||||
/* |
||||
* rename-detecting three-way merge with recursive ancestor consolidation. |
||||
* working tree and index are untouched. |
||||
*/ |
||||
void merge_incore_recursive(struct merge_options *opt, |
||||
struct commit_list *merge_bases, |
||||
struct commit *side1, |
||||
struct commit *side2, |
||||
struct merge_result *result); |
||||
|
||||
/* |
||||
* rename-detecting three-way merge, no recursion. |
||||
* working tree and index are untouched. |
||||
*/ |
||||
void merge_incore_nonrecursive(struct merge_options *opt, |
||||
struct tree *merge_base, |
||||
struct tree *side1, |
||||
struct tree *side2, |
||||
struct merge_result *result); |
||||
|
||||
/* Update the working tree and index from head to result after incore merge */ |
||||
void merge_switch_to_result(struct merge_options *opt, |
||||
struct tree *head, |
||||
struct merge_result *result, |
||||
int update_worktree_and_index, |
||||
int display_update_msgs); |
||||
|
||||
/* Do needed cleanup when not calling merge_switch_to_result() */ |
||||
void merge_finalize(struct merge_options *opt, |
||||
struct merge_result *result); |
||||
|
||||
#endif |
@ -0,0 +1,211 @@
@@ -0,0 +1,211 @@
|
||||
/* |
||||
* "git fast-rebase" builtin command |
||||
* |
||||
* FAST: Forking Any Subprocesses (is) Taboo |
||||
* |
||||
* This is meant SOLELY as a demo of what is possible. sequencer.c and |
||||
* rebase.c should be refactored to use the ideas here, rather than attempting |
||||
* to extend this file to replace those (unless Phillip or Dscho say that |
||||
* refactoring is too hard and we need a clean slate, but I'm guessing that |
||||
* refactoring is the better route). |
||||
*/ |
||||
|
||||
#define USE_THE_INDEX_COMPATIBILITY_MACROS |
||||
#include "test-tool.h" |
||||
|
||||
#include "cache-tree.h" |
||||
#include "commit.h" |
||||
#include "lockfile.h" |
||||
#include "merge-ort.h" |
||||
#include "refs.h" |
||||
#include "revision.h" |
||||
#include "sequencer.h" |
||||
#include "strvec.h" |
||||
#include "tree.h" |
||||
|
||||
static const char *short_commit_name(struct commit *commit) |
||||
{ |
||||
return find_unique_abbrev(&commit->object.oid, DEFAULT_ABBREV); |
||||
} |
||||
|
||||
static struct commit *peel_committish(const char *name) |
||||
{ |
||||
struct object *obj; |
||||
struct object_id oid; |
||||
|
||||
if (get_oid(name, &oid)) |
||||
return NULL; |
||||
obj = parse_object(the_repository, &oid); |
||||
return (struct commit *)peel_to_type(name, 0, obj, OBJ_COMMIT); |
||||
} |
||||
|
||||
static char *get_author(const char *message) |
||||
{ |
||||
size_t len; |
||||
const char *a; |
||||
|
||||
a = find_commit_header(message, "author", &len); |
||||
if (a) |
||||
return xmemdupz(a, len); |
||||
|
||||
return NULL; |
||||
} |
||||
|
||||
static struct commit *create_commit(struct tree *tree, |
||||
struct commit *based_on, |
||||
struct commit *parent) |
||||
{ |
||||
struct object_id ret; |
||||
struct object *obj; |
||||
struct commit_list *parents = NULL; |
||||
char *author; |
||||
char *sign_commit = NULL; |
||||
struct commit_extra_header *extra; |
||||
struct strbuf msg = STRBUF_INIT; |
||||
const char *out_enc = get_commit_output_encoding(); |
||||
const char *message = logmsg_reencode(based_on, NULL, out_enc); |
||||
const char *orig_message = NULL; |
||||
const char *exclude_gpgsig[] = { "gpgsig", NULL }; |
||||
|
||||
commit_list_insert(parent, &parents); |
||||
extra = read_commit_extra_headers(based_on, exclude_gpgsig); |
||||
find_commit_subject(message, &orig_message); |
||||
strbuf_addstr(&msg, orig_message); |
||||
author = get_author(message); |
||||
reset_ident_date(); |
||||
if (commit_tree_extended(msg.buf, msg.len, &tree->object.oid, parents, |
||||
&ret, author, NULL, sign_commit, extra)) { |
||||
error(_("failed to write commit object")); |
||||
return NULL; |
||||
} |
||||
free(author); |
||||
strbuf_release(&msg); |
||||
|
||||
obj = parse_object(the_repository, &ret); |
||||
return (struct commit *)obj; |
||||
} |
||||
|
||||
int cmd__fast_rebase(int argc, const char **argv) |
||||
{ |
||||
struct commit *onto; |
||||
struct commit *last_commit = NULL, *last_picked_commit = NULL; |
||||
struct object_id head; |
||||
struct lock_file lock = LOCK_INIT; |
||||
int clean = 1; |
||||
struct strvec rev_walk_args = STRVEC_INIT; |
||||
struct rev_info revs; |
||||
struct commit *commit; |
||||
struct merge_options merge_opt; |
||||
struct tree *next_tree, *base_tree, *head_tree; |
||||
struct merge_result result; |
||||
struct strbuf reflog_msg = STRBUF_INIT; |
||||
struct strbuf branch_name = STRBUF_INIT; |
||||
|
||||
/* |
||||
* test-tool stuff doesn't set up the git directory by default; need to |
||||
* do that manually. |
||||
*/ |
||||
setup_git_directory(); |
||||
|
||||
if (argc == 2 && !strcmp(argv[1], "-h")) { |
||||
printf("Sorry, I am not a psychiatrist; I can not give you the help you need. Oh, you meant usage...\n"); |
||||
exit(129); |
||||
} |
||||
|
||||
if (argc != 5 || strcmp(argv[1], "--onto")) |
||||
die("usage: read the code, figure out how to use it, then do so"); |
||||
|
||||
onto = peel_committish(argv[2]); |
||||
strbuf_addf(&branch_name, "refs/heads/%s", argv[4]); |
||||
|
||||
/* Sanity check */ |
||||
if (get_oid("HEAD", &head)) |
||||
die(_("Cannot read HEAD")); |
||||
assert(oideq(&onto->object.oid, &head)); |
||||
|
||||
hold_locked_index(&lock, LOCK_DIE_ON_ERROR); |
||||
assert(repo_read_index(the_repository) >= 0); |
||||
|
||||
repo_init_revisions(the_repository, &revs, NULL); |
||||
revs.verbose_header = 1; |
||||
revs.max_parents = 1; |
||||
revs.cherry_mark = 1; |
||||
revs.limited = 1; |
||||
revs.reverse = 1; |
||||
revs.right_only = 1; |
||||
revs.sort_order = REV_SORT_IN_GRAPH_ORDER; |
||||
revs.topo_order = 1; |
||||
strvec_pushl(&rev_walk_args, "", argv[4], "--not", argv[3], NULL); |
||||
|
||||
if (setup_revisions(rev_walk_args.nr, rev_walk_args.v, &revs, NULL) > 1) |
||||
return error(_("unhandled options")); |
||||
|
||||
strvec_clear(&rev_walk_args); |
||||
|
||||
if (prepare_revision_walk(&revs) < 0) |
||||
return error(_("error preparing revisions")); |
||||
|
||||
init_merge_options(&merge_opt, the_repository); |
||||
memset(&result, 0, sizeof(result)); |
||||
merge_opt.show_rename_progress = 1; |
||||
merge_opt.branch1 = "HEAD"; |
||||
head_tree = get_commit_tree(onto); |
||||
result.tree = head_tree; |
||||
last_commit = onto; |
||||
while ((commit = get_revision(&revs))) { |
||||
struct commit *base; |
||||
|
||||
fprintf(stderr, "Rebasing %s...\r", |
||||
oid_to_hex(&commit->object.oid)); |
||||
assert(commit->parents && !commit->parents->next); |
||||
base = commit->parents->item; |
||||
|
||||
next_tree = get_commit_tree(commit); |
||||
base_tree = get_commit_tree(base); |
||||
|
||||
merge_opt.branch2 = short_commit_name(commit); |
||||
merge_opt.ancestor = xstrfmt("parent of %s", merge_opt.branch2); |
||||
|
||||
merge_incore_nonrecursive(&merge_opt, |
||||
base_tree, |
||||
result.tree, |
||||
next_tree, |
||||
&result); |
||||
|
||||
free((char*)merge_opt.ancestor); |
||||
merge_opt.ancestor = NULL; |
||||
if (!result.clean) |
||||
die("Aborting: Hit a conflict and restarting is not implemented."); |
||||
last_picked_commit = commit; |
||||
last_commit = create_commit(result.tree, commit, last_commit); |
||||
} |
||||
fprintf(stderr, "\nDone.\n"); |
||||
/* TODO: There should be some kind of rev_info_free(&revs) call... */ |
||||
memset(&revs, 0, sizeof(revs)); |
||||
|
||||
merge_switch_to_result(&merge_opt, head_tree, &result, 1, !result.clean); |
||||
|
||||
if (result.clean < 0) |
||||
exit(128); |
||||
|
||||
strbuf_addf(&reflog_msg, "finish rebase %s onto %s", |
||||
oid_to_hex(&last_picked_commit->object.oid), |
||||
oid_to_hex(&last_commit->object.oid)); |
||||
if (update_ref(reflog_msg.buf, branch_name.buf, |
||||
&last_commit->object.oid, |
||||
&last_picked_commit->object.oid, |
||||
REF_NO_DEREF, UPDATE_REFS_MSG_ON_ERR)) { |
||||
error(_("could not update %s"), argv[4]); |
||||
die("Failed to update %s", argv[4]); |
||||
} |
||||
if (create_symref("HEAD", branch_name.buf, reflog_msg.buf) < 0) |
||||
die(_("unable to update HEAD")); |
||||
strbuf_release(&reflog_msg); |
||||
strbuf_release(&branch_name); |
||||
|
||||
prime_cache_tree(the_repository, the_repository->index, result.tree); |
||||
if (write_locked_index(&the_index, &lock, |
||||
COMMIT_LOCK | SKIP_IF_UNCHANGED)) |
||||
die(_("unable to write %s"), get_index_file()); |
||||
return (clean == 0); |
||||
} |
Loading…
Reference in new issue