summaryrefslogtreecommitdiffstats
path: root/delta.c
blob: b31957a1d38046b9ead2569bbdd59be061f0157b (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
#include "object.h"
#include "blob.h"
#include "tree.h"
#include "commit.h"
#include "tag.h"
#include "delta.h"
#include "cache.h"

/* the delta object definition (it can alias any other object) */
struct delta {
	union {
		struct object object;
		struct blob blob;
		struct tree tree;
		struct commit commit;
		struct tag tag;
	} u;
};

struct delta *lookup_delta(unsigned char *sha1)
{
	struct object *obj = lookup_object(sha1);
	if (!obj) {
		struct delta *ret = xmalloc(sizeof(struct delta));
		memset(ret, 0, sizeof(struct delta));
		created_object(sha1, &ret->u.object);
		return ret;
	}
	return (struct delta *) obj;
}

int parse_delta_buffer(struct delta *item, void *buffer, unsigned long size)
{
	struct object *reference;
	struct object_list *p;

	if (item->u.object.delta)
		return 0;
	item->u.object.delta = 1;
	if (size <= 20)
		return -1;
	reference = lookup_object(buffer);
	if (!reference) {
		struct delta *ref = xmalloc(sizeof(struct delta));
		memset(ref, 0, sizeof(struct delta));
		created_object(buffer, &ref->u.object);
		reference = &ref->u.object;
	}

	p = xmalloc(sizeof(*p));
	p->item = &item->u.object;
	p->next = reference->attached_deltas;
	reference->attached_deltas = p;
	return 0;
}

int process_deltas(void *src, unsigned long src_size, const char *src_type,
		   struct object_list *delta_list)
{
	int deepest = 0;
	do {
		struct object *obj = delta_list->item;
		static char type[10];
		void *map, *delta, *buf;
		unsigned long map_size, delta_size, buf_size;
		map = map_sha1_file(obj->sha1, &map_size);
		if (!map)
			continue;
		delta = unpack_sha1_file(map, map_size, type, &delta_size);
		munmap(map, map_size);
		if (!delta)
			continue;
		if (strcmp(type, "delta") || delta_size <= 20) {
			free(delta);
			continue;
		}
		buf = patch_delta(src, src_size,
				  delta+20, delta_size-20,
				  &buf_size);
		free(delta);
		if (!buf)
			continue;
		if (check_sha1_signature(obj->sha1, buf, buf_size, src_type) < 0)
			printf("sha1 mismatch for delta %s\n", sha1_to_hex(obj->sha1));
		if (obj->type && obj->type != src_type) {
			error("got %s when expecting %s for delta %s",
			      src_type, obj->type, sha1_to_hex(obj->sha1));
			free(buf);
			continue;
		}
		obj->type = src_type;
		if (src_type == blob_type) {
			parse_blob_buffer((struct blob *)obj, buf, buf_size);
		} else if (src_type == tree_type) {
			parse_tree_buffer((struct tree *)obj, buf, buf_size);
		} else if (src_type == commit_type) {
			parse_commit_buffer((struct commit *)obj, buf, buf_size);
		} else if (src_type == tag_type) {
			parse_tag_buffer((struct tag *)obj, buf, buf_size);
		} else {
			error("unknown object type %s", src_type);
			free(buf);
			continue;
		}
		if (obj->attached_deltas) {
			int depth = process_deltas(buf, buf_size, src_type,
						   obj->attached_deltas);
			if (deepest < depth)
				deepest = depth;
		}
		free(buf);
	} while ((delta_list = delta_list->next));
	return deepest + 1;
}