pack.h 4.35 KB
Newer Older
1
/*
Edward Thomson committed
2
 * Copyright (C) the libgit2 contributors. All rights reserved.
3
 *
Vicent Marti committed
4 5
 * This file is part of libgit2, distributed under the GNU GPL v2 with
 * a Linking Exception. For full terms see the included COPYING file.
6 7 8 9 10
 */

#ifndef INCLUDE_pack_h__
#define INCLUDE_pack_h__

11 12
#include <zlib.h>

13 14 15 16 17
#include "git2/oid.h"

#include "common.h"
#include "map.h"
#include "mwindow.h"
18
#include "odb.h"
19
#include "oidmap.h"
20
#include "array.h"
21

22 23
#define GIT_PACK_FILE_MODE 0444

24 25 26
#define PACK_SIGNATURE 0x5041434b	/* "PACK" */
#define PACK_VERSION 2
#define pack_version_ok(v) ((v) == htonl(2) || (v) == htonl(3))
27
struct git_pack_header {
28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46
	uint32_t hdr_signature;
	uint32_t hdr_version;
	uint32_t hdr_entries;
};

/*
 * The first four bytes of index formats later than version 1 should
 * start with this signature, as all older git binaries would find this
 * value illegal and abort reading the file.
 *
 * This is the case because the number of objects in a packfile
 * cannot exceed 1,431,660,000 as every object would need at least
 * 3 bytes of data and the overall packfile cannot exceed 4 GiB with
 * version 1 of the index file due to the offsets limited to 32 bits.
 * Clearly the signature exceeds this maximum.
 *
 * Very old git binaries will also compare the first 4 bytes to the
 * next 4 bytes in the index and abort with a "non-monotonic index"
 * error if the second 4 byte word is smaller than the first 4
Vicent Marti committed
47
 * byte word. This would be true in the proposed future index
48 49 50 51 52
 * format as idx_signature would be greater than idx_version.
 */

#define PACK_IDX_SIGNATURE 0xff744f63	/* "\377tOc" */

53
struct git_pack_idx_header {
54 55 56 57
	uint32_t idx_signature;
	uint32_t idx_version;
};

58
typedef struct git_pack_cache_entry {
59
	size_t last_usage; /* enough? */
60
	git_atomic refcount;
61 62 63
	git_rawobj raw;
} git_pack_cache_entry;

64
struct pack_chain_elem {
65
	git_off_t base_key;
66 67 68 69 70 71 72
	git_off_t offset;
	size_t size;
	git_otype type;
};

typedef git_array_t(struct pack_chain_elem) git_dependency_chain;

73 74 75
#include "offmap.h"

GIT__USE_OFFMAP;
76
GIT__USE_OIDMAP;
77

78 79
#define GIT_PACK_CACHE_MEMORY_LIMIT 16 * 1024 * 1024
#define GIT_PACK_CACHE_SIZE_LIMIT 1024 * 1024 /* don't bother caching anything over 1MB */
80 81 82 83

typedef struct {
	size_t memory_used;
	size_t memory_limit;
84
	size_t use_ctr;
85 86 87 88
	git_mutex lock;
	git_offmap *entries;
} git_pack_cache;

89
struct git_pack_file {
90 91
	git_mwindow_file mwf;
	git_map index_map;
Russell Belfer committed
92
	git_mutex lock; /* protect updates to mwf and index_map */
93
	git_atomic refcount;
94 95 96 97 98 99 100

	uint32_t num_objects;
	uint32_t num_bad_objects;
	git_oid *bad_object_sha1; /* array of git_oid */

	int index_version;
	git_time_t mtime;
101
	unsigned pack_local:1, pack_keep:1, has_cache:1;
102
	git_oidmap *idx_cache;
103
	git_oid **oids;
104

105
	git_pack_cache bases; /* delta base cache */
106

107 108 109 110
	/* something like ".git/objects/pack/xxxxx.pack" */
	char pack_name[GIT_FLEX_ARRAY]; /* more */
};

111
struct git_pack_entry {
112
	git_off_t offset;
113
	git_oid sha1;
114
	struct git_pack_file *p;
115 116
};

117 118 119 120 121 122 123 124
typedef struct git_packfile_stream {
	git_off_t curpos;
	int done;
	z_stream zstream;
	struct git_pack_file *p;
	git_mwindow *mw;
} git_packfile_stream;

125
size_t git_packfile__object_header(unsigned char *hdr, size_t size, git_otype type);
126

127 128
int git_packfile__name(char **out, const char *path);

129 130 131 132 133
int git_packfile_unpack_header(
		size_t *size_p,
		git_otype *type_p,
		git_mwindow_file *mwf,
		git_mwindow **w_curs,
134
		git_off_t *curpos);
135

136 137 138 139 140 141
int git_packfile_resolve_header(
		size_t *size_p,
		git_otype *type_p,
		struct git_pack_file *p,
		git_off_t offset);

142
int git_packfile_unpack(git_rawobj *obj, struct git_pack_file *p, git_off_t *obj_offset);
143 144 145 146 147 148 149
int packfile_unpack_compressed(
	git_rawobj *obj,
	struct git_pack_file *p,
	git_mwindow **w_curs,
	git_off_t *curpos,
	size_t size,
	git_otype type);
150

151 152 153 154
int git_packfile_stream_open(git_packfile_stream *obj, struct git_pack_file *p, git_off_t curpos);
ssize_t git_packfile_stream_read(git_packfile_stream *obj, void *buffer, size_t len);
void git_packfile_stream_free(git_packfile_stream *obj);

155 156 157
git_off_t get_delta_base(struct git_pack_file *p, git_mwindow **w_curs,
		git_off_t *curpos, git_otype type,
		git_off_t delta_obj_offset);
158

159
void git_packfile_free(struct git_pack_file *p);
160 161
int git_packfile_alloc(struct git_pack_file **pack_out, const char *path);

162 163 164 165
int git_pack_entry_find(
		struct git_pack_entry *e,
		struct git_pack_file *p,
		const git_oid *short_oid,
166
		size_t len);
167 168
int git_pack_foreach_entry(
		struct git_pack_file *p,
169
		git_odb_foreach_cb cb,
170
		void *data);
171

172
#endif