Merge branch 'jc/upload-pack'

This commit is contained in:
Junio C Hamano 2006-08-12 22:16:51 -07:00
commit 182a8dabd5

View File

@ -14,12 +14,10 @@ static const char upload_pack_usage[] = "git-upload-pack [--strict] [--timeout=n
#define THEY_HAVE (1U << 0) #define THEY_HAVE (1U << 0)
#define OUR_REF (1U << 1) #define OUR_REF (1U << 1)
#define WANTED (1U << 2) #define WANTED (1U << 2)
#define MAX_HAS 256 static int multi_ack = 0, nr_our_refs = 0;
#define MAX_NEEDS 256
static int nr_has = 0, nr_needs = 0, multi_ack = 0, nr_our_refs = 0;
static int use_thin_pack = 0; static int use_thin_pack = 0;
static unsigned char has_sha1[MAX_HAS][20]; static struct object_array have_obj;
static unsigned char needs_sha1[MAX_NEEDS][20]; static struct object_array want_obj;
static unsigned int timeout = 0; static unsigned int timeout = 0;
static int use_sideband = 0; static int use_sideband = 0;
@ -83,7 +81,7 @@ static void create_pack_file(void)
*/ */
int lp_pipe[2], pu_pipe[2], pe_pipe[2]; int lp_pipe[2], pu_pipe[2], pe_pipe[2];
pid_t pid_rev_list, pid_pack_objects; pid_t pid_rev_list, pid_pack_objects;
int create_full_pack = (nr_our_refs == nr_needs && !nr_has); int create_full_pack = (nr_our_refs == want_obj.nr && !have_obj.nr);
char data[8193], progress[128]; char data[8193], progress[128];
char abort_msg[] = "aborting due to possible repository " char abort_msg[] = "aborting due to possible repository "
"corruption on the remote side."; "corruption on the remote side.";
@ -107,7 +105,7 @@ static void create_pack_file(void)
use_thin_pack = 0; /* no point doing it */ use_thin_pack = 0; /* no point doing it */
} }
else else
args = nr_has + nr_needs + 5; args = have_obj.nr + want_obj.nr + 5;
p = xmalloc(args * sizeof(char *)); p = xmalloc(args * sizeof(char *));
argv = (const char **) p; argv = (const char **) p;
buf = xmalloc(args * 45); buf = xmalloc(args * 45);
@ -118,20 +116,22 @@ static void create_pack_file(void)
close(lp_pipe[1]); close(lp_pipe[1]);
*p++ = "rev-list"; *p++ = "rev-list";
*p++ = use_thin_pack ? "--objects-edge" : "--objects"; *p++ = use_thin_pack ? "--objects-edge" : "--objects";
if (create_full_pack || MAX_NEEDS <= nr_needs) if (create_full_pack)
*p++ = "--all"; *p++ = "--all";
else { else {
for (i = 0; i < nr_needs; i++) { for (i = 0; i < want_obj.nr; i++) {
struct object *o = want_obj.objects[i].item;
*p++ = buf; *p++ = buf;
memcpy(buf, sha1_to_hex(needs_sha1[i]), 41); memcpy(buf, sha1_to_hex(o->sha1), 41);
buf += 41; buf += 41;
} }
} }
if (!create_full_pack) if (!create_full_pack)
for (i = 0; i < nr_has; i++) { for (i = 0; i < have_obj.nr; i++) {
struct object *o = have_obj.objects[i].item;
*p++ = buf; *p++ = buf;
*buf++ = '^'; *buf++ = '^';
memcpy(buf, sha1_to_hex(has_sha1[i]), 41); memcpy(buf, sha1_to_hex(o->sha1), 41);
buf += 41; buf += 41;
} }
*p++ = NULL; *p++ = NULL;
@ -322,12 +322,14 @@ static void create_pack_file(void)
static int got_sha1(char *hex, unsigned char *sha1) static int got_sha1(char *hex, unsigned char *sha1)
{ {
struct object *o;
if (get_sha1_hex(hex, sha1)) if (get_sha1_hex(hex, sha1))
die("git-upload-pack: expected SHA1 object, got '%s'", hex); die("git-upload-pack: expected SHA1 object, got '%s'", hex);
if (!has_sha1_file(sha1)) if (!has_sha1_file(sha1))
return 0; return 0;
if (nr_has < MAX_HAS) {
struct object *o = lookup_object(sha1); o = lookup_object(sha1);
if (!(o && o->parsed)) if (!(o && o->parsed))
o = parse_object(sha1); o = parse_object(sha1);
if (!o) if (!o)
@ -342,8 +344,7 @@ static int got_sha1(char *hex, unsigned char *sha1)
parents = parents->next) parents = parents->next)
parents->item->object.flags |= THEY_HAVE; parents->item->object.flags |= THEY_HAVE;
} }
memcpy(has_sha1[nr_has++], sha1, 20); add_object_array(o, NULL, &have_obj);
}
return 1; return 1;
} }
@ -361,16 +362,14 @@ static int get_common_commits(void)
reset_timeout(); reset_timeout();
if (!len) { if (!len) {
if (nr_has == 0 || multi_ack) if (have_obj.nr == 0 || multi_ack)
packet_write(1, "NAK\n"); packet_write(1, "NAK\n");
continue; continue;
} }
len = strip(line, len); len = strip(line, len);
if (!strncmp(line, "have ", 5)) { if (!strncmp(line, "have ", 5)) {
if (got_sha1(line+5, sha1) && if (got_sha1(line+5, sha1) &&
(multi_ack || nr_has == 1)) { (multi_ack || have_obj.nr == 1)) {
if (nr_has >= MAX_HAS)
multi_ack = 0;
packet_write(1, "ACK %s%s\n", packet_write(1, "ACK %s%s\n",
sha1_to_hex(sha1), sha1_to_hex(sha1),
multi_ack ? " continue" : ""); multi_ack ? " continue" : "");
@ -380,7 +379,7 @@ static int get_common_commits(void)
continue; continue;
} }
if (!strcmp(line, "done")) { if (!strcmp(line, "done")) {
if (nr_has > 0) { if (have_obj.nr > 0) {
if (multi_ack) if (multi_ack)
packet_write(1, "ACK %s\n", packet_write(1, "ACK %s\n",
sha1_to_hex(last_sha1)); sha1_to_hex(last_sha1));
@ -393,31 +392,21 @@ static int get_common_commits(void)
} }
} }
static int receive_needs(void) static void receive_needs(void)
{ {
static char line[1000]; static char line[1000];
int len, needs; int len;
needs = 0;
for (;;) { for (;;) {
struct object *o; struct object *o;
unsigned char dummy[20], *sha1_buf; unsigned char sha1_buf[20];
len = packet_read_line(0, line, sizeof(line)); len = packet_read_line(0, line, sizeof(line));
reset_timeout(); reset_timeout();
if (!len) if (!len)
return needs; return;
sha1_buf = dummy; if (strncmp("want ", line, 5) ||
if (needs == MAX_NEEDS) { get_sha1_hex(line+5, sha1_buf))
fprintf(stderr,
"warning: supporting only a max of %d requests. "
"sending everything instead.\n",
MAX_NEEDS);
}
else if (needs < MAX_NEEDS)
sha1_buf = needs_sha1[needs];
if (strncmp("want ", line, 5) || get_sha1_hex(line+5, sha1_buf))
die("git-upload-pack: protocol error, " die("git-upload-pack: protocol error, "
"expected to get sha, not '%s'", line); "expected to get sha, not '%s'", line);
if (strstr(line+45, "multi_ack")) if (strstr(line+45, "multi_ack"))
@ -440,7 +429,7 @@ static int receive_needs(void)
die("git-upload-pack: not our ref %s", line+5); die("git-upload-pack: not our ref %s", line+5);
if (!(o->flags & WANTED)) { if (!(o->flags & WANTED)) {
o->flags |= WANTED; o->flags |= WANTED;
needs++; add_object_array(o, NULL, &want_obj);
} }
} }
} }
@ -476,8 +465,8 @@ static int upload_pack(void)
head_ref(send_ref); head_ref(send_ref);
for_each_ref(send_ref); for_each_ref(send_ref);
packet_flush(1); packet_flush(1);
nr_needs = receive_needs(); receive_needs();
if (!nr_needs) if (!want_obj.nr)
return 0; return 0;
get_common_commits(); get_common_commits();
create_pack_file(); create_pack_file();