2008-07-09 20:58:57 +08:00
|
|
|
#include "cache.h"
|
2014-10-01 18:28:42 +08:00
|
|
|
#include "lockfile.h"
|
2008-07-22 02:03:49 +08:00
|
|
|
#include "string-list.h"
|
2008-07-09 20:58:57 +08:00
|
|
|
#include "rerere.h"
|
|
|
|
#include "xdiff-interface.h"
|
2009-12-26 07:51:32 +08:00
|
|
|
#include "dir.h"
|
|
|
|
#include "resolve-undo.h"
|
|
|
|
#include "ll-merge.h"
|
2010-01-17 15:28:46 +08:00
|
|
|
#include "attr.h"
|
2013-07-14 16:35:40 +08:00
|
|
|
#include "pathspec.h"
|
rerere: split conflict ID further
The plan is to keep assigning the backward compatible conflict ID
based on the hash of the (normalized) text of conflicts, keep using
that conflict ID as the directory name under $GIT_DIR/rr-cache/, but
allow each conflicted path to use a separate "variant" to record
resolutions, i.e. having more than one <preimage,postimage> pairs
under $GIT_DIR/rr-cache/$ID/ directory. As the first step in that
direction, separate the shared "conflict ID" out of the rerere_id
structure.
The plan is to keep information per $ID in rerere_dir, that can be
shared among rerere_id that is per conflicted path.
When we are done with rerere(), which can be directly called from
other programs like "git apply", "git commit" and "git merge", the
shared rerere_dir structures can be freed entirely, so they are not
reference-counted and they are not freed when we release rerere_id's
that reference them.
Signed-off-by: Junio C Hamano <gitster@pobox.com>
2015-07-17 05:50:05 +08:00
|
|
|
#include "sha1-lookup.h"
|
2008-07-09 20:58:57 +08:00
|
|
|
|
2011-02-16 18:47:44 +08:00
|
|
|
#define RESOLVED 0
|
|
|
|
#define PUNTED 1
|
|
|
|
#define THREE_STAGED 2
|
|
|
|
void *RERERE_RESOLVED = &RERERE_RESOLVED;
|
|
|
|
|
2008-07-09 20:58:57 +08:00
|
|
|
/* if rerere_enabled == -1, fall back to detection of .git/rr-cache */
|
|
|
|
static int rerere_enabled = -1;
|
|
|
|
|
|
|
|
/* automatically update cleanly resolved paths to the index */
|
|
|
|
static int rerere_autoupdate;
|
|
|
|
|
|
|
|
static char *merge_rr_path;
|
|
|
|
|
rerere: split conflict ID further
The plan is to keep assigning the backward compatible conflict ID
based on the hash of the (normalized) text of conflicts, keep using
that conflict ID as the directory name under $GIT_DIR/rr-cache/, but
allow each conflicted path to use a separate "variant" to record
resolutions, i.e. having more than one <preimage,postimage> pairs
under $GIT_DIR/rr-cache/$ID/ directory. As the first step in that
direction, separate the shared "conflict ID" out of the rerere_id
structure.
The plan is to keep information per $ID in rerere_dir, that can be
shared among rerere_id that is per conflicted path.
When we are done with rerere(), which can be directly called from
other programs like "git apply", "git commit" and "git merge", the
shared rerere_dir structures can be freed entirely, so they are not
reference-counted and they are not freed when we release rerere_id's
that reference them.
Signed-off-by: Junio C Hamano <gitster@pobox.com>
2015-07-17 05:50:05 +08:00
|
|
|
static int rerere_dir_nr;
|
|
|
|
static int rerere_dir_alloc;
|
|
|
|
|
2015-07-17 06:47:13 +08:00
|
|
|
#define RR_HAS_POSTIMAGE 1
|
|
|
|
#define RR_HAS_PREIMAGE 2
|
rerere: split conflict ID further
The plan is to keep assigning the backward compatible conflict ID
based on the hash of the (normalized) text of conflicts, keep using
that conflict ID as the directory name under $GIT_DIR/rr-cache/, but
allow each conflicted path to use a separate "variant" to record
resolutions, i.e. having more than one <preimage,postimage> pairs
under $GIT_DIR/rr-cache/$ID/ directory. As the first step in that
direction, separate the shared "conflict ID" out of the rerere_id
structure.
The plan is to keep information per $ID in rerere_dir, that can be
shared among rerere_id that is per conflicted path.
When we are done with rerere(), which can be directly called from
other programs like "git apply", "git commit" and "git merge", the
shared rerere_dir structures can be freed entirely, so they are not
reference-counted and they are not freed when we release rerere_id's
that reference them.
Signed-off-by: Junio C Hamano <gitster@pobox.com>
2015-07-17 05:50:05 +08:00
|
|
|
static struct rerere_dir {
|
|
|
|
unsigned char sha1[20];
|
2015-07-17 06:47:13 +08:00
|
|
|
unsigned char status;
|
rerere: split conflict ID further
The plan is to keep assigning the backward compatible conflict ID
based on the hash of the (normalized) text of conflicts, keep using
that conflict ID as the directory name under $GIT_DIR/rr-cache/, but
allow each conflicted path to use a separate "variant" to record
resolutions, i.e. having more than one <preimage,postimage> pairs
under $GIT_DIR/rr-cache/$ID/ directory. As the first step in that
direction, separate the shared "conflict ID" out of the rerere_id
structure.
The plan is to keep information per $ID in rerere_dir, that can be
shared among rerere_id that is per conflicted path.
When we are done with rerere(), which can be directly called from
other programs like "git apply", "git commit" and "git merge", the
shared rerere_dir structures can be freed entirely, so they are not
reference-counted and they are not freed when we release rerere_id's
that reference them.
Signed-off-by: Junio C Hamano <gitster@pobox.com>
2015-07-17 05:50:05 +08:00
|
|
|
} **rerere_dir;
|
|
|
|
|
|
|
|
static void free_rerere_dirs(void)
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
for (i = 0; i < rerere_dir_nr; i++)
|
|
|
|
free(rerere_dir[i]);
|
|
|
|
free(rerere_dir);
|
|
|
|
rerere_dir_nr = rerere_dir_alloc = 0;
|
|
|
|
rerere_dir = NULL;
|
|
|
|
}
|
|
|
|
|
2015-07-05 08:38:34 +08:00
|
|
|
static void free_rerere_id(struct string_list_item *item)
|
2008-07-09 20:58:57 +08:00
|
|
|
{
|
2015-07-05 08:38:34 +08:00
|
|
|
free(item->util);
|
|
|
|
}
|
|
|
|
|
|
|
|
static const char *rerere_id_hex(const struct rerere_id *id)
|
|
|
|
{
|
rerere: split conflict ID further
The plan is to keep assigning the backward compatible conflict ID
based on the hash of the (normalized) text of conflicts, keep using
that conflict ID as the directory name under $GIT_DIR/rr-cache/, but
allow each conflicted path to use a separate "variant" to record
resolutions, i.e. having more than one <preimage,postimage> pairs
under $GIT_DIR/rr-cache/$ID/ directory. As the first step in that
direction, separate the shared "conflict ID" out of the rerere_id
structure.
The plan is to keep information per $ID in rerere_dir, that can be
shared among rerere_id that is per conflicted path.
When we are done with rerere(), which can be directly called from
other programs like "git apply", "git commit" and "git merge", the
shared rerere_dir structures can be freed entirely, so they are not
reference-counted and they are not freed when we release rerere_id's
that reference them.
Signed-off-by: Junio C Hamano <gitster@pobox.com>
2015-07-17 05:50:05 +08:00
|
|
|
return sha1_to_hex(id->collection->sha1);
|
2015-07-05 08:38:34 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
const char *rerere_path(const struct rerere_id *id, const char *file)
|
|
|
|
{
|
|
|
|
if (!file)
|
|
|
|
return git_path("rr-cache/%s", rerere_id_hex(id));
|
|
|
|
|
|
|
|
return git_path("rr-cache/%s/%s", rerere_id_hex(id), file);
|
2008-07-09 20:58:57 +08:00
|
|
|
}
|
|
|
|
|
2015-07-17 06:47:13 +08:00
|
|
|
static int is_rr_file(const char *name, const char *filename)
|
|
|
|
{
|
|
|
|
return !strcmp(name, filename);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void scan_rerere_dir(struct rerere_dir *rr_dir)
|
|
|
|
{
|
|
|
|
struct dirent *de;
|
|
|
|
DIR *dir = opendir(git_path("rr-cache/%s", sha1_to_hex(rr_dir->sha1)));
|
|
|
|
|
|
|
|
if (!dir)
|
|
|
|
return;
|
|
|
|
while ((de = readdir(dir)) != NULL) {
|
|
|
|
if (is_rr_file(de->d_name, "postimage"))
|
|
|
|
rr_dir->status |= RR_HAS_POSTIMAGE;
|
|
|
|
else if (is_rr_file(de->d_name, "preimage"))
|
|
|
|
rr_dir->status |= RR_HAS_PREIMAGE;
|
|
|
|
}
|
|
|
|
closedir(dir);
|
|
|
|
}
|
|
|
|
|
rerere: split conflict ID further
The plan is to keep assigning the backward compatible conflict ID
based on the hash of the (normalized) text of conflicts, keep using
that conflict ID as the directory name under $GIT_DIR/rr-cache/, but
allow each conflicted path to use a separate "variant" to record
resolutions, i.e. having more than one <preimage,postimage> pairs
under $GIT_DIR/rr-cache/$ID/ directory. As the first step in that
direction, separate the shared "conflict ID" out of the rerere_id
structure.
The plan is to keep information per $ID in rerere_dir, that can be
shared among rerere_id that is per conflicted path.
When we are done with rerere(), which can be directly called from
other programs like "git apply", "git commit" and "git merge", the
shared rerere_dir structures can be freed entirely, so they are not
reference-counted and they are not freed when we release rerere_id's
that reference them.
Signed-off-by: Junio C Hamano <gitster@pobox.com>
2015-07-17 05:50:05 +08:00
|
|
|
static const unsigned char *rerere_dir_sha1(size_t i, void *table)
|
|
|
|
{
|
|
|
|
struct rerere_dir **rr_dir = table;
|
|
|
|
return rr_dir[i]->sha1;
|
|
|
|
}
|
|
|
|
|
|
|
|
static struct rerere_dir *find_rerere_dir(const char *hex)
|
|
|
|
{
|
|
|
|
unsigned char sha1[20];
|
|
|
|
struct rerere_dir *rr_dir;
|
|
|
|
int pos;
|
|
|
|
|
|
|
|
if (get_sha1_hex(hex, sha1))
|
|
|
|
return NULL; /* BUG */
|
|
|
|
pos = sha1_pos(sha1, rerere_dir, rerere_dir_nr, rerere_dir_sha1);
|
|
|
|
if (pos < 0) {
|
|
|
|
rr_dir = xmalloc(sizeof(*rr_dir));
|
|
|
|
hashcpy(rr_dir->sha1, sha1);
|
2015-07-17 06:47:13 +08:00
|
|
|
rr_dir->status = 0;
|
rerere: split conflict ID further
The plan is to keep assigning the backward compatible conflict ID
based on the hash of the (normalized) text of conflicts, keep using
that conflict ID as the directory name under $GIT_DIR/rr-cache/, but
allow each conflicted path to use a separate "variant" to record
resolutions, i.e. having more than one <preimage,postimage> pairs
under $GIT_DIR/rr-cache/$ID/ directory. As the first step in that
direction, separate the shared "conflict ID" out of the rerere_id
structure.
The plan is to keep information per $ID in rerere_dir, that can be
shared among rerere_id that is per conflicted path.
When we are done with rerere(), which can be directly called from
other programs like "git apply", "git commit" and "git merge", the
shared rerere_dir structures can be freed entirely, so they are not
reference-counted and they are not freed when we release rerere_id's
that reference them.
Signed-off-by: Junio C Hamano <gitster@pobox.com>
2015-07-17 05:50:05 +08:00
|
|
|
pos = -1 - pos;
|
|
|
|
|
|
|
|
/* Make sure the array is big enough ... */
|
|
|
|
ALLOC_GROW(rerere_dir, rerere_dir_nr + 1, rerere_dir_alloc);
|
|
|
|
/* ... and add it in. */
|
|
|
|
rerere_dir_nr++;
|
|
|
|
memmove(rerere_dir + pos + 1, rerere_dir + pos,
|
|
|
|
(rerere_dir_nr - pos - 1) * sizeof(*rerere_dir));
|
|
|
|
rerere_dir[pos] = rr_dir;
|
2015-07-17 06:47:13 +08:00
|
|
|
scan_rerere_dir(rr_dir);
|
rerere: split conflict ID further
The plan is to keep assigning the backward compatible conflict ID
based on the hash of the (normalized) text of conflicts, keep using
that conflict ID as the directory name under $GIT_DIR/rr-cache/, but
allow each conflicted path to use a separate "variant" to record
resolutions, i.e. having more than one <preimage,postimage> pairs
under $GIT_DIR/rr-cache/$ID/ directory. As the first step in that
direction, separate the shared "conflict ID" out of the rerere_id
structure.
The plan is to keep information per $ID in rerere_dir, that can be
shared among rerere_id that is per conflicted path.
When we are done with rerere(), which can be directly called from
other programs like "git apply", "git commit" and "git merge", the
shared rerere_dir structures can be freed entirely, so they are not
reference-counted and they are not freed when we release rerere_id's
that reference them.
Signed-off-by: Junio C Hamano <gitster@pobox.com>
2015-07-17 05:50:05 +08:00
|
|
|
}
|
|
|
|
return rerere_dir[pos];
|
|
|
|
}
|
|
|
|
|
2015-07-05 08:38:34 +08:00
|
|
|
static int has_rerere_resolution(const struct rerere_id *id)
|
2008-07-09 20:58:57 +08:00
|
|
|
{
|
2015-07-17 06:47:13 +08:00
|
|
|
return (id->collection->status & RR_HAS_POSTIMAGE);
|
2008-07-09 20:58:57 +08:00
|
|
|
}
|
|
|
|
|
2015-07-05 08:38:34 +08:00
|
|
|
static struct rerere_id *new_rerere_id_hex(char *hex)
|
|
|
|
{
|
|
|
|
struct rerere_id *id = xmalloc(sizeof(*id));
|
rerere: split conflict ID further
The plan is to keep assigning the backward compatible conflict ID
based on the hash of the (normalized) text of conflicts, keep using
that conflict ID as the directory name under $GIT_DIR/rr-cache/, but
allow each conflicted path to use a separate "variant" to record
resolutions, i.e. having more than one <preimage,postimage> pairs
under $GIT_DIR/rr-cache/$ID/ directory. As the first step in that
direction, separate the shared "conflict ID" out of the rerere_id
structure.
The plan is to keep information per $ID in rerere_dir, that can be
shared among rerere_id that is per conflicted path.
When we are done with rerere(), which can be directly called from
other programs like "git apply", "git commit" and "git merge", the
shared rerere_dir structures can be freed entirely, so they are not
reference-counted and they are not freed when we release rerere_id's
that reference them.
Signed-off-by: Junio C Hamano <gitster@pobox.com>
2015-07-17 05:50:05 +08:00
|
|
|
id->collection = find_rerere_dir(hex);
|
2015-07-05 08:38:34 +08:00
|
|
|
return id;
|
|
|
|
}
|
|
|
|
|
|
|
|
static struct rerere_id *new_rerere_id(unsigned char *sha1)
|
|
|
|
{
|
|
|
|
return new_rerere_id_hex(sha1_to_hex(sha1));
|
|
|
|
}
|
|
|
|
|
2015-07-01 13:36:35 +08:00
|
|
|
/*
|
|
|
|
* $GIT_DIR/MERGE_RR file is a collection of records, each of which is
|
|
|
|
* "conflict ID", a HT and pathname, terminated with a NUL, and is
|
|
|
|
* used to keep track of the set of paths that "rerere" may need to
|
|
|
|
* work on (i.e. what is left by the previous invocation of "git
|
|
|
|
* rerere" during the current conflict resolution session).
|
|
|
|
*/
|
2008-07-22 02:03:49 +08:00
|
|
|
static void read_rr(struct string_list *rr)
|
2008-07-09 20:58:57 +08:00
|
|
|
{
|
2015-06-29 06:51:59 +08:00
|
|
|
struct strbuf buf = STRBUF_INIT;
|
2008-07-09 20:58:57 +08:00
|
|
|
FILE *in = fopen(merge_rr_path, "r");
|
2015-06-29 06:51:59 +08:00
|
|
|
|
2008-07-09 20:58:57 +08:00
|
|
|
if (!in)
|
|
|
|
return;
|
2015-06-29 06:51:59 +08:00
|
|
|
while (!strbuf_getwholeline(&buf, in, '\0')) {
|
|
|
|
char *path;
|
|
|
|
unsigned char sha1[20];
|
2015-07-05 08:38:34 +08:00
|
|
|
struct rerere_id *id;
|
2015-06-29 06:51:59 +08:00
|
|
|
|
|
|
|
/* There has to be the hash, tab, path and then NUL */
|
|
|
|
if (buf.len < 42 || get_sha1_hex(buf.buf, sha1))
|
2008-07-09 20:58:57 +08:00
|
|
|
die("corrupt MERGE_RR");
|
2015-06-29 06:51:59 +08:00
|
|
|
|
|
|
|
if (buf.buf[40] != '\t')
|
2008-07-09 20:58:57 +08:00
|
|
|
die("corrupt MERGE_RR");
|
2015-06-29 06:51:59 +08:00
|
|
|
buf.buf[40] = '\0';
|
|
|
|
path = buf.buf + 41;
|
2015-07-05 08:38:34 +08:00
|
|
|
id = new_rerere_id_hex(buf.buf);
|
|
|
|
string_list_insert(rr, path)->util = id;
|
2008-07-09 20:58:57 +08:00
|
|
|
}
|
2015-06-29 06:51:59 +08:00
|
|
|
strbuf_release(&buf);
|
2008-07-09 20:58:57 +08:00
|
|
|
fclose(in);
|
|
|
|
}
|
|
|
|
|
|
|
|
static struct lock_file write_lock;
|
|
|
|
|
2008-07-22 02:03:49 +08:00
|
|
|
static int write_rr(struct string_list *rr, int out_fd)
|
2008-07-09 20:58:57 +08:00
|
|
|
{
|
|
|
|
int i;
|
|
|
|
for (i = 0; i < rr->nr; i++) {
|
2015-06-29 07:28:00 +08:00
|
|
|
struct strbuf buf = STRBUF_INIT;
|
2015-07-05 08:38:34 +08:00
|
|
|
struct rerere_id *id;
|
2015-06-29 07:28:00 +08:00
|
|
|
|
|
|
|
assert(rr->items[i].util != RERERE_RESOLVED);
|
2015-07-05 08:38:34 +08:00
|
|
|
|
|
|
|
id = rr->items[i].util;
|
|
|
|
if (!id)
|
2008-07-09 20:58:57 +08:00
|
|
|
continue;
|
2015-06-29 07:28:00 +08:00
|
|
|
strbuf_addf(&buf, "%s\t%s%c",
|
2015-07-05 08:38:34 +08:00
|
|
|
rerere_id_hex(id),
|
2015-06-29 07:28:00 +08:00
|
|
|
rr->items[i].string, 0);
|
|
|
|
if (write_in_full(out_fd, buf.buf, buf.len) != buf.len)
|
2008-07-09 20:58:57 +08:00
|
|
|
die("unable to write rerere record");
|
2015-06-29 07:28:00 +08:00
|
|
|
|
|
|
|
strbuf_release(&buf);
|
2008-07-09 20:58:57 +08:00
|
|
|
}
|
|
|
|
if (commit_lock_file(&write_lock) != 0)
|
|
|
|
die("unable to write rerere record");
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2015-07-01 13:33:19 +08:00
|
|
|
/*
|
|
|
|
* "rerere" interacts with conflicted file contents using this I/O
|
|
|
|
* abstraction. It reads a conflicted contents from one place via
|
|
|
|
* "getline()" method, and optionally can write it out after
|
|
|
|
* normalizing the conflicted hunks to the "output". Subclasses of
|
|
|
|
* rerere_io embed this structure at the beginning of their own
|
|
|
|
* rerere_io object.
|
|
|
|
*/
|
|
|
|
struct rerere_io {
|
|
|
|
int (*getline)(struct strbuf *, struct rerere_io *);
|
|
|
|
FILE *output;
|
|
|
|
int wrerror;
|
|
|
|
/* some more stuff */
|
|
|
|
};
|
|
|
|
|
2008-12-05 08:35:48 +08:00
|
|
|
static void ferr_write(const void *p, size_t count, FILE *fp, int *err)
|
|
|
|
{
|
|
|
|
if (!count || *err)
|
|
|
|
return;
|
|
|
|
if (fwrite(p, count, 1, fp) != 1)
|
|
|
|
*err = errno;
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline void ferr_puts(const char *s, FILE *fp, int *err)
|
|
|
|
{
|
|
|
|
ferr_write(s, strlen(s), fp, err);
|
|
|
|
}
|
|
|
|
|
2009-12-26 06:34:53 +08:00
|
|
|
static void rerere_io_putstr(const char *str, struct rerere_io *io)
|
|
|
|
{
|
|
|
|
if (io->output)
|
|
|
|
ferr_puts(str, io->output, &io->wrerror);
|
|
|
|
}
|
|
|
|
|
2015-07-01 13:33:19 +08:00
|
|
|
/*
|
|
|
|
* Write a conflict marker to io->output (if defined).
|
|
|
|
*/
|
2010-01-17 15:06:45 +08:00
|
|
|
static void rerere_io_putconflict(int ch, int size, struct rerere_io *io)
|
|
|
|
{
|
|
|
|
char buf[64];
|
|
|
|
|
|
|
|
while (size) {
|
2015-07-25 07:01:48 +08:00
|
|
|
if (size <= sizeof(buf) - 2) {
|
2010-01-17 15:06:45 +08:00
|
|
|
memset(buf, ch, size);
|
|
|
|
buf[size] = '\n';
|
|
|
|
buf[size + 1] = '\0';
|
|
|
|
size = 0;
|
|
|
|
} else {
|
|
|
|
int sz = sizeof(buf) - 1;
|
2015-07-25 07:01:48 +08:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Make sure we will not write everything out
|
|
|
|
* in this round by leaving at least 1 byte
|
|
|
|
* for the next round, giving the next round
|
|
|
|
* a chance to add the terminating LF. Yuck.
|
|
|
|
*/
|
2010-01-17 15:06:45 +08:00
|
|
|
if (size <= sz)
|
|
|
|
sz -= (sz - size) + 1;
|
|
|
|
memset(buf, ch, sz);
|
|
|
|
buf[sz] = '\0';
|
|
|
|
size -= sz;
|
|
|
|
}
|
|
|
|
rerere_io_putstr(buf, io);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-12-26 06:34:53 +08:00
|
|
|
static void rerere_io_putmem(const char *mem, size_t sz, struct rerere_io *io)
|
|
|
|
{
|
|
|
|
if (io->output)
|
|
|
|
ferr_write(mem, sz, io->output, &io->wrerror);
|
|
|
|
}
|
|
|
|
|
2015-07-01 13:33:19 +08:00
|
|
|
/*
|
|
|
|
* Subclass of rerere_io that reads from an on-disk file
|
|
|
|
*/
|
2009-12-26 06:34:53 +08:00
|
|
|
struct rerere_io_file {
|
|
|
|
struct rerere_io io;
|
|
|
|
FILE *input;
|
|
|
|
};
|
|
|
|
|
2015-07-01 13:33:19 +08:00
|
|
|
/*
|
|
|
|
* ... and its getline() method implementation
|
|
|
|
*/
|
2009-12-26 06:34:53 +08:00
|
|
|
static int rerere_file_getline(struct strbuf *sb, struct rerere_io *io_)
|
|
|
|
{
|
|
|
|
struct rerere_io_file *io = (struct rerere_io_file *)io_;
|
|
|
|
return strbuf_getwholeline(sb, io->input, '\n');
|
|
|
|
}
|
|
|
|
|
2015-06-30 06:05:24 +08:00
|
|
|
/*
|
|
|
|
* Require the exact number of conflict marker letters, no more, no
|
|
|
|
* less, followed by SP or any whitespace
|
|
|
|
* (including LF).
|
|
|
|
*/
|
|
|
|
static int is_cmarker(char *buf, int marker_char, int marker_size)
|
2010-01-17 15:06:45 +08:00
|
|
|
{
|
2015-06-30 06:05:24 +08:00
|
|
|
int want_sp;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* The beginning of our version and the end of their version
|
|
|
|
* always are labeled like "<<<<< ours" or ">>>>> theirs",
|
|
|
|
* hence we set want_sp for them. Note that the version from
|
|
|
|
* the common ancestor in diff3-style output is not always
|
|
|
|
* labelled (e.g. "||||| common" is often seen but "|||||"
|
|
|
|
* alone is also valid), so we do not set want_sp.
|
|
|
|
*/
|
|
|
|
want_sp = (marker_char == '<') || (marker_char == '>');
|
|
|
|
|
2010-01-17 15:06:45 +08:00
|
|
|
while (marker_size--)
|
|
|
|
if (*buf++ != marker_char)
|
|
|
|
return 0;
|
|
|
|
if (want_sp && *buf != ' ')
|
|
|
|
return 0;
|
|
|
|
return isspace(*buf);
|
|
|
|
}
|
|
|
|
|
2015-07-01 13:40:35 +08:00
|
|
|
/*
|
|
|
|
* Read contents a file with conflicts, normalize the conflicts
|
|
|
|
* by (1) discarding the common ancestor version in diff3-style,
|
|
|
|
* (2) reordering our side and their side so that whichever sorts
|
|
|
|
* alphabetically earlier comes before the other one, while
|
|
|
|
* computing the "conflict ID", which is just an SHA-1 hash of
|
|
|
|
* one side of the conflict, NUL, the other side of the conflict,
|
|
|
|
* and NUL concatenated together.
|
|
|
|
*
|
|
|
|
* Return the number of conflict hunks found.
|
|
|
|
*
|
|
|
|
* NEEDSWORK: the logic and theory of operation behind this conflict
|
|
|
|
* normalization may deserve to be documented somewhere, perhaps in
|
|
|
|
* Documentation/technical/rerere.txt.
|
|
|
|
*/
|
2010-01-17 15:06:45 +08:00
|
|
|
static int handle_path(unsigned char *sha1, struct rerere_io *io, int marker_size)
|
2008-07-09 20:58:57 +08:00
|
|
|
{
|
2008-10-02 02:05:20 +08:00
|
|
|
git_SHA_CTX ctx;
|
2008-08-30 01:12:23 +08:00
|
|
|
int hunk_no = 0;
|
|
|
|
enum {
|
2010-05-14 17:31:35 +08:00
|
|
|
RR_CONTEXT = 0, RR_SIDE_1, RR_SIDE_2, RR_ORIGINAL
|
2008-08-30 01:12:23 +08:00
|
|
|
} hunk = RR_CONTEXT;
|
2008-10-10 03:12:12 +08:00
|
|
|
struct strbuf one = STRBUF_INIT, two = STRBUF_INIT;
|
2009-12-26 05:55:29 +08:00
|
|
|
struct strbuf buf = STRBUF_INIT;
|
2008-07-09 20:58:57 +08:00
|
|
|
|
|
|
|
if (sha1)
|
2008-10-02 02:05:20 +08:00
|
|
|
git_SHA1_Init(&ctx);
|
2008-07-09 20:58:57 +08:00
|
|
|
|
2009-12-26 06:34:53 +08:00
|
|
|
while (!io->getline(&buf, io)) {
|
2015-06-30 06:05:24 +08:00
|
|
|
if (is_cmarker(buf.buf, '<', marker_size)) {
|
2008-08-30 01:12:23 +08:00
|
|
|
if (hunk != RR_CONTEXT)
|
2008-07-09 20:58:57 +08:00
|
|
|
goto bad;
|
2008-08-30 01:12:23 +08:00
|
|
|
hunk = RR_SIDE_1;
|
2015-06-30 06:05:24 +08:00
|
|
|
} else if (is_cmarker(buf.buf, '|', marker_size)) {
|
2008-08-30 01:12:23 +08:00
|
|
|
if (hunk != RR_SIDE_1)
|
2008-07-09 20:58:57 +08:00
|
|
|
goto bad;
|
2008-08-30 01:24:45 +08:00
|
|
|
hunk = RR_ORIGINAL;
|
2015-06-30 06:05:24 +08:00
|
|
|
} else if (is_cmarker(buf.buf, '=', marker_size)) {
|
2008-08-30 01:24:45 +08:00
|
|
|
if (hunk != RR_SIDE_1 && hunk != RR_ORIGINAL)
|
2008-07-09 20:58:57 +08:00
|
|
|
goto bad;
|
2008-08-30 01:12:23 +08:00
|
|
|
hunk = RR_SIDE_2;
|
2015-06-30 06:05:24 +08:00
|
|
|
} else if (is_cmarker(buf.buf, '>', marker_size)) {
|
2008-08-30 01:12:23 +08:00
|
|
|
if (hunk != RR_SIDE_2)
|
2008-07-09 20:58:57 +08:00
|
|
|
goto bad;
|
|
|
|
if (strbuf_cmp(&one, &two) > 0)
|
|
|
|
strbuf_swap(&one, &two);
|
|
|
|
hunk_no++;
|
2008-08-30 01:12:23 +08:00
|
|
|
hunk = RR_CONTEXT;
|
2010-01-17 15:06:45 +08:00
|
|
|
rerere_io_putconflict('<', marker_size, io);
|
2009-12-26 06:34:53 +08:00
|
|
|
rerere_io_putmem(one.buf, one.len, io);
|
2010-01-17 15:06:45 +08:00
|
|
|
rerere_io_putconflict('=', marker_size, io);
|
2009-12-26 06:34:53 +08:00
|
|
|
rerere_io_putmem(two.buf, two.len, io);
|
2010-01-17 15:06:45 +08:00
|
|
|
rerere_io_putconflict('>', marker_size, io);
|
2008-07-09 20:58:57 +08:00
|
|
|
if (sha1) {
|
2008-10-02 02:05:20 +08:00
|
|
|
git_SHA1_Update(&ctx, one.buf ? one.buf : "",
|
2008-07-09 20:58:57 +08:00
|
|
|
one.len + 1);
|
2008-10-02 02:05:20 +08:00
|
|
|
git_SHA1_Update(&ctx, two.buf ? two.buf : "",
|
2008-07-09 20:58:57 +08:00
|
|
|
two.len + 1);
|
|
|
|
}
|
|
|
|
strbuf_reset(&one);
|
|
|
|
strbuf_reset(&two);
|
2008-08-30 01:12:23 +08:00
|
|
|
} else if (hunk == RR_SIDE_1)
|
2014-07-10 16:52:21 +08:00
|
|
|
strbuf_addbuf(&one, &buf);
|
2008-08-30 01:24:45 +08:00
|
|
|
else if (hunk == RR_ORIGINAL)
|
|
|
|
; /* discard */
|
2008-08-30 01:12:23 +08:00
|
|
|
else if (hunk == RR_SIDE_2)
|
2014-07-10 16:52:21 +08:00
|
|
|
strbuf_addbuf(&two, &buf);
|
2009-12-26 06:34:53 +08:00
|
|
|
else
|
|
|
|
rerere_io_putstr(buf.buf, io);
|
2008-07-09 20:58:57 +08:00
|
|
|
continue;
|
|
|
|
bad:
|
|
|
|
hunk = 99; /* force error exit */
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
strbuf_release(&one);
|
|
|
|
strbuf_release(&two);
|
2009-12-26 05:55:29 +08:00
|
|
|
strbuf_release(&buf);
|
2008-07-09 20:58:57 +08:00
|
|
|
|
|
|
|
if (sha1)
|
2008-10-02 02:05:20 +08:00
|
|
|
git_SHA1_Final(sha1, &ctx);
|
2009-12-26 06:34:53 +08:00
|
|
|
if (hunk != RR_CONTEXT)
|
|
|
|
return -1;
|
|
|
|
return hunk_no;
|
|
|
|
}
|
|
|
|
|
2015-07-01 13:40:35 +08:00
|
|
|
/*
|
|
|
|
* Scan the path for conflicts, do the "handle_path()" thing above, and
|
|
|
|
* return the number of conflict hunks found.
|
|
|
|
*/
|
2009-12-26 06:34:53 +08:00
|
|
|
static int handle_file(const char *path, unsigned char *sha1, const char *output)
|
|
|
|
{
|
|
|
|
int hunk_no = 0;
|
|
|
|
struct rerere_io_file io;
|
2010-01-17 15:28:46 +08:00
|
|
|
int marker_size = ll_merge_marker_size(path);
|
2009-12-26 06:34:53 +08:00
|
|
|
|
|
|
|
memset(&io, 0, sizeof(io));
|
|
|
|
io.io.getline = rerere_file_getline;
|
|
|
|
io.input = fopen(path, "r");
|
|
|
|
io.io.wrerror = 0;
|
|
|
|
if (!io.input)
|
|
|
|
return error("Could not open %s", path);
|
|
|
|
|
|
|
|
if (output) {
|
|
|
|
io.io.output = fopen(output, "w");
|
|
|
|
if (!io.io.output) {
|
|
|
|
fclose(io.input);
|
|
|
|
return error("Could not write %s", output);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-01-17 15:06:45 +08:00
|
|
|
hunk_no = handle_path(sha1, (struct rerere_io *)&io, marker_size);
|
2009-12-26 06:34:53 +08:00
|
|
|
|
|
|
|
fclose(io.input);
|
|
|
|
if (io.io.wrerror)
|
|
|
|
error("There were errors while writing %s (%s)",
|
|
|
|
path, strerror(io.io.wrerror));
|
|
|
|
if (io.io.output && fclose(io.io.output))
|
|
|
|
io.io.wrerror = error("Failed to flush %s: %s",
|
|
|
|
path, strerror(errno));
|
|
|
|
|
|
|
|
if (hunk_no < 0) {
|
2008-07-09 20:58:57 +08:00
|
|
|
if (output)
|
2009-04-30 05:22:56 +08:00
|
|
|
unlink_or_warn(output);
|
2008-07-09 20:58:57 +08:00
|
|
|
return error("Could not parse conflict hunks in %s", path);
|
|
|
|
}
|
2009-12-26 06:34:53 +08:00
|
|
|
if (io.io.wrerror)
|
2008-12-05 08:35:48 +08:00
|
|
|
return -1;
|
2008-07-09 20:58:57 +08:00
|
|
|
return hunk_no;
|
|
|
|
}
|
|
|
|
|
2015-07-01 13:33:19 +08:00
|
|
|
/*
|
|
|
|
* Subclass of rerere_io that reads from an in-core buffer that is a
|
|
|
|
* strbuf
|
|
|
|
*/
|
2009-12-26 07:51:32 +08:00
|
|
|
struct rerere_io_mem {
|
|
|
|
struct rerere_io io;
|
|
|
|
struct strbuf input;
|
|
|
|
};
|
|
|
|
|
2015-07-01 13:33:19 +08:00
|
|
|
/*
|
|
|
|
* ... and its getline() method implementation
|
|
|
|
*/
|
2009-12-26 07:51:32 +08:00
|
|
|
static int rerere_mem_getline(struct strbuf *sb, struct rerere_io *io_)
|
|
|
|
{
|
|
|
|
struct rerere_io_mem *io = (struct rerere_io_mem *)io_;
|
|
|
|
char *ep;
|
|
|
|
size_t len;
|
|
|
|
|
|
|
|
strbuf_release(sb);
|
|
|
|
if (!io->input.len)
|
|
|
|
return -1;
|
2013-04-02 05:36:36 +08:00
|
|
|
ep = memchr(io->input.buf, '\n', io->input.len);
|
|
|
|
if (!ep)
|
|
|
|
ep = io->input.buf + io->input.len;
|
|
|
|
else if (*ep == '\n')
|
2009-12-26 07:51:32 +08:00
|
|
|
ep++;
|
|
|
|
len = ep - io->input.buf;
|
|
|
|
strbuf_add(sb, io->input.buf, len);
|
|
|
|
strbuf_remove(&io->input, 0, len);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int handle_cache(const char *path, unsigned char *sha1, const char *output)
|
|
|
|
{
|
2013-04-05 02:41:43 +08:00
|
|
|
mmfile_t mmfile[3] = {{NULL}};
|
2009-12-26 07:51:32 +08:00
|
|
|
mmbuffer_t result = {NULL, 0};
|
Convert "struct cache_entry *" to "const ..." wherever possible
I attempted to make index_state->cache[] a "const struct cache_entry **"
to find out how existing entries in index are modified and where. The
question I have is what do we do if we really need to keep track of on-disk
changes in the index. The result is
- diff-lib.c: setting CE_UPTODATE
- name-hash.c: setting CE_HASHED
- preload-index.c, read-cache.c, unpack-trees.c and
builtin/update-index: obvious
- entry.c: write_entry() may refresh the checked out entry via
fill_stat_cache_info(). This causes "non-const struct cache_entry
*" in builtin/apply.c, builtin/checkout-index.c and
builtin/checkout.c
- builtin/ls-files.c: --with-tree changes stagemask and may set
CE_UPDATE
Of these, write_entry() and its call sites are probably most
interesting because it modifies on-disk info. But this is stat info
and can be retrieved via refresh, at least for porcelain
commands. Other just uses ce_flags for local purposes.
So, keeping track of "dirty" entries is just a matter of setting a
flag in index modification functions exposed by read-cache.c. Except
unpack-trees, the rest of the code base does not do anything funny
behind read-cache's back.
The actual patch is less valueable than the summary above. But if
anyone wants to re-identify the above sites. Applying this patch, then
this:
diff --git a/cache.h b/cache.h
index 430d021..1692891 100644
--- a/cache.h
+++ b/cache.h
@@ -267,7 +267,7 @@ static inline unsigned int canon_mode(unsigned int mode)
#define cache_entry_size(len) (offsetof(struct cache_entry,name) + (len) + 1)
struct index_state {
- struct cache_entry **cache;
+ const struct cache_entry **cache;
unsigned int version;
unsigned int cache_nr, cache_alloc, cache_changed;
struct string_list *resolve_undo;
will help quickly identify them without bogus warnings.
Signed-off-by: Nguyễn Thái Ngọc Duy <pclouds@gmail.com>
Signed-off-by: Junio C Hamano <gitster@pobox.com>
2013-07-09 23:29:00 +08:00
|
|
|
const struct cache_entry *ce;
|
2009-12-26 07:51:32 +08:00
|
|
|
int pos, len, i, hunk_no;
|
|
|
|
struct rerere_io_mem io;
|
2010-01-17 15:28:46 +08:00
|
|
|
int marker_size = ll_merge_marker_size(path);
|
2009-12-26 07:51:32 +08:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Reproduce the conflicted merge in-core
|
|
|
|
*/
|
|
|
|
len = strlen(path);
|
|
|
|
pos = cache_name_pos(path, len);
|
|
|
|
if (0 <= pos)
|
2008-12-05 08:35:48 +08:00
|
|
|
return -1;
|
2009-12-26 07:51:32 +08:00
|
|
|
pos = -pos - 1;
|
|
|
|
|
2015-07-01 07:10:10 +08:00
|
|
|
while (pos < active_nr) {
|
2009-12-26 07:51:32 +08:00
|
|
|
enum object_type type;
|
|
|
|
unsigned long size;
|
|
|
|
|
|
|
|
ce = active_cache[pos++];
|
2013-04-05 02:41:43 +08:00
|
|
|
if (ce_namelen(ce) != len || memcmp(ce->name, path, len))
|
2015-07-01 07:10:10 +08:00
|
|
|
break;
|
|
|
|
i = ce_stage(ce) - 1;
|
2015-07-25 06:10:52 +08:00
|
|
|
if (!mmfile[i].ptr) {
|
|
|
|
mmfile[i].ptr = read_sha1_file(ce->sha1, &type, &size);
|
|
|
|
mmfile[i].size = size;
|
|
|
|
}
|
2009-12-26 07:51:32 +08:00
|
|
|
}
|
2015-07-01 07:10:10 +08:00
|
|
|
for (i = 0; i < 3; i++)
|
2009-12-26 07:51:32 +08:00
|
|
|
if (!mmfile[i].ptr && !mmfile[i].size)
|
|
|
|
mmfile[i].ptr = xstrdup("");
|
2015-07-01 07:10:10 +08:00
|
|
|
|
2010-08-05 19:24:58 +08:00
|
|
|
/*
|
|
|
|
* NEEDSWORK: handle conflicts from merges with
|
|
|
|
* merge.renormalize set, too
|
|
|
|
*/
|
2010-03-21 08:38:58 +08:00
|
|
|
ll_merge(&result, path, &mmfile[0], NULL,
|
2009-12-26 07:51:32 +08:00
|
|
|
&mmfile[1], "ours",
|
2010-08-26 13:49:53 +08:00
|
|
|
&mmfile[2], "theirs", NULL);
|
2009-12-26 07:51:32 +08:00
|
|
|
for (i = 0; i < 3; i++)
|
|
|
|
free(mmfile[i].ptr);
|
|
|
|
|
2010-01-28 22:52:16 +08:00
|
|
|
memset(&io, 0, sizeof(io));
|
2009-12-26 07:51:32 +08:00
|
|
|
io.io.getline = rerere_mem_getline;
|
|
|
|
if (output)
|
|
|
|
io.io.output = fopen(output, "w");
|
|
|
|
else
|
|
|
|
io.io.output = NULL;
|
|
|
|
strbuf_init(&io.input, 0);
|
|
|
|
strbuf_attach(&io.input, result.ptr, result.size, result.size);
|
|
|
|
|
2015-07-01 13:42:34 +08:00
|
|
|
/*
|
|
|
|
* Grab the conflict ID and optionally write the original
|
|
|
|
* contents with conflict markers out.
|
|
|
|
*/
|
2010-01-17 15:06:45 +08:00
|
|
|
hunk_no = handle_path(sha1, (struct rerere_io *)&io, marker_size);
|
2009-12-26 07:51:32 +08:00
|
|
|
strbuf_release(&io.input);
|
|
|
|
if (io.io.output)
|
|
|
|
fclose(io.io.output);
|
2008-07-09 20:58:57 +08:00
|
|
|
return hunk_no;
|
|
|
|
}
|
|
|
|
|
2015-07-01 13:36:35 +08:00
|
|
|
/*
|
|
|
|
* Look at a cache entry at "i" and see if it is not conflicting,
|
|
|
|
* conflicting and we are willing to handle, or conflicting and
|
|
|
|
* we are unable to handle, and return the determination in *type.
|
|
|
|
* Return the cache index to be looked at next, by skipping the
|
|
|
|
* stages we have already looked at in this invocation of this
|
|
|
|
* function.
|
|
|
|
*/
|
2011-02-16 18:47:44 +08:00
|
|
|
static int check_one_conflict(int i, int *type)
|
2008-07-09 20:58:57 +08:00
|
|
|
{
|
Convert "struct cache_entry *" to "const ..." wherever possible
I attempted to make index_state->cache[] a "const struct cache_entry **"
to find out how existing entries in index are modified and where. The
question I have is what do we do if we really need to keep track of on-disk
changes in the index. The result is
- diff-lib.c: setting CE_UPTODATE
- name-hash.c: setting CE_HASHED
- preload-index.c, read-cache.c, unpack-trees.c and
builtin/update-index: obvious
- entry.c: write_entry() may refresh the checked out entry via
fill_stat_cache_info(). This causes "non-const struct cache_entry
*" in builtin/apply.c, builtin/checkout-index.c and
builtin/checkout.c
- builtin/ls-files.c: --with-tree changes stagemask and may set
CE_UPDATE
Of these, write_entry() and its call sites are probably most
interesting because it modifies on-disk info. But this is stat info
and can be retrieved via refresh, at least for porcelain
commands. Other just uses ce_flags for local purposes.
So, keeping track of "dirty" entries is just a matter of setting a
flag in index modification functions exposed by read-cache.c. Except
unpack-trees, the rest of the code base does not do anything funny
behind read-cache's back.
The actual patch is less valueable than the summary above. But if
anyone wants to re-identify the above sites. Applying this patch, then
this:
diff --git a/cache.h b/cache.h
index 430d021..1692891 100644
--- a/cache.h
+++ b/cache.h
@@ -267,7 +267,7 @@ static inline unsigned int canon_mode(unsigned int mode)
#define cache_entry_size(len) (offsetof(struct cache_entry,name) + (len) + 1)
struct index_state {
- struct cache_entry **cache;
+ const struct cache_entry **cache;
unsigned int version;
unsigned int cache_nr, cache_alloc, cache_changed;
struct string_list *resolve_undo;
will help quickly identify them without bogus warnings.
Signed-off-by: Nguyễn Thái Ngọc Duy <pclouds@gmail.com>
Signed-off-by: Junio C Hamano <gitster@pobox.com>
2013-07-09 23:29:00 +08:00
|
|
|
const struct cache_entry *e = active_cache[i];
|
2011-02-16 18:47:44 +08:00
|
|
|
|
|
|
|
if (!ce_stage(e)) {
|
|
|
|
*type = RESOLVED;
|
|
|
|
return i + 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
*type = PUNTED;
|
2015-07-25 06:08:03 +08:00
|
|
|
while (ce_stage(active_cache[i]) == 1)
|
2015-06-29 05:35:13 +08:00
|
|
|
i++;
|
2011-02-16 18:47:44 +08:00
|
|
|
|
|
|
|
/* Only handle regular files with both stages #2 and #3 */
|
|
|
|
if (i + 1 < active_nr) {
|
Convert "struct cache_entry *" to "const ..." wherever possible
I attempted to make index_state->cache[] a "const struct cache_entry **"
to find out how existing entries in index are modified and where. The
question I have is what do we do if we really need to keep track of on-disk
changes in the index. The result is
- diff-lib.c: setting CE_UPTODATE
- name-hash.c: setting CE_HASHED
- preload-index.c, read-cache.c, unpack-trees.c and
builtin/update-index: obvious
- entry.c: write_entry() may refresh the checked out entry via
fill_stat_cache_info(). This causes "non-const struct cache_entry
*" in builtin/apply.c, builtin/checkout-index.c and
builtin/checkout.c
- builtin/ls-files.c: --with-tree changes stagemask and may set
CE_UPDATE
Of these, write_entry() and its call sites are probably most
interesting because it modifies on-disk info. But this is stat info
and can be retrieved via refresh, at least for porcelain
commands. Other just uses ce_flags for local purposes.
So, keeping track of "dirty" entries is just a matter of setting a
flag in index modification functions exposed by read-cache.c. Except
unpack-trees, the rest of the code base does not do anything funny
behind read-cache's back.
The actual patch is less valueable than the summary above. But if
anyone wants to re-identify the above sites. Applying this patch, then
this:
diff --git a/cache.h b/cache.h
index 430d021..1692891 100644
--- a/cache.h
+++ b/cache.h
@@ -267,7 +267,7 @@ static inline unsigned int canon_mode(unsigned int mode)
#define cache_entry_size(len) (offsetof(struct cache_entry,name) + (len) + 1)
struct index_state {
- struct cache_entry **cache;
+ const struct cache_entry **cache;
unsigned int version;
unsigned int cache_nr, cache_alloc, cache_changed;
struct string_list *resolve_undo;
will help quickly identify them without bogus warnings.
Signed-off-by: Nguyễn Thái Ngọc Duy <pclouds@gmail.com>
Signed-off-by: Junio C Hamano <gitster@pobox.com>
2013-07-09 23:29:00 +08:00
|
|
|
const struct cache_entry *e2 = active_cache[i];
|
|
|
|
const struct cache_entry *e3 = active_cache[i + 1];
|
2008-07-09 20:58:57 +08:00
|
|
|
if (ce_stage(e2) == 2 &&
|
|
|
|
ce_stage(e3) == 3 &&
|
2011-02-16 18:47:44 +08:00
|
|
|
ce_same_name(e, e3) &&
|
2008-07-09 20:58:57 +08:00
|
|
|
S_ISREG(e2->ce_mode) &&
|
2011-02-16 18:47:44 +08:00
|
|
|
S_ISREG(e3->ce_mode))
|
|
|
|
*type = THREE_STAGED;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Skip the entries with the same name */
|
|
|
|
while (i < active_nr && ce_same_name(e, active_cache[i]))
|
|
|
|
i++;
|
|
|
|
return i;
|
|
|
|
}
|
|
|
|
|
2015-07-01 13:36:35 +08:00
|
|
|
/*
|
|
|
|
* Scan the index and find paths that have conflicts that rerere can
|
|
|
|
* handle, i.e. the ones that has both stages #2 and #3.
|
|
|
|
*
|
|
|
|
* NEEDSWORK: we do not record or replay a previous "resolve by
|
|
|
|
* deletion" for a delete-modify conflict, as that is inherently risky
|
|
|
|
* without knowing what modification is being discarded. The only
|
|
|
|
* safe case, i.e. both side doing the deletion and modification that
|
|
|
|
* are identical to the previous round, might want to be handled,
|
|
|
|
* though.
|
|
|
|
*/
|
2011-02-16 18:47:44 +08:00
|
|
|
static int find_conflict(struct string_list *conflict)
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
if (read_cache() < 0)
|
|
|
|
return error("Could not read index");
|
|
|
|
|
|
|
|
for (i = 0; i < active_nr;) {
|
|
|
|
int conflict_type;
|
Convert "struct cache_entry *" to "const ..." wherever possible
I attempted to make index_state->cache[] a "const struct cache_entry **"
to find out how existing entries in index are modified and where. The
question I have is what do we do if we really need to keep track of on-disk
changes in the index. The result is
- diff-lib.c: setting CE_UPTODATE
- name-hash.c: setting CE_HASHED
- preload-index.c, read-cache.c, unpack-trees.c and
builtin/update-index: obvious
- entry.c: write_entry() may refresh the checked out entry via
fill_stat_cache_info(). This causes "non-const struct cache_entry
*" in builtin/apply.c, builtin/checkout-index.c and
builtin/checkout.c
- builtin/ls-files.c: --with-tree changes stagemask and may set
CE_UPDATE
Of these, write_entry() and its call sites are probably most
interesting because it modifies on-disk info. But this is stat info
and can be retrieved via refresh, at least for porcelain
commands. Other just uses ce_flags for local purposes.
So, keeping track of "dirty" entries is just a matter of setting a
flag in index modification functions exposed by read-cache.c. Except
unpack-trees, the rest of the code base does not do anything funny
behind read-cache's back.
The actual patch is less valueable than the summary above. But if
anyone wants to re-identify the above sites. Applying this patch, then
this:
diff --git a/cache.h b/cache.h
index 430d021..1692891 100644
--- a/cache.h
+++ b/cache.h
@@ -267,7 +267,7 @@ static inline unsigned int canon_mode(unsigned int mode)
#define cache_entry_size(len) (offsetof(struct cache_entry,name) + (len) + 1)
struct index_state {
- struct cache_entry **cache;
+ const struct cache_entry **cache;
unsigned int version;
unsigned int cache_nr, cache_alloc, cache_changed;
struct string_list *resolve_undo;
will help quickly identify them without bogus warnings.
Signed-off-by: Nguyễn Thái Ngọc Duy <pclouds@gmail.com>
Signed-off-by: Junio C Hamano <gitster@pobox.com>
2013-07-09 23:29:00 +08:00
|
|
|
const struct cache_entry *e = active_cache[i];
|
2011-02-16 18:47:44 +08:00
|
|
|
i = check_one_conflict(i, &conflict_type);
|
|
|
|
if (conflict_type == THREE_STAGED)
|
|
|
|
string_list_insert(conflict, (const char *)e->name);
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2015-07-01 13:36:35 +08:00
|
|
|
/*
|
|
|
|
* The merge_rr list is meant to hold outstanding conflicted paths
|
|
|
|
* that rerere could handle. Abuse the list by adding other types of
|
|
|
|
* entries to allow the caller to show "rerere remaining".
|
|
|
|
*
|
|
|
|
* - Conflicted paths that rerere does not handle are added
|
|
|
|
* - Conflicted paths that have been resolved are marked as such
|
|
|
|
* by storing RERERE_RESOLVED to .util field (where conflict ID
|
|
|
|
* is expected to be stored).
|
|
|
|
*
|
|
|
|
* Do *not* write MERGE_RR file out after calling this function.
|
|
|
|
*
|
|
|
|
* NEEDSWORK: we may want to fix the caller that implements "rerere
|
|
|
|
* remaining" to do this without abusing merge_rr.
|
|
|
|
*/
|
2011-02-16 18:47:44 +08:00
|
|
|
int rerere_remaining(struct string_list *merge_rr)
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
if (read_cache() < 0)
|
|
|
|
return error("Could not read index");
|
|
|
|
|
|
|
|
for (i = 0; i < active_nr;) {
|
|
|
|
int conflict_type;
|
Convert "struct cache_entry *" to "const ..." wherever possible
I attempted to make index_state->cache[] a "const struct cache_entry **"
to find out how existing entries in index are modified and where. The
question I have is what do we do if we really need to keep track of on-disk
changes in the index. The result is
- diff-lib.c: setting CE_UPTODATE
- name-hash.c: setting CE_HASHED
- preload-index.c, read-cache.c, unpack-trees.c and
builtin/update-index: obvious
- entry.c: write_entry() may refresh the checked out entry via
fill_stat_cache_info(). This causes "non-const struct cache_entry
*" in builtin/apply.c, builtin/checkout-index.c and
builtin/checkout.c
- builtin/ls-files.c: --with-tree changes stagemask and may set
CE_UPDATE
Of these, write_entry() and its call sites are probably most
interesting because it modifies on-disk info. But this is stat info
and can be retrieved via refresh, at least for porcelain
commands. Other just uses ce_flags for local purposes.
So, keeping track of "dirty" entries is just a matter of setting a
flag in index modification functions exposed by read-cache.c. Except
unpack-trees, the rest of the code base does not do anything funny
behind read-cache's back.
The actual patch is less valueable than the summary above. But if
anyone wants to re-identify the above sites. Applying this patch, then
this:
diff --git a/cache.h b/cache.h
index 430d021..1692891 100644
--- a/cache.h
+++ b/cache.h
@@ -267,7 +267,7 @@ static inline unsigned int canon_mode(unsigned int mode)
#define cache_entry_size(len) (offsetof(struct cache_entry,name) + (len) + 1)
struct index_state {
- struct cache_entry **cache;
+ const struct cache_entry **cache;
unsigned int version;
unsigned int cache_nr, cache_alloc, cache_changed;
struct string_list *resolve_undo;
will help quickly identify them without bogus warnings.
Signed-off-by: Nguyễn Thái Ngọc Duy <pclouds@gmail.com>
Signed-off-by: Junio C Hamano <gitster@pobox.com>
2013-07-09 23:29:00 +08:00
|
|
|
const struct cache_entry *e = active_cache[i];
|
2011-02-16 18:47:44 +08:00
|
|
|
i = check_one_conflict(i, &conflict_type);
|
|
|
|
if (conflict_type == PUNTED)
|
|
|
|
string_list_insert(merge_rr, (const char *)e->name);
|
|
|
|
else if (conflict_type == RESOLVED) {
|
|
|
|
struct string_list_item *it;
|
|
|
|
it = string_list_lookup(merge_rr, (const char *)e->name);
|
|
|
|
if (it != NULL) {
|
2015-07-05 08:38:34 +08:00
|
|
|
free_rerere_id(it);
|
2011-02-16 18:47:44 +08:00
|
|
|
it->util = RERERE_RESOLVED;
|
|
|
|
}
|
2008-07-09 20:58:57 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2015-07-01 13:40:35 +08:00
|
|
|
/*
|
2015-07-07 05:45:55 +08:00
|
|
|
* Find the conflict identified by "id"; the change between its
|
2015-07-01 13:40:35 +08:00
|
|
|
* "preimage" (i.e. a previous contents with conflict markers) and its
|
|
|
|
* "postimage" (i.e. the corresponding contents with conflicts
|
|
|
|
* resolved) may apply cleanly to the contents stored in "path", i.e.
|
|
|
|
* the conflict this time around.
|
|
|
|
*
|
|
|
|
* Returns 0 for successful replay of recorded resolution, or non-zero
|
|
|
|
* for failure.
|
|
|
|
*/
|
2015-07-05 08:38:34 +08:00
|
|
|
static int merge(const struct rerere_id *id, const char *path)
|
2008-07-09 20:58:57 +08:00
|
|
|
{
|
2015-07-07 06:32:53 +08:00
|
|
|
FILE *f;
|
2008-07-09 20:58:57 +08:00
|
|
|
int ret;
|
2010-02-24 04:11:53 +08:00
|
|
|
mmfile_t cur = {NULL, 0}, base = {NULL, 0}, other = {NULL, 0};
|
2008-07-09 20:58:57 +08:00
|
|
|
mmbuffer_t result = {NULL, 0};
|
|
|
|
|
2015-07-01 13:40:35 +08:00
|
|
|
/*
|
|
|
|
* Normalize the conflicts in path and write it out to
|
|
|
|
* "thisimage" temporary file.
|
|
|
|
*/
|
2015-07-07 06:32:53 +08:00
|
|
|
if (handle_file(path, NULL, rerere_path(id, "thisimage")) < 0) {
|
|
|
|
ret = 1;
|
|
|
|
goto out;
|
|
|
|
}
|
2008-07-09 20:58:57 +08:00
|
|
|
|
2015-07-07 05:45:55 +08:00
|
|
|
if (read_mmfile(&cur, rerere_path(id, "thisimage")) ||
|
|
|
|
read_mmfile(&base, rerere_path(id, "preimage")) ||
|
|
|
|
read_mmfile(&other, rerere_path(id, "postimage"))) {
|
2010-02-24 04:11:53 +08:00
|
|
|
ret = 1;
|
|
|
|
goto out;
|
|
|
|
}
|
2015-07-01 13:40:35 +08:00
|
|
|
|
|
|
|
/*
|
|
|
|
* A three-way merge. Note that this honors user-customizable
|
|
|
|
* low-level merge driver settings.
|
|
|
|
*/
|
2011-04-03 15:06:54 +08:00
|
|
|
ret = ll_merge(&result, path, &base, NULL, &cur, "", &other, "", NULL);
|
2015-07-07 06:32:53 +08:00
|
|
|
if (ret)
|
|
|
|
goto out;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* A successful replay of recorded resolution.
|
|
|
|
* Mark that "postimage" was used to help gc.
|
|
|
|
*/
|
|
|
|
if (utime(rerere_path(id, "postimage"), NULL) < 0)
|
|
|
|
warning("failed utime() on %s: %s",
|
|
|
|
rerere_path(id, "postimage"),
|
|
|
|
strerror(errno));
|
|
|
|
|
|
|
|
/* Update "path" with the resolution */
|
|
|
|
f = fopen(path, "w");
|
|
|
|
if (!f)
|
|
|
|
return error("Could not open %s: %s", path,
|
|
|
|
strerror(errno));
|
|
|
|
if (fwrite(result.ptr, result.size, 1, f) != 1)
|
|
|
|
error("Could not write %s: %s", path, strerror(errno));
|
|
|
|
if (fclose(f))
|
|
|
|
return error("Writing %s failed: %s", path,
|
|
|
|
strerror(errno));
|
2008-07-09 20:58:57 +08:00
|
|
|
|
2010-02-24 04:11:53 +08:00
|
|
|
out:
|
2008-07-09 20:58:57 +08:00
|
|
|
free(cur.ptr);
|
|
|
|
free(base.ptr);
|
|
|
|
free(other.ptr);
|
|
|
|
free(result.ptr);
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
static struct lock_file index_lock;
|
|
|
|
|
2014-12-03 12:20:49 +08:00
|
|
|
static void update_paths(struct string_list *update)
|
2008-07-09 20:58:57 +08:00
|
|
|
{
|
|
|
|
int i;
|
|
|
|
|
2014-12-03 12:20:49 +08:00
|
|
|
hold_locked_index(&index_lock, 1);
|
2008-07-09 20:58:57 +08:00
|
|
|
|
|
|
|
for (i = 0; i < update->nr; i++) {
|
2008-07-22 02:03:49 +08:00
|
|
|
struct string_list_item *item = &update->items[i];
|
2014-12-03 12:20:49 +08:00
|
|
|
if (add_file_to_cache(item->string, 0))
|
|
|
|
exit(128);
|
2015-06-29 12:13:24 +08:00
|
|
|
fprintf(stderr, "Staged '%s' using previous resolution.\n",
|
|
|
|
item->string);
|
2008-07-09 20:58:57 +08:00
|
|
|
}
|
|
|
|
|
2014-12-03 12:20:49 +08:00
|
|
|
if (active_cache_changed) {
|
2014-06-13 20:19:23 +08:00
|
|
|
if (write_locked_index(&the_index, &index_lock, COMMIT_LOCK))
|
2008-07-09 20:58:57 +08:00
|
|
|
die("Unable to write new index file");
|
2014-12-03 12:20:49 +08:00
|
|
|
} else
|
2008-07-09 20:58:57 +08:00
|
|
|
rollback_lock_file(&index_lock);
|
|
|
|
}
|
|
|
|
|
2015-07-01 10:36:24 +08:00
|
|
|
/*
|
|
|
|
* The path indicated by rr_item may still have conflict for which we
|
|
|
|
* have a recorded resolution, in which case replay it and optionally
|
|
|
|
* update it. Or it may have been resolved by the user and we may
|
|
|
|
* only have the preimage for that conflict, in which case the result
|
|
|
|
* needs to be recorded as a resolution in a postimage file.
|
|
|
|
*/
|
|
|
|
static void do_rerere_one_path(struct string_list_item *rr_item,
|
|
|
|
struct string_list *update)
|
|
|
|
{
|
|
|
|
const char *path = rr_item->string;
|
2015-07-05 08:38:34 +08:00
|
|
|
const struct rerere_id *id = rr_item->util;
|
2015-07-01 10:36:24 +08:00
|
|
|
|
|
|
|
/* Is there a recorded resolution we could attempt to apply? */
|
2015-07-07 05:45:55 +08:00
|
|
|
if (has_rerere_resolution(id)) {
|
|
|
|
if (merge(id, path))
|
2015-07-01 10:36:24 +08:00
|
|
|
return; /* failed to replay */
|
|
|
|
|
|
|
|
if (rerere_autoupdate)
|
|
|
|
string_list_insert(update, path);
|
|
|
|
else
|
|
|
|
fprintf(stderr,
|
|
|
|
"Resolved '%s' using previous resolution.\n",
|
|
|
|
path);
|
2015-07-07 05:18:09 +08:00
|
|
|
} else if (!handle_file(path, NULL, NULL)) {
|
|
|
|
/* The user has resolved it. */
|
2015-07-07 05:45:55 +08:00
|
|
|
copy_file(rerere_path(id, "postimage"), path, 0666);
|
2015-07-17 06:47:13 +08:00
|
|
|
id->collection->status |= RR_HAS_POSTIMAGE;
|
2015-07-07 05:18:09 +08:00
|
|
|
fprintf(stderr, "Recorded resolution for '%s'.\n", path);
|
|
|
|
} else {
|
|
|
|
return;
|
2015-07-01 10:36:24 +08:00
|
|
|
}
|
2015-07-05 08:38:34 +08:00
|
|
|
free_rerere_id(rr_item);
|
2015-07-01 10:36:24 +08:00
|
|
|
rr_item->util = NULL;
|
|
|
|
}
|
|
|
|
|
2008-07-22 02:03:49 +08:00
|
|
|
static int do_plain_rerere(struct string_list *rr, int fd)
|
2008-07-09 20:58:57 +08:00
|
|
|
{
|
2010-07-05 03:46:19 +08:00
|
|
|
struct string_list conflict = STRING_LIST_INIT_DUP;
|
|
|
|
struct string_list update = STRING_LIST_INIT_DUP;
|
2008-07-09 20:58:57 +08:00
|
|
|
int i;
|
|
|
|
|
|
|
|
find_conflict(&conflict);
|
|
|
|
|
|
|
|
/*
|
2015-07-01 13:40:35 +08:00
|
|
|
* MERGE_RR records paths with conflicts immediately after
|
|
|
|
* merge failed. Some of the conflicted paths might have been
|
|
|
|
* hand resolved in the working tree since then, but the
|
|
|
|
* initial run would catch all and register their preimages.
|
2008-07-09 20:58:57 +08:00
|
|
|
*/
|
|
|
|
for (i = 0; i < conflict.nr; i++) {
|
2015-07-05 08:38:34 +08:00
|
|
|
struct rerere_id *id;
|
2015-07-05 08:17:38 +08:00
|
|
|
unsigned char sha1[20];
|
2008-07-22 02:03:49 +08:00
|
|
|
const char *path = conflict.items[i].string;
|
2015-07-05 08:38:34 +08:00
|
|
|
int ret;
|
2015-07-01 13:40:35 +08:00
|
|
|
|
2015-07-05 08:17:38 +08:00
|
|
|
if (string_list_has_string(rr, path))
|
|
|
|
continue;
|
2015-07-01 13:40:35 +08:00
|
|
|
|
2015-07-05 08:17:38 +08:00
|
|
|
/*
|
|
|
|
* Ask handle_file() to scan and assign a
|
|
|
|
* conflict ID. No need to write anything out
|
|
|
|
* yet.
|
|
|
|
*/
|
|
|
|
ret = handle_file(path, sha1, NULL);
|
|
|
|
if (ret < 1)
|
|
|
|
continue;
|
2015-07-05 08:38:34 +08:00
|
|
|
|
|
|
|
id = new_rerere_id(sha1);
|
2015-07-07 05:45:55 +08:00
|
|
|
string_list_insert(rr, path)->util = id;
|
2015-07-01 13:40:35 +08:00
|
|
|
|
2015-07-05 08:17:38 +08:00
|
|
|
/*
|
|
|
|
* If the directory does not exist, create
|
|
|
|
* it. mkdir_in_gitdir() will fail with
|
|
|
|
* EEXIST if there already is one.
|
|
|
|
*
|
|
|
|
* NEEDSWORK: make sure "gc" does not remove
|
|
|
|
* preimage without removing the directory.
|
|
|
|
*/
|
2015-07-05 08:38:34 +08:00
|
|
|
if (mkdir_in_gitdir(rerere_path(id, NULL)))
|
2015-07-05 08:17:38 +08:00
|
|
|
continue;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* We are the first to encounter this
|
|
|
|
* conflict. Ask handle_file() to write the
|
|
|
|
* normalized contents to the "preimage" file.
|
|
|
|
*/
|
2015-07-07 05:45:55 +08:00
|
|
|
handle_file(path, NULL, rerere_path(id, "preimage"));
|
2015-07-17 06:47:13 +08:00
|
|
|
id->collection->status |= RR_HAS_PREIMAGE;
|
2015-07-05 08:17:38 +08:00
|
|
|
fprintf(stderr, "Recorded preimage for '%s'\n", path);
|
2008-07-09 20:58:57 +08:00
|
|
|
}
|
|
|
|
|
2015-07-01 10:36:24 +08:00
|
|
|
for (i = 0; i < rr->nr; i++)
|
|
|
|
do_rerere_one_path(&rr->items[i], &update);
|
2008-07-09 20:58:57 +08:00
|
|
|
|
|
|
|
if (update.nr)
|
|
|
|
update_paths(&update);
|
|
|
|
|
|
|
|
return write_rr(rr, fd);
|
|
|
|
}
|
|
|
|
|
2014-08-08 00:21:21 +08:00
|
|
|
static void git_rerere_config(void)
|
2008-07-09 20:58:57 +08:00
|
|
|
{
|
2014-08-08 00:21:21 +08:00
|
|
|
git_config_get_bool("rerere.enabled", &rerere_enabled);
|
|
|
|
git_config_get_bool("rerere.autoupdate", &rerere_autoupdate);
|
|
|
|
git_config(git_default_config, NULL);
|
2008-07-09 20:58:57 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static int is_rerere_enabled(void)
|
|
|
|
{
|
|
|
|
const char *rr_cache;
|
|
|
|
int rr_cache_exists;
|
|
|
|
|
|
|
|
if (!rerere_enabled)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
rr_cache = git_path("rr-cache");
|
2008-09-09 16:27:07 +08:00
|
|
|
rr_cache_exists = is_directory(rr_cache);
|
2008-07-09 20:58:57 +08:00
|
|
|
if (rerere_enabled < 0)
|
|
|
|
return rr_cache_exists;
|
|
|
|
|
2011-03-11 08:02:50 +08:00
|
|
|
if (!rr_cache_exists && mkdir_in_gitdir(rr_cache))
|
2008-07-09 20:58:57 +08:00
|
|
|
die("Could not create directory %s", rr_cache);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2009-12-04 16:20:48 +08:00
|
|
|
int setup_rerere(struct string_list *merge_rr, int flags)
|
2008-07-09 20:58:57 +08:00
|
|
|
{
|
|
|
|
int fd;
|
|
|
|
|
2014-08-08 00:21:21 +08:00
|
|
|
git_rerere_config();
|
2008-07-09 20:58:57 +08:00
|
|
|
if (!is_rerere_enabled())
|
|
|
|
return -1;
|
|
|
|
|
2009-12-04 16:20:48 +08:00
|
|
|
if (flags & (RERERE_AUTOUPDATE|RERERE_NOAUTOUPDATE))
|
|
|
|
rerere_autoupdate = !!(flags & RERERE_AUTOUPDATE);
|
2008-10-27 18:22:09 +08:00
|
|
|
merge_rr_path = git_pathdup("MERGE_RR");
|
2008-10-18 06:44:39 +08:00
|
|
|
fd = hold_lock_file_for_update(&write_lock, merge_rr_path,
|
|
|
|
LOCK_DIE_ON_ERROR);
|
2008-07-09 20:58:57 +08:00
|
|
|
read_rr(merge_rr);
|
|
|
|
return fd;
|
|
|
|
}
|
|
|
|
|
2015-07-01 13:40:35 +08:00
|
|
|
/*
|
|
|
|
* The main entry point that is called internally from codepaths that
|
|
|
|
* perform mergy operations, possibly leaving conflicted index entries
|
|
|
|
* and working tree files.
|
|
|
|
*/
|
2009-12-04 16:20:48 +08:00
|
|
|
int rerere(int flags)
|
2008-07-09 20:58:57 +08:00
|
|
|
{
|
2010-07-05 03:46:19 +08:00
|
|
|
struct string_list merge_rr = STRING_LIST_INIT_DUP;
|
rerere: split conflict ID further
The plan is to keep assigning the backward compatible conflict ID
based on the hash of the (normalized) text of conflicts, keep using
that conflict ID as the directory name under $GIT_DIR/rr-cache/, but
allow each conflicted path to use a separate "variant" to record
resolutions, i.e. having more than one <preimage,postimage> pairs
under $GIT_DIR/rr-cache/$ID/ directory. As the first step in that
direction, separate the shared "conflict ID" out of the rerere_id
structure.
The plan is to keep information per $ID in rerere_dir, that can be
shared among rerere_id that is per conflicted path.
When we are done with rerere(), which can be directly called from
other programs like "git apply", "git commit" and "git merge", the
shared rerere_dir structures can be freed entirely, so they are not
reference-counted and they are not freed when we release rerere_id's
that reference them.
Signed-off-by: Junio C Hamano <gitster@pobox.com>
2015-07-17 05:50:05 +08:00
|
|
|
int fd, status;
|
2008-07-09 20:58:57 +08:00
|
|
|
|
2009-12-04 16:20:48 +08:00
|
|
|
fd = setup_rerere(&merge_rr, flags);
|
2008-07-09 20:58:57 +08:00
|
|
|
if (fd < 0)
|
|
|
|
return 0;
|
rerere: split conflict ID further
The plan is to keep assigning the backward compatible conflict ID
based on the hash of the (normalized) text of conflicts, keep using
that conflict ID as the directory name under $GIT_DIR/rr-cache/, but
allow each conflicted path to use a separate "variant" to record
resolutions, i.e. having more than one <preimage,postimage> pairs
under $GIT_DIR/rr-cache/$ID/ directory. As the first step in that
direction, separate the shared "conflict ID" out of the rerere_id
structure.
The plan is to keep information per $ID in rerere_dir, that can be
shared among rerere_id that is per conflicted path.
When we are done with rerere(), which can be directly called from
other programs like "git apply", "git commit" and "git merge", the
shared rerere_dir structures can be freed entirely, so they are not
reference-counted and they are not freed when we release rerere_id's
that reference them.
Signed-off-by: Junio C Hamano <gitster@pobox.com>
2015-07-17 05:50:05 +08:00
|
|
|
status = do_plain_rerere(&merge_rr, fd);
|
|
|
|
free_rerere_dirs();
|
|
|
|
return status;
|
2008-07-09 20:58:57 +08:00
|
|
|
}
|
2009-12-26 07:51:32 +08:00
|
|
|
|
|
|
|
static int rerere_forget_one_path(const char *path, struct string_list *rr)
|
|
|
|
{
|
|
|
|
const char *filename;
|
2015-07-05 08:38:34 +08:00
|
|
|
struct rerere_id *id;
|
2009-12-26 07:51:32 +08:00
|
|
|
unsigned char sha1[20];
|
|
|
|
int ret;
|
2015-07-01 04:03:36 +08:00
|
|
|
struct string_list_item *item;
|
2009-12-26 07:51:32 +08:00
|
|
|
|
2015-07-01 13:42:34 +08:00
|
|
|
/*
|
|
|
|
* Recreate the original conflict from the stages in the
|
|
|
|
* index and compute the conflict ID
|
|
|
|
*/
|
2009-12-26 07:51:32 +08:00
|
|
|
ret = handle_cache(path, sha1, NULL);
|
|
|
|
if (ret < 1)
|
|
|
|
return error("Could not parse conflict hunks in '%s'", path);
|
2015-07-01 13:42:34 +08:00
|
|
|
|
|
|
|
/* Nuke the recorded resolution for the conflict */
|
2015-07-05 08:38:34 +08:00
|
|
|
id = new_rerere_id(sha1);
|
2015-07-07 05:45:55 +08:00
|
|
|
filename = rerere_path(id, "postimage");
|
2009-12-26 07:51:32 +08:00
|
|
|
if (unlink(filename))
|
|
|
|
return (errno == ENOENT
|
|
|
|
? error("no remembered resolution for %s", path)
|
|
|
|
: error("cannot unlink %s: %s", filename, strerror(errno)));
|
|
|
|
|
2015-07-01 13:42:34 +08:00
|
|
|
/*
|
|
|
|
* Update the preimage so that the user can resolve the
|
|
|
|
* conflict in the working tree, run us again to record
|
|
|
|
* the postimage.
|
|
|
|
*/
|
2015-07-07 05:45:55 +08:00
|
|
|
handle_cache(path, sha1, rerere_path(id, "preimage"));
|
2009-12-26 07:51:32 +08:00
|
|
|
fprintf(stderr, "Updated preimage for '%s'\n", path);
|
|
|
|
|
2015-07-01 13:42:34 +08:00
|
|
|
/*
|
|
|
|
* And remember that we can record resolution for this
|
|
|
|
* conflict when the user is done.
|
|
|
|
*/
|
2015-07-01 04:03:36 +08:00
|
|
|
item = string_list_insert(rr, path);
|
2015-07-05 08:38:34 +08:00
|
|
|
free_rerere_id(item);
|
2015-07-07 05:45:55 +08:00
|
|
|
item->util = id;
|
2009-12-26 07:51:32 +08:00
|
|
|
fprintf(stderr, "Forgot resolution for %s\n", path);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2013-07-14 16:35:40 +08:00
|
|
|
int rerere_forget(struct pathspec *pathspec)
|
2009-12-26 07:51:32 +08:00
|
|
|
{
|
|
|
|
int i, fd;
|
2010-07-05 03:46:19 +08:00
|
|
|
struct string_list conflict = STRING_LIST_INIT_DUP;
|
|
|
|
struct string_list merge_rr = STRING_LIST_INIT_DUP;
|
2009-12-26 07:51:32 +08:00
|
|
|
|
|
|
|
if (read_cache() < 0)
|
|
|
|
return error("Could not read index");
|
|
|
|
|
2010-01-21 06:44:31 +08:00
|
|
|
fd = setup_rerere(&merge_rr, RERERE_NOAUTOUPDATE);
|
2009-12-26 07:51:32 +08:00
|
|
|
|
2015-07-01 13:42:34 +08:00
|
|
|
/*
|
|
|
|
* The paths may have been resolved (incorrectly);
|
|
|
|
* recover the original conflicted state and then
|
|
|
|
* find the conflicted paths.
|
|
|
|
*/
|
2009-12-26 07:51:32 +08:00
|
|
|
unmerge_cache(pathspec);
|
|
|
|
find_conflict(&conflict);
|
|
|
|
for (i = 0; i < conflict.nr; i++) {
|
|
|
|
struct string_list_item *it = &conflict.items[i];
|
2014-01-24 21:40:30 +08:00
|
|
|
if (!match_pathspec(pathspec, it->string,
|
2014-01-24 21:40:33 +08:00
|
|
|
strlen(it->string), 0, NULL, 0))
|
2009-12-26 07:51:32 +08:00
|
|
|
continue;
|
|
|
|
rerere_forget_one_path(it->string, &merge_rr);
|
|
|
|
}
|
|
|
|
return write_rr(&merge_rr, fd);
|
|
|
|
}
|
2011-05-09 03:55:34 +08:00
|
|
|
|
2015-07-01 13:43:37 +08:00
|
|
|
/*
|
|
|
|
* Garbage collection support
|
|
|
|
*/
|
2015-07-05 08:38:34 +08:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Note that this is not reentrant but is used only one-at-a-time
|
|
|
|
* so it does not matter right now.
|
|
|
|
*/
|
|
|
|
static struct rerere_id *dirname_to_id(const char *name)
|
|
|
|
{
|
|
|
|
static struct rerere_id id;
|
rerere: split conflict ID further
The plan is to keep assigning the backward compatible conflict ID
based on the hash of the (normalized) text of conflicts, keep using
that conflict ID as the directory name under $GIT_DIR/rr-cache/, but
allow each conflicted path to use a separate "variant" to record
resolutions, i.e. having more than one <preimage,postimage> pairs
under $GIT_DIR/rr-cache/$ID/ directory. As the first step in that
direction, separate the shared "conflict ID" out of the rerere_id
structure.
The plan is to keep information per $ID in rerere_dir, that can be
shared among rerere_id that is per conflicted path.
When we are done with rerere(), which can be directly called from
other programs like "git apply", "git commit" and "git merge", the
shared rerere_dir structures can be freed entirely, so they are not
reference-counted and they are not freed when we release rerere_id's
that reference them.
Signed-off-by: Junio C Hamano <gitster@pobox.com>
2015-07-17 05:50:05 +08:00
|
|
|
id.collection = find_rerere_dir(name);
|
2015-07-05 08:38:34 +08:00
|
|
|
return &id;
|
|
|
|
}
|
|
|
|
|
|
|
|
static time_t rerere_created_at(const char *dir_name)
|
2011-05-09 03:55:34 +08:00
|
|
|
{
|
|
|
|
struct stat st;
|
2015-07-05 08:38:34 +08:00
|
|
|
struct rerere_id *id = dirname_to_id(dir_name);
|
|
|
|
|
2015-07-07 05:45:55 +08:00
|
|
|
return stat(rerere_path(id, "preimage"), &st) ? (time_t) 0 : st.st_mtime;
|
2011-05-09 03:55:34 +08:00
|
|
|
}
|
|
|
|
|
2015-07-05 08:38:34 +08:00
|
|
|
static time_t rerere_last_used_at(const char *dir_name)
|
2011-05-09 03:55:34 +08:00
|
|
|
{
|
|
|
|
struct stat st;
|
2015-07-05 08:38:34 +08:00
|
|
|
struct rerere_id *id = dirname_to_id(dir_name);
|
|
|
|
|
2015-07-07 05:45:55 +08:00
|
|
|
return stat(rerere_path(id, "postimage"), &st) ? (time_t) 0 : st.st_mtime;
|
2011-05-09 03:55:34 +08:00
|
|
|
}
|
|
|
|
|
2015-07-01 13:43:37 +08:00
|
|
|
/*
|
|
|
|
* Remove the recorded resolution for a given conflict ID
|
|
|
|
*/
|
2015-07-05 08:38:34 +08:00
|
|
|
static void unlink_rr_item(struct rerere_id *id)
|
2011-05-09 03:55:34 +08:00
|
|
|
{
|
2015-07-07 05:45:55 +08:00
|
|
|
unlink(rerere_path(id, "thisimage"));
|
|
|
|
unlink(rerere_path(id, "preimage"));
|
|
|
|
unlink(rerere_path(id, "postimage"));
|
2015-07-01 13:43:37 +08:00
|
|
|
/*
|
|
|
|
* NEEDSWORK: what if this rmdir() fails? Wouldn't we then
|
|
|
|
* assume that we already have preimage recorded in
|
|
|
|
* do_plain_rerere()?
|
|
|
|
*/
|
2015-07-05 08:38:34 +08:00
|
|
|
rmdir(rerere_path(id, NULL));
|
2011-05-09 03:55:34 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
void rerere_gc(struct string_list *rr)
|
|
|
|
{
|
|
|
|
struct string_list to_remove = STRING_LIST_INIT_DUP;
|
|
|
|
DIR *dir;
|
|
|
|
struct dirent *e;
|
|
|
|
int i, cutoff;
|
|
|
|
time_t now = time(NULL), then;
|
2014-08-08 00:21:21 +08:00
|
|
|
int cutoff_noresolve = 15;
|
|
|
|
int cutoff_resolve = 60;
|
2011-05-09 03:55:34 +08:00
|
|
|
|
2014-08-08 00:21:21 +08:00
|
|
|
git_config_get_int("gc.rerereresolved", &cutoff_resolve);
|
|
|
|
git_config_get_int("gc.rerereunresolved", &cutoff_noresolve);
|
|
|
|
git_config(git_default_config, NULL);
|
2011-05-09 03:55:34 +08:00
|
|
|
dir = opendir(git_path("rr-cache"));
|
|
|
|
if (!dir)
|
|
|
|
die_errno("unable to open rr-cache directory");
|
2015-07-01 13:43:37 +08:00
|
|
|
/* Collect stale conflict IDs ... */
|
2011-05-09 03:55:34 +08:00
|
|
|
while ((e = readdir(dir))) {
|
|
|
|
if (is_dot_or_dotdot(e->d_name))
|
|
|
|
continue;
|
|
|
|
|
|
|
|
then = rerere_last_used_at(e->d_name);
|
|
|
|
if (then) {
|
2014-08-08 00:21:21 +08:00
|
|
|
cutoff = cutoff_resolve;
|
2011-05-09 03:55:34 +08:00
|
|
|
} else {
|
|
|
|
then = rerere_created_at(e->d_name);
|
|
|
|
if (!then)
|
|
|
|
continue;
|
2014-08-08 00:21:21 +08:00
|
|
|
cutoff = cutoff_noresolve;
|
2011-05-09 03:55:34 +08:00
|
|
|
}
|
|
|
|
if (then < now - cutoff * 86400)
|
|
|
|
string_list_append(&to_remove, e->d_name);
|
|
|
|
}
|
2011-05-26 21:55:50 +08:00
|
|
|
closedir(dir);
|
2015-07-01 13:43:37 +08:00
|
|
|
/* ... and then remove them one-by-one */
|
2011-05-09 03:55:34 +08:00
|
|
|
for (i = 0; i < to_remove.nr; i++)
|
2015-07-05 08:38:34 +08:00
|
|
|
unlink_rr_item(dirname_to_id(to_remove.items[i].string));
|
2011-05-09 03:55:34 +08:00
|
|
|
string_list_clear(&to_remove, 0);
|
|
|
|
}
|
|
|
|
|
2015-07-01 13:43:37 +08:00
|
|
|
/*
|
|
|
|
* During a conflict resolution, after "rerere" recorded the
|
|
|
|
* preimages, abandon them if the user did not resolve them or
|
|
|
|
* record their resolutions. And drop $GIT_DIR/MERGE_RR.
|
|
|
|
*
|
|
|
|
* NEEDSWORK: shouldn't we be calling this from "reset --hard"?
|
|
|
|
*/
|
2011-05-09 03:55:34 +08:00
|
|
|
void rerere_clear(struct string_list *merge_rr)
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
|
|
|
|
for (i = 0; i < merge_rr->nr; i++) {
|
2015-07-05 08:38:34 +08:00
|
|
|
struct rerere_id *id = merge_rr->items[i].util;
|
2015-07-07 05:45:55 +08:00
|
|
|
if (!has_rerere_resolution(id))
|
|
|
|
unlink_rr_item(id);
|
2011-05-09 03:55:34 +08:00
|
|
|
}
|
|
|
|
unlink_or_warn(git_path("MERGE_RR"));
|
|
|
|
}
|