You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
222 lines
5.3 KiB
222 lines
5.3 KiB
#include "cache.h" |
|
#include "commit.h" |
|
#include "tag.h" |
|
#include "pkt-line.h" |
|
|
|
static int is_shallow = -1; |
|
static struct stat shallow_stat; |
|
static char *alternate_shallow_file; |
|
|
|
void set_alternate_shallow_file(const char *path) |
|
{ |
|
if (is_shallow != -1) |
|
die("BUG: is_repository_shallow must not be called before set_alternate_shallow_file"); |
|
free(alternate_shallow_file); |
|
alternate_shallow_file = path ? xstrdup(path) : NULL; |
|
} |
|
|
|
int register_shallow(const unsigned char *sha1) |
|
{ |
|
struct commit_graft *graft = |
|
xmalloc(sizeof(struct commit_graft)); |
|
struct commit *commit = lookup_commit(sha1); |
|
|
|
hashcpy(graft->sha1, sha1); |
|
graft->nr_parent = -1; |
|
if (commit && commit->object.parsed) |
|
commit->parents = NULL; |
|
return register_commit_graft(graft, 0); |
|
} |
|
|
|
int is_repository_shallow(void) |
|
{ |
|
FILE *fp; |
|
char buf[1024]; |
|
const char *path = alternate_shallow_file; |
|
|
|
if (is_shallow >= 0) |
|
return is_shallow; |
|
|
|
if (!path) |
|
path = git_path("shallow"); |
|
/* |
|
* fetch-pack sets '--shallow-file ""' as an indicator that no |
|
* shallow file should be used. We could just open it and it |
|
* will likely fail. But let's do an explicit check instead. |
|
*/ |
|
if (!*path || |
|
stat(path, &shallow_stat) || |
|
(fp = fopen(path, "r")) == NULL) { |
|
is_shallow = 0; |
|
return is_shallow; |
|
} |
|
is_shallow = 1; |
|
|
|
while (fgets(buf, sizeof(buf), fp)) { |
|
unsigned char sha1[20]; |
|
if (get_sha1_hex(buf, sha1)) |
|
die("bad shallow line: %s", buf); |
|
register_shallow(sha1); |
|
} |
|
fclose(fp); |
|
return is_shallow; |
|
} |
|
|
|
struct commit_list *get_shallow_commits(struct object_array *heads, int depth, |
|
int shallow_flag, int not_shallow_flag) |
|
{ |
|
int i = 0, cur_depth = 0; |
|
struct commit_list *result = NULL; |
|
struct object_array stack = OBJECT_ARRAY_INIT; |
|
struct commit *commit = NULL; |
|
|
|
while (commit || i < heads->nr || stack.nr) { |
|
struct commit_list *p; |
|
if (!commit) { |
|
if (i < heads->nr) { |
|
commit = (struct commit *) |
|
deref_tag(heads->objects[i++].item, NULL, 0); |
|
if (!commit || commit->object.type != OBJ_COMMIT) { |
|
commit = NULL; |
|
continue; |
|
} |
|
if (!commit->util) |
|
commit->util = xmalloc(sizeof(int)); |
|
*(int *)commit->util = 0; |
|
cur_depth = 0; |
|
} else { |
|
commit = (struct commit *) |
|
stack.objects[--stack.nr].item; |
|
cur_depth = *(int *)commit->util; |
|
} |
|
} |
|
if (parse_commit(commit)) |
|
die("invalid commit"); |
|
cur_depth++; |
|
if (cur_depth >= depth) { |
|
commit_list_insert(commit, &result); |
|
commit->object.flags |= shallow_flag; |
|
commit = NULL; |
|
continue; |
|
} |
|
commit->object.flags |= not_shallow_flag; |
|
for (p = commit->parents, commit = NULL; p; p = p->next) { |
|
if (!p->item->util) { |
|
int *pointer = xmalloc(sizeof(int)); |
|
p->item->util = pointer; |
|
*pointer = cur_depth; |
|
} else { |
|
int *pointer = p->item->util; |
|
if (cur_depth >= *pointer) |
|
continue; |
|
*pointer = cur_depth; |
|
} |
|
if (p->next) |
|
add_object_array(&p->item->object, |
|
NULL, &stack); |
|
else { |
|
commit = p->item; |
|
cur_depth = *(int *)commit->util; |
|
} |
|
} |
|
} |
|
|
|
return result; |
|
} |
|
|
|
void check_shallow_file_for_update(void) |
|
{ |
|
struct stat st; |
|
|
|
if (!is_shallow) |
|
return; |
|
else if (is_shallow == -1) |
|
die("BUG: shallow must be initialized by now"); |
|
|
|
if (stat(git_path("shallow"), &st)) |
|
die("shallow file was removed during fetch"); |
|
else if (st.st_mtime != shallow_stat.st_mtime |
|
#ifdef USE_NSEC |
|
|| ST_MTIME_NSEC(st) != ST_MTIME_NSEC(shallow_stat) |
|
#endif |
|
) |
|
die("shallow file was changed during fetch"); |
|
} |
|
|
|
struct write_shallow_data { |
|
struct strbuf *out; |
|
int use_pack_protocol; |
|
int count; |
|
}; |
|
|
|
static int write_one_shallow(const struct commit_graft *graft, void *cb_data) |
|
{ |
|
struct write_shallow_data *data = cb_data; |
|
const char *hex = sha1_to_hex(graft->sha1); |
|
if (graft->nr_parent != -1) |
|
return 0; |
|
data->count++; |
|
if (data->use_pack_protocol) |
|
packet_buf_write(data->out, "shallow %s", hex); |
|
else { |
|
strbuf_addstr(data->out, hex); |
|
strbuf_addch(data->out, '\n'); |
|
} |
|
return 0; |
|
} |
|
|
|
int write_shallow_commits(struct strbuf *out, int use_pack_protocol) |
|
{ |
|
struct write_shallow_data data; |
|
data.out = out; |
|
data.use_pack_protocol = use_pack_protocol; |
|
data.count = 0; |
|
for_each_commit_graft(write_one_shallow, &data); |
|
return data.count; |
|
} |
|
|
|
char *setup_temporary_shallow(void) |
|
{ |
|
struct strbuf sb = STRBUF_INIT; |
|
int fd; |
|
|
|
if (write_shallow_commits(&sb, 0)) { |
|
struct strbuf path = STRBUF_INIT; |
|
strbuf_addstr(&path, git_path("shallow_XXXXXX")); |
|
fd = xmkstemp(path.buf); |
|
if (write_in_full(fd, sb.buf, sb.len) != sb.len) |
|
die_errno("failed to write to %s", |
|
path.buf); |
|
close(fd); |
|
strbuf_release(&sb); |
|
return strbuf_detach(&path, NULL); |
|
} |
|
/* |
|
* is_repository_shallow() sees empty string as "no shallow |
|
* file". |
|
*/ |
|
return xstrdup(""); |
|
} |
|
|
|
void setup_alternate_shallow(struct lock_file *shallow_lock, |
|
const char **alternate_shallow_file) |
|
{ |
|
struct strbuf sb = STRBUF_INIT; |
|
int fd; |
|
|
|
check_shallow_file_for_update(); |
|
fd = hold_lock_file_for_update(shallow_lock, git_path("shallow"), |
|
LOCK_DIE_ON_ERROR); |
|
if (write_shallow_commits(&sb, 0)) { |
|
if (write_in_full(fd, sb.buf, sb.len) != sb.len) |
|
die_errno("failed to write to %s", |
|
shallow_lock->filename); |
|
*alternate_shallow_file = shallow_lock->filename; |
|
} else |
|
/* |
|
* is_repository_shallow() sees empty string as "no |
|
* shallow file". |
|
*/ |
|
*alternate_shallow_file = ""; |
|
strbuf_release(&sb); |
|
}
|
|
|