Elijah Newren | 5579f44 | 2023-04-11 00:41:48 -0700 | [diff] [blame] | 1 | #include "git-compat-util.h" |
Elijah Newren | 32a8f51 | 2023-03-21 06:26:03 +0000 | [diff] [blame] | 2 | #include "environment.h" |
Elijah Newren | f394e09 | 2023-03-21 06:25:54 +0000 | [diff] [blame] | 3 | #include "gettext.h" |
Elijah Newren | 41771fa | 2023-02-24 00:09:27 +0000 | [diff] [blame] | 4 | #include "hex.h" |
Dana L. How | 8b0eca7 | 2007-05-02 12:13:14 -0400 | [diff] [blame] | 5 | #include "pack.h" |
Geert Bosch | aa7e44b | 2007-06-01 15:18:05 -0400 | [diff] [blame] | 6 | #include "csum-file.h" |
Christian Couder | 33add2a | 2021-01-12 09:21:59 +0100 | [diff] [blame] | 7 | #include "remote.h" |
Taylor Blau | d9fef9d | 2022-05-20 19:17:41 -0400 | [diff] [blame] | 8 | #include "chunk-format.h" |
Taylor Blau | 5dfaf49 | 2022-05-20 19:17:43 -0400 | [diff] [blame] | 9 | #include "pack-mtimes.h" |
| 10 | #include "oidmap.h" |
Taylor Blau | 5dfaf49 | 2022-05-20 19:17:43 -0400 | [diff] [blame] | 11 | #include "pack-objects.h" |
Elijah Newren | 75f273d | 2023-04-11 03:00:41 +0000 | [diff] [blame] | 12 | #include "pack-revindex.h" |
Elijah Newren | c339932 | 2023-05-16 06:33:59 +0000 | [diff] [blame] | 13 | #include "path.h" |
Elijah Newren | a034e91 | 2023-05-16 06:34:06 +0000 | [diff] [blame] | 14 | #include "strbuf.h" |
Geert Bosch | aa7e44b | 2007-06-01 15:18:05 -0400 | [diff] [blame] | 15 | |
Junio C Hamano | ebcfb37 | 2011-02-25 15:43:25 -0800 | [diff] [blame] | 16 | void reset_pack_idx_option(struct pack_idx_option *opts) |
| 17 | { |
| 18 | memset(opts, 0, sizeof(*opts)); |
| 19 | opts->version = 2; |
| 20 | opts->off32_limit = 0x7fffffff; |
| 21 | } |
Geert Bosch | aa7e44b | 2007-06-01 15:18:05 -0400 | [diff] [blame] | 22 | |
| 23 | static int sha1_compare(const void *_a, const void *_b) |
| 24 | { |
| 25 | struct pack_idx_entry *a = *(struct pack_idx_entry **)_a; |
| 26 | struct pack_idx_entry *b = *(struct pack_idx_entry **)_b; |
brian m. carlson | e6a492b | 2017-05-06 22:10:11 +0000 | [diff] [blame] | 27 | return oidcmp(&a->oid, &b->oid); |
Geert Bosch | aa7e44b | 2007-06-01 15:18:05 -0400 | [diff] [blame] | 28 | } |
| 29 | |
Junio C Hamano | 3c9fc07 | 2011-02-25 16:55:26 -0800 | [diff] [blame] | 30 | static int cmp_uint32(const void *a_, const void *b_) |
| 31 | { |
| 32 | uint32_t a = *((uint32_t *)a_); |
| 33 | uint32_t b = *((uint32_t *)b_); |
| 34 | |
| 35 | return (a < b) ? -1 : (a != b); |
| 36 | } |
| 37 | |
Junio C Hamano | fb956c1 | 2011-02-25 16:54:00 -0800 | [diff] [blame] | 38 | static int need_large_offset(off_t offset, const struct pack_idx_option *opts) |
| 39 | { |
Junio C Hamano | 3c9fc07 | 2011-02-25 16:55:26 -0800 | [diff] [blame] | 40 | uint32_t ofsval; |
| 41 | |
| 42 | if ((offset >> 31) || (opts->off32_limit < offset)) |
| 43 | return 1; |
| 44 | if (!opts->anomaly_nr) |
| 45 | return 0; |
| 46 | ofsval = offset; |
| 47 | return !!bsearch(&ofsval, opts->anomaly, opts->anomaly_nr, |
| 48 | sizeof(ofsval), cmp_uint32); |
Junio C Hamano | fb956c1 | 2011-02-25 16:54:00 -0800 | [diff] [blame] | 49 | } |
| 50 | |
Geert Bosch | aa7e44b | 2007-06-01 15:18:05 -0400 | [diff] [blame] | 51 | /* |
Johannes Berg | e2bfa50 | 2020-07-22 23:40:31 +0200 | [diff] [blame] | 52 | * The *sha1 contains the pack content SHA1 hash. |
| 53 | * The objects array passed in will be sorted by SHA1 on exit. |
Geert Bosch | aa7e44b | 2007-06-01 15:18:05 -0400 | [diff] [blame] | 54 | */ |
Linus Torvalds | 3bb7256 | 2010-01-22 07:55:19 -0800 | [diff] [blame] | 55 | const char *write_idx_file(const char *index_name, struct pack_idx_entry **objects, |
Junio C Hamano | ebcfb37 | 2011-02-25 15:43:25 -0800 | [diff] [blame] | 56 | int nr_objects, const struct pack_idx_option *opts, |
Jeff King | 1190a1a | 2013-12-05 15:28:07 -0500 | [diff] [blame] | 57 | const unsigned char *sha1) |
Geert Bosch | aa7e44b | 2007-06-01 15:18:05 -0400 | [diff] [blame] | 58 | { |
brian m. carlson | 98a3bea | 2018-02-01 02:18:46 +0000 | [diff] [blame] | 59 | struct hashfile *f; |
Geert Bosch | aa7e44b | 2007-06-01 15:18:05 -0400 | [diff] [blame] | 60 | struct pack_idx_entry **sorted_by_sha, **list, **last; |
| 61 | off_t last_obj_offset = 0; |
Geert Bosch | aa7e44b | 2007-06-01 15:18:05 -0400 | [diff] [blame] | 62 | int i, fd; |
Geert Bosch | aa7e44b | 2007-06-01 15:18:05 -0400 | [diff] [blame] | 63 | uint32_t index_version; |
| 64 | |
| 65 | if (nr_objects) { |
| 66 | sorted_by_sha = objects; |
| 67 | list = sorted_by_sha; |
| 68 | last = sorted_by_sha + nr_objects; |
| 69 | for (i = 0; i < nr_objects; ++i) { |
| 70 | if (objects[i]->offset > last_obj_offset) |
| 71 | last_obj_offset = objects[i]->offset; |
| 72 | } |
René Scharfe | 9ed0d8d | 2016-09-29 17:27:31 +0200 | [diff] [blame] | 73 | QSORT(sorted_by_sha, nr_objects, sha1_compare); |
Geert Bosch | aa7e44b | 2007-06-01 15:18:05 -0400 | [diff] [blame] | 74 | } |
| 75 | else |
| 76 | sorted_by_sha = list = last = NULL; |
| 77 | |
Junio C Hamano | e337a04 | 2011-02-02 17:29:01 -0800 | [diff] [blame] | 78 | if (opts->flags & WRITE_IDX_VERIFY) { |
| 79 | assert(index_name); |
brian m. carlson | 98a3bea | 2018-02-01 02:18:46 +0000 | [diff] [blame] | 80 | f = hashfd_check(index_name); |
Geert Bosch | aa7e44b | 2007-06-01 15:18:05 -0400 | [diff] [blame] | 81 | } else { |
Junio C Hamano | e337a04 | 2011-02-02 17:29:01 -0800 | [diff] [blame] | 82 | if (!index_name) { |
Jeff King | 594fa99 | 2017-03-28 15:45:43 -0400 | [diff] [blame] | 83 | struct strbuf tmp_file = STRBUF_INIT; |
| 84 | fd = odb_mkstemp(&tmp_file, "pack/tmp_idx_XXXXXX"); |
| 85 | index_name = strbuf_detach(&tmp_file, NULL); |
Junio C Hamano | e337a04 | 2011-02-02 17:29:01 -0800 | [diff] [blame] | 86 | } else { |
| 87 | unlink(index_name); |
René Scharfe | 66e905b | 2021-08-25 22:16:46 +0200 | [diff] [blame] | 88 | fd = xopen(index_name, O_CREAT|O_EXCL|O_WRONLY, 0600); |
Junio C Hamano | e337a04 | 2011-02-02 17:29:01 -0800 | [diff] [blame] | 89 | } |
brian m. carlson | 98a3bea | 2018-02-01 02:18:46 +0000 | [diff] [blame] | 90 | f = hashfd(fd, index_name); |
Geert Bosch | aa7e44b | 2007-06-01 15:18:05 -0400 | [diff] [blame] | 91 | } |
Geert Bosch | aa7e44b | 2007-06-01 15:18:05 -0400 | [diff] [blame] | 92 | |
| 93 | /* if last object's offset is >= 2^31 we should use index V2 */ |
Junio C Hamano | fb956c1 | 2011-02-25 16:54:00 -0800 | [diff] [blame] | 94 | index_version = need_large_offset(last_obj_offset, opts) ? 2 : opts->version; |
Geert Bosch | aa7e44b | 2007-06-01 15:18:05 -0400 | [diff] [blame] | 95 | |
| 96 | /* index versions 2 and above need a header */ |
| 97 | if (index_version >= 2) { |
| 98 | struct pack_idx_header hdr; |
| 99 | hdr.idx_signature = htonl(PACK_IDX_SIGNATURE); |
| 100 | hdr.idx_version = htonl(index_version); |
brian m. carlson | 98a3bea | 2018-02-01 02:18:46 +0000 | [diff] [blame] | 101 | hashwrite(f, &hdr, sizeof(hdr)); |
Geert Bosch | aa7e44b | 2007-06-01 15:18:05 -0400 | [diff] [blame] | 102 | } |
| 103 | |
| 104 | /* |
| 105 | * Write the first-level table (the list is sorted, |
| 106 | * but we use a 256-entry lookup to be able to avoid |
| 107 | * having to do eight extra binary search iterations). |
| 108 | */ |
| 109 | for (i = 0; i < 256; i++) { |
| 110 | struct pack_idx_entry **next = list; |
| 111 | while (next < last) { |
| 112 | struct pack_idx_entry *obj = *next; |
brian m. carlson | e6a492b | 2017-05-06 22:10:11 +0000 | [diff] [blame] | 113 | if (obj->oid.hash[0] != i) |
Geert Bosch | aa7e44b | 2007-06-01 15:18:05 -0400 | [diff] [blame] | 114 | break; |
| 115 | next++; |
| 116 | } |
René Scharfe | 06d43fa | 2020-11-01 09:52:12 +0100 | [diff] [blame] | 117 | hashwrite_be32(f, next - sorted_by_sha); |
Geert Bosch | aa7e44b | 2007-06-01 15:18:05 -0400 | [diff] [blame] | 118 | list = next; |
| 119 | } |
Geert Bosch | aa7e44b | 2007-06-01 15:18:05 -0400 | [diff] [blame] | 120 | |
Geert Bosch | aa7e44b | 2007-06-01 15:18:05 -0400 | [diff] [blame] | 121 | /* |
| 122 | * Write the actual SHA1 entries.. |
| 123 | */ |
| 124 | list = sorted_by_sha; |
| 125 | for (i = 0; i < nr_objects; i++) { |
| 126 | struct pack_idx_entry *obj = *list++; |
René Scharfe | 389cf68 | 2020-09-19 20:26:36 +0200 | [diff] [blame] | 127 | if (index_version < 2) |
| 128 | hashwrite_be32(f, obj->offset); |
brian m. carlson | 98a3bea | 2018-02-01 02:18:46 +0000 | [diff] [blame] | 129 | hashwrite(f, obj->oid.hash, the_hash_algo->rawsz); |
Junio C Hamano | 68be2fe | 2011-11-16 22:04:13 -0800 | [diff] [blame] | 130 | if ((opts->flags & WRITE_IDX_STRICT) && |
Jeff King | 4a7e27e | 2018-08-28 17:22:40 -0400 | [diff] [blame] | 131 | (i && oideq(&list[-2]->oid, &obj->oid))) |
Junio C Hamano | 68be2fe | 2011-11-16 22:04:13 -0800 | [diff] [blame] | 132 | die("The same object %s appears twice in the pack", |
brian m. carlson | e6a492b | 2017-05-06 22:10:11 +0000 | [diff] [blame] | 133 | oid_to_hex(&obj->oid)); |
Geert Bosch | aa7e44b | 2007-06-01 15:18:05 -0400 | [diff] [blame] | 134 | } |
| 135 | |
| 136 | if (index_version >= 2) { |
| 137 | unsigned int nr_large_offset = 0; |
| 138 | |
| 139 | /* write the crc32 table */ |
| 140 | list = sorted_by_sha; |
| 141 | for (i = 0; i < nr_objects; i++) { |
| 142 | struct pack_idx_entry *obj = *list++; |
René Scharfe | 389cf68 | 2020-09-19 20:26:36 +0200 | [diff] [blame] | 143 | hashwrite_be32(f, obj->crc32); |
Geert Bosch | aa7e44b | 2007-06-01 15:18:05 -0400 | [diff] [blame] | 144 | } |
| 145 | |
| 146 | /* write the 32-bit offset table */ |
| 147 | list = sorted_by_sha; |
| 148 | for (i = 0; i < nr_objects; i++) { |
| 149 | struct pack_idx_entry *obj = *list++; |
Junio C Hamano | fb956c1 | 2011-02-25 16:54:00 -0800 | [diff] [blame] | 150 | uint32_t offset; |
| 151 | |
| 152 | offset = (need_large_offset(obj->offset, opts) |
| 153 | ? (0x80000000 | nr_large_offset++) |
| 154 | : obj->offset); |
René Scharfe | 389cf68 | 2020-09-19 20:26:36 +0200 | [diff] [blame] | 155 | hashwrite_be32(f, offset); |
Geert Bosch | aa7e44b | 2007-06-01 15:18:05 -0400 | [diff] [blame] | 156 | } |
| 157 | |
| 158 | /* write the large offset table */ |
| 159 | list = sorted_by_sha; |
| 160 | while (nr_large_offset) { |
| 161 | struct pack_idx_entry *obj = *list++; |
| 162 | uint64_t offset = obj->offset; |
Junio C Hamano | fb956c1 | 2011-02-25 16:54:00 -0800 | [diff] [blame] | 163 | |
| 164 | if (!need_large_offset(offset, opts)) |
| 165 | continue; |
René Scharfe | 970909c | 2020-11-12 13:23:10 +0100 | [diff] [blame] | 166 | hashwrite_be64(f, offset); |
Junio C Hamano | fb956c1 | 2011-02-25 16:54:00 -0800 | [diff] [blame] | 167 | nr_large_offset--; |
Geert Bosch | aa7e44b | 2007-06-01 15:18:05 -0400 | [diff] [blame] | 168 | } |
| 169 | } |
| 170 | |
brian m. carlson | 98a3bea | 2018-02-01 02:18:46 +0000 | [diff] [blame] | 171 | hashwrite(f, sha1, the_hash_algo->rawsz); |
Neeraj Singh | 020406e | 2022-03-10 22:43:21 +0000 | [diff] [blame] | 172 | finalize_hashfile(f, NULL, FSYNC_COMPONENT_PACK_METADATA, |
| 173 | CSUM_HASH_IN_STREAM | CSUM_CLOSE | |
| 174 | ((opts->flags & WRITE_IDX_VERIFY) ? 0 : CSUM_FSYNC)); |
Geert Bosch | aa7e44b | 2007-06-01 15:18:05 -0400 | [diff] [blame] | 175 | return index_name; |
| 176 | } |
Dana L. How | 8b0eca7 | 2007-05-02 12:13:14 -0400 | [diff] [blame] | 177 | |
Taylor Blau | 8ef50d9 | 2021-01-25 18:37:18 -0500 | [diff] [blame] | 178 | static int pack_order_cmp(const void *va, const void *vb, void *ctx) |
| 179 | { |
| 180 | struct pack_idx_entry **objects = ctx; |
| 181 | |
| 182 | off_t oa = objects[*(uint32_t*)va]->offset; |
| 183 | off_t ob = objects[*(uint32_t*)vb]->offset; |
| 184 | |
| 185 | if (oa < ob) |
| 186 | return -1; |
| 187 | if (oa > ob) |
| 188 | return 1; |
| 189 | return 0; |
| 190 | } |
| 191 | |
| 192 | static void write_rev_header(struct hashfile *f) |
| 193 | { |
Taylor Blau | 8ef50d9 | 2021-01-25 18:37:18 -0500 | [diff] [blame] | 194 | hashwrite_be32(f, RIDX_SIGNATURE); |
| 195 | hashwrite_be32(f, RIDX_VERSION); |
Taylor Blau | d9fef9d | 2022-05-20 19:17:41 -0400 | [diff] [blame] | 196 | hashwrite_be32(f, oid_version(the_hash_algo)); |
Taylor Blau | 8ef50d9 | 2021-01-25 18:37:18 -0500 | [diff] [blame] | 197 | } |
| 198 | |
| 199 | static void write_rev_index_positions(struct hashfile *f, |
Taylor Blau | a587b5a | 2021-03-30 11:04:29 -0400 | [diff] [blame] | 200 | uint32_t *pack_order, |
Taylor Blau | 8ef50d9 | 2021-01-25 18:37:18 -0500 | [diff] [blame] | 201 | uint32_t nr_objects) |
| 202 | { |
Taylor Blau | 8ef50d9 | 2021-01-25 18:37:18 -0500 | [diff] [blame] | 203 | uint32_t i; |
Taylor Blau | 8ef50d9 | 2021-01-25 18:37:18 -0500 | [diff] [blame] | 204 | for (i = 0; i < nr_objects; i++) |
| 205 | hashwrite_be32(f, pack_order[i]); |
Taylor Blau | 8ef50d9 | 2021-01-25 18:37:18 -0500 | [diff] [blame] | 206 | } |
| 207 | |
| 208 | static void write_rev_trailer(struct hashfile *f, const unsigned char *hash) |
| 209 | { |
| 210 | hashwrite(f, hash, the_hash_algo->rawsz); |
| 211 | } |
| 212 | |
| 213 | const char *write_rev_file(const char *rev_name, |
| 214 | struct pack_idx_entry **objects, |
| 215 | uint32_t nr_objects, |
| 216 | const unsigned char *hash, |
| 217 | unsigned flags) |
| 218 | { |
Taylor Blau | a587b5a | 2021-03-30 11:04:29 -0400 | [diff] [blame] | 219 | uint32_t *pack_order; |
| 220 | uint32_t i; |
| 221 | const char *ret; |
| 222 | |
Ævar Arnfjörð Bjarmason | 8fe8bae | 2021-09-08 03:08:03 +0200 | [diff] [blame] | 223 | if (!(flags & WRITE_REV) && !(flags & WRITE_REV_VERIFY)) |
| 224 | return NULL; |
| 225 | |
Taylor Blau | a587b5a | 2021-03-30 11:04:29 -0400 | [diff] [blame] | 226 | ALLOC_ARRAY(pack_order, nr_objects); |
| 227 | for (i = 0; i < nr_objects; i++) |
| 228 | pack_order[i] = i; |
| 229 | QSORT_S(pack_order, nr_objects, pack_order_cmp, objects); |
| 230 | |
| 231 | ret = write_rev_file_order(rev_name, pack_order, nr_objects, hash, |
| 232 | flags); |
| 233 | |
| 234 | free(pack_order); |
| 235 | |
| 236 | return ret; |
| 237 | } |
| 238 | |
| 239 | const char *write_rev_file_order(const char *rev_name, |
| 240 | uint32_t *pack_order, |
| 241 | uint32_t nr_objects, |
| 242 | const unsigned char *hash, |
| 243 | unsigned flags) |
| 244 | { |
Taylor Blau | 8ef50d9 | 2021-01-25 18:37:18 -0500 | [diff] [blame] | 245 | struct hashfile *f; |
| 246 | int fd; |
| 247 | |
| 248 | if ((flags & WRITE_REV) && (flags & WRITE_REV_VERIFY)) |
| 249 | die(_("cannot both write and verify reverse index")); |
| 250 | |
| 251 | if (flags & WRITE_REV) { |
| 252 | if (!rev_name) { |
| 253 | struct strbuf tmp_file = STRBUF_INIT; |
| 254 | fd = odb_mkstemp(&tmp_file, "pack/tmp_rev_XXXXXX"); |
| 255 | rev_name = strbuf_detach(&tmp_file, NULL); |
| 256 | } else { |
| 257 | unlink(rev_name); |
René Scharfe | 66e905b | 2021-08-25 22:16:46 +0200 | [diff] [blame] | 258 | fd = xopen(rev_name, O_CREAT|O_EXCL|O_WRONLY, 0600); |
Taylor Blau | 8ef50d9 | 2021-01-25 18:37:18 -0500 | [diff] [blame] | 259 | } |
| 260 | f = hashfd(fd, rev_name); |
| 261 | } else if (flags & WRITE_REV_VERIFY) { |
| 262 | struct stat statbuf; |
| 263 | if (stat(rev_name, &statbuf)) { |
| 264 | if (errno == ENOENT) { |
| 265 | /* .rev files are optional */ |
| 266 | return NULL; |
| 267 | } else |
| 268 | die_errno(_("could not stat: %s"), rev_name); |
| 269 | } |
| 270 | f = hashfd_check(rev_name); |
| 271 | } else |
| 272 | return NULL; |
| 273 | |
| 274 | write_rev_header(f); |
| 275 | |
Taylor Blau | a587b5a | 2021-03-30 11:04:29 -0400 | [diff] [blame] | 276 | write_rev_index_positions(f, pack_order, nr_objects); |
Taylor Blau | 8ef50d9 | 2021-01-25 18:37:18 -0500 | [diff] [blame] | 277 | write_rev_trailer(f, hash); |
| 278 | |
| 279 | if (rev_name && adjust_shared_perm(rev_name) < 0) |
| 280 | die(_("failed to make %s readable"), rev_name); |
| 281 | |
Neeraj Singh | 020406e | 2022-03-10 22:43:21 +0000 | [diff] [blame] | 282 | finalize_hashfile(f, NULL, FSYNC_COMPONENT_PACK_METADATA, |
| 283 | CSUM_HASH_IN_STREAM | CSUM_CLOSE | |
| 284 | ((flags & WRITE_IDX_VERIFY) ? 0 : CSUM_FSYNC)); |
Taylor Blau | 8ef50d9 | 2021-01-25 18:37:18 -0500 | [diff] [blame] | 285 | |
| 286 | return rev_name; |
| 287 | } |
| 288 | |
Taylor Blau | 5dfaf49 | 2022-05-20 19:17:43 -0400 | [diff] [blame] | 289 | static void write_mtimes_header(struct hashfile *f) |
| 290 | { |
| 291 | hashwrite_be32(f, MTIMES_SIGNATURE); |
| 292 | hashwrite_be32(f, MTIMES_VERSION); |
| 293 | hashwrite_be32(f, oid_version(the_hash_algo)); |
| 294 | } |
| 295 | |
| 296 | /* |
| 297 | * Writes the object mtimes of "objects" for use in a .mtimes file. |
| 298 | * Note that objects must be in lexicographic (index) order, which is |
| 299 | * the expected ordering of these values in the .mtimes file. |
| 300 | */ |
| 301 | static void write_mtimes_objects(struct hashfile *f, |
| 302 | struct packing_data *to_pack, |
| 303 | struct pack_idx_entry **objects, |
| 304 | uint32_t nr_objects) |
| 305 | { |
| 306 | uint32_t i; |
| 307 | for (i = 0; i < nr_objects; i++) { |
| 308 | struct object_entry *e = (struct object_entry*)objects[i]; |
| 309 | hashwrite_be32(f, oe_cruft_mtime(to_pack, e)); |
| 310 | } |
| 311 | } |
| 312 | |
| 313 | static void write_mtimes_trailer(struct hashfile *f, const unsigned char *hash) |
| 314 | { |
| 315 | hashwrite(f, hash, the_hash_algo->rawsz); |
| 316 | } |
| 317 | |
Taylor Blau | c412583 | 2023-04-18 16:40:32 -0400 | [diff] [blame] | 318 | static char *write_mtimes_file(struct packing_data *to_pack, |
| 319 | struct pack_idx_entry **objects, |
| 320 | uint32_t nr_objects, |
| 321 | const unsigned char *hash) |
Taylor Blau | 5dfaf49 | 2022-05-20 19:17:43 -0400 | [diff] [blame] | 322 | { |
Derrick Stolee | 82db195 | 2022-06-16 13:13:49 +0000 | [diff] [blame] | 323 | struct strbuf tmp_file = STRBUF_INIT; |
Taylor Blau | c412583 | 2023-04-18 16:40:32 -0400 | [diff] [blame] | 324 | char *mtimes_name; |
Taylor Blau | 5dfaf49 | 2022-05-20 19:17:43 -0400 | [diff] [blame] | 325 | struct hashfile *f; |
| 326 | int fd; |
| 327 | |
| 328 | if (!to_pack) |
| 329 | BUG("cannot call write_mtimes_file with NULL packing_data"); |
| 330 | |
Derrick Stolee | 82db195 | 2022-06-16 13:13:49 +0000 | [diff] [blame] | 331 | fd = odb_mkstemp(&tmp_file, "pack/tmp_mtimes_XXXXXX"); |
| 332 | mtimes_name = strbuf_detach(&tmp_file, NULL); |
Taylor Blau | 5dfaf49 | 2022-05-20 19:17:43 -0400 | [diff] [blame] | 333 | f = hashfd(fd, mtimes_name); |
| 334 | |
| 335 | write_mtimes_header(f); |
| 336 | write_mtimes_objects(f, to_pack, objects, nr_objects); |
| 337 | write_mtimes_trailer(f, hash); |
| 338 | |
| 339 | if (adjust_shared_perm(mtimes_name) < 0) |
| 340 | die(_("failed to make %s readable"), mtimes_name); |
| 341 | |
| 342 | finalize_hashfile(f, NULL, FSYNC_COMPONENT_PACK_METADATA, |
| 343 | CSUM_HASH_IN_STREAM | CSUM_CLOSE | CSUM_FSYNC); |
| 344 | |
| 345 | return mtimes_name; |
| 346 | } |
| 347 | |
brian m. carlson | 98a3bea | 2018-02-01 02:18:46 +0000 | [diff] [blame] | 348 | off_t write_pack_header(struct hashfile *f, uint32_t nr_entries) |
Junio C Hamano | c0ad465 | 2011-10-28 11:40:48 -0700 | [diff] [blame] | 349 | { |
| 350 | struct pack_header hdr; |
| 351 | |
| 352 | hdr.hdr_signature = htonl(PACK_SIGNATURE); |
| 353 | hdr.hdr_version = htonl(PACK_VERSION); |
| 354 | hdr.hdr_entries = htonl(nr_entries); |
brian m. carlson | 98a3bea | 2018-02-01 02:18:46 +0000 | [diff] [blame] | 355 | hashwrite(f, &hdr, sizeof(hdr)); |
Junio C Hamano | c0ad465 | 2011-10-28 11:40:48 -0700 | [diff] [blame] | 356 | return sizeof(hdr); |
| 357 | } |
| 358 | |
Nicolas Pitre | abeb40e | 2008-08-29 16:07:59 -0400 | [diff] [blame] | 359 | /* |
| 360 | * Update pack header with object_count and compute new SHA1 for pack data |
| 361 | * associated to pack_fd, and write that SHA1 at the end. That new SHA1 |
| 362 | * is also returned in new_pack_sha1. |
| 363 | * |
| 364 | * If partial_pack_sha1 is non null, then the SHA1 of the existing pack |
| 365 | * (without the header update) is computed and validated against the |
| 366 | * one provided in partial_pack_sha1. The validation is performed at |
| 367 | * partial_pack_offset bytes in the pack file. The SHA1 of the remaining |
| 368 | * data (i.e. from partial_pack_offset to the end) is then computed and |
| 369 | * returned in partial_pack_sha1. |
| 370 | * |
| 371 | * Note that new_pack_sha1 is updated last, so both new_pack_sha1 and |
| 372 | * partial_pack_sha1 can refer to the same buffer if the caller is not |
| 373 | * interested in the resulting SHA1 of pack data above partial_pack_offset. |
| 374 | */ |
Dana L. How | 8b0eca7 | 2007-05-02 12:13:14 -0400 | [diff] [blame] | 375 | void fixup_pack_header_footer(int pack_fd, |
brian m. carlson | 81c58cd | 2018-02-01 02:18:44 +0000 | [diff] [blame] | 376 | unsigned char *new_pack_hash, |
Dana L. How | 8b0eca7 | 2007-05-02 12:13:14 -0400 | [diff] [blame] | 377 | const char *pack_name, |
Nicolas Pitre | abeb40e | 2008-08-29 16:07:59 -0400 | [diff] [blame] | 378 | uint32_t object_count, |
brian m. carlson | 81c58cd | 2018-02-01 02:18:44 +0000 | [diff] [blame] | 379 | unsigned char *partial_pack_hash, |
Nicolas Pitre | abeb40e | 2008-08-29 16:07:59 -0400 | [diff] [blame] | 380 | off_t partial_pack_offset) |
Dana L. How | 8b0eca7 | 2007-05-02 12:13:14 -0400 | [diff] [blame] | 381 | { |
Nicolas Pitre | d35825d | 2008-08-29 16:08:02 -0400 | [diff] [blame] | 382 | int aligned_sz, buf_sz = 8 * 1024; |
brian m. carlson | 81c58cd | 2018-02-01 02:18:44 +0000 | [diff] [blame] | 383 | git_hash_ctx old_hash_ctx, new_hash_ctx; |
Dana L. How | 8b0eca7 | 2007-05-02 12:13:14 -0400 | [diff] [blame] | 384 | struct pack_header hdr; |
| 385 | char *buf; |
Jeff King | 90dca67 | 2017-09-27 02:01:07 -0400 | [diff] [blame] | 386 | ssize_t read_result; |
Dana L. How | 8b0eca7 | 2007-05-02 12:13:14 -0400 | [diff] [blame] | 387 | |
brian m. carlson | 81c58cd | 2018-02-01 02:18:44 +0000 | [diff] [blame] | 388 | the_hash_algo->init_fn(&old_hash_ctx); |
| 389 | the_hash_algo->init_fn(&new_hash_ctx); |
Nicolas Pitre | abeb40e | 2008-08-29 16:07:59 -0400 | [diff] [blame] | 390 | |
Dana L. How | 8b0eca7 | 2007-05-02 12:13:14 -0400 | [diff] [blame] | 391 | if (lseek(pack_fd, 0, SEEK_SET) != 0) |
Thomas Rast | d824cbb | 2009-06-27 17:58:46 +0200 | [diff] [blame] | 392 | die_errno("Failed seeking to start of '%s'", pack_name); |
Jeff King | 90dca67 | 2017-09-27 02:01:07 -0400 | [diff] [blame] | 393 | read_result = read_in_full(pack_fd, &hdr, sizeof(hdr)); |
| 394 | if (read_result < 0) |
Thomas Rast | d824cbb | 2009-06-27 17:58:46 +0200 | [diff] [blame] | 395 | die_errno("Unable to reread header of '%s'", pack_name); |
Jeff King | 90dca67 | 2017-09-27 02:01:07 -0400 | [diff] [blame] | 396 | else if (read_result != sizeof(hdr)) |
| 397 | die_errno("Unexpected short read for header of '%s'", |
| 398 | pack_name); |
Dana L. How | 8b0eca7 | 2007-05-02 12:13:14 -0400 | [diff] [blame] | 399 | if (lseek(pack_fd, 0, SEEK_SET) != 0) |
Thomas Rast | d824cbb | 2009-06-27 17:58:46 +0200 | [diff] [blame] | 400 | die_errno("Failed seeking to start of '%s'", pack_name); |
brian m. carlson | 81c58cd | 2018-02-01 02:18:44 +0000 | [diff] [blame] | 401 | the_hash_algo->update_fn(&old_hash_ctx, &hdr, sizeof(hdr)); |
Dana L. How | 8b0eca7 | 2007-05-02 12:13:14 -0400 | [diff] [blame] | 402 | hdr.hdr_entries = htonl(object_count); |
brian m. carlson | 81c58cd | 2018-02-01 02:18:44 +0000 | [diff] [blame] | 403 | the_hash_algo->update_fn(&new_hash_ctx, &hdr, sizeof(hdr)); |
Dana L. How | 8b0eca7 | 2007-05-02 12:13:14 -0400 | [diff] [blame] | 404 | write_or_die(pack_fd, &hdr, sizeof(hdr)); |
Nicolas Pitre | abeb40e | 2008-08-29 16:07:59 -0400 | [diff] [blame] | 405 | partial_pack_offset -= sizeof(hdr); |
Dana L. How | 8b0eca7 | 2007-05-02 12:13:14 -0400 | [diff] [blame] | 406 | |
| 407 | buf = xmalloc(buf_sz); |
Nicolas Pitre | d35825d | 2008-08-29 16:08:02 -0400 | [diff] [blame] | 408 | aligned_sz = buf_sz - sizeof(hdr); |
Dana L. How | 8b0eca7 | 2007-05-02 12:13:14 -0400 | [diff] [blame] | 409 | for (;;) { |
Nicolas Pitre | abeb40e | 2008-08-29 16:07:59 -0400 | [diff] [blame] | 410 | ssize_t m, n; |
brian m. carlson | 81c58cd | 2018-02-01 02:18:44 +0000 | [diff] [blame] | 411 | m = (partial_pack_hash && partial_pack_offset < aligned_sz) ? |
Nicolas Pitre | d35825d | 2008-08-29 16:08:02 -0400 | [diff] [blame] | 412 | partial_pack_offset : aligned_sz; |
Nicolas Pitre | abeb40e | 2008-08-29 16:07:59 -0400 | [diff] [blame] | 413 | n = xread(pack_fd, buf, m); |
Dana L. How | 8b0eca7 | 2007-05-02 12:13:14 -0400 | [diff] [blame] | 414 | if (!n) |
| 415 | break; |
| 416 | if (n < 0) |
Thomas Rast | d824cbb | 2009-06-27 17:58:46 +0200 | [diff] [blame] | 417 | die_errno("Failed to checksum '%s'", pack_name); |
brian m. carlson | 81c58cd | 2018-02-01 02:18:44 +0000 | [diff] [blame] | 418 | the_hash_algo->update_fn(&new_hash_ctx, buf, n); |
Nicolas Pitre | abeb40e | 2008-08-29 16:07:59 -0400 | [diff] [blame] | 419 | |
Nicolas Pitre | d35825d | 2008-08-29 16:08:02 -0400 | [diff] [blame] | 420 | aligned_sz -= n; |
| 421 | if (!aligned_sz) |
| 422 | aligned_sz = buf_sz; |
| 423 | |
brian m. carlson | 81c58cd | 2018-02-01 02:18:44 +0000 | [diff] [blame] | 424 | if (!partial_pack_hash) |
Nicolas Pitre | abeb40e | 2008-08-29 16:07:59 -0400 | [diff] [blame] | 425 | continue; |
| 426 | |
brian m. carlson | 81c58cd | 2018-02-01 02:18:44 +0000 | [diff] [blame] | 427 | the_hash_algo->update_fn(&old_hash_ctx, buf, n); |
Nicolas Pitre | abeb40e | 2008-08-29 16:07:59 -0400 | [diff] [blame] | 428 | partial_pack_offset -= n; |
| 429 | if (partial_pack_offset == 0) { |
brian m. carlson | 81c58cd | 2018-02-01 02:18:44 +0000 | [diff] [blame] | 430 | unsigned char hash[GIT_MAX_RAWSZ]; |
| 431 | the_hash_algo->final_fn(hash, &old_hash_ctx); |
Jeff King | 67947c3 | 2018-08-28 17:22:52 -0400 | [diff] [blame] | 432 | if (!hasheq(hash, partial_pack_hash)) |
Nicolas Pitre | abeb40e | 2008-08-29 16:07:59 -0400 | [diff] [blame] | 433 | die("Unexpected checksum for %s " |
| 434 | "(disk corruption?)", pack_name); |
| 435 | /* |
| 436 | * Now let's compute the SHA1 of the remainder of the |
| 437 | * pack, which also means making partial_pack_offset |
| 438 | * big enough not to matter anymore. |
| 439 | */ |
brian m. carlson | 81c58cd | 2018-02-01 02:18:44 +0000 | [diff] [blame] | 440 | the_hash_algo->init_fn(&old_hash_ctx); |
Nicolas Pitre | abeb40e | 2008-08-29 16:07:59 -0400 | [diff] [blame] | 441 | partial_pack_offset = ~partial_pack_offset; |
| 442 | partial_pack_offset -= MSB(partial_pack_offset, 1); |
| 443 | } |
Dana L. How | 8b0eca7 | 2007-05-02 12:13:14 -0400 | [diff] [blame] | 444 | } |
| 445 | free(buf); |
| 446 | |
brian m. carlson | 81c58cd | 2018-02-01 02:18:44 +0000 | [diff] [blame] | 447 | if (partial_pack_hash) |
| 448 | the_hash_algo->final_fn(partial_pack_hash, &old_hash_ctx); |
| 449 | the_hash_algo->final_fn(new_pack_hash, &new_hash_ctx); |
| 450 | write_or_die(pack_fd, new_pack_hash, the_hash_algo->rawsz); |
Neeraj Singh | 020406e | 2022-03-10 22:43:21 +0000 | [diff] [blame] | 451 | fsync_component_or_die(FSYNC_COMPONENT_PACK, pack_fd, pack_name); |
Dana L. How | 8b0eca7 | 2007-05-02 12:13:14 -0400 | [diff] [blame] | 452 | } |
Shawn O. Pearce | 106764e | 2007-09-14 03:31:16 -0400 | [diff] [blame] | 453 | |
Jonathan Tan | 5476e1e | 2021-02-22 11:20:09 -0800 | [diff] [blame] | 454 | char *index_pack_lockfile(int ip_out, int *is_well_formed) |
Shawn O. Pearce | 106764e | 2007-09-14 03:31:16 -0400 | [diff] [blame] | 455 | { |
brian m. carlson | 81c58cd | 2018-02-01 02:18:44 +0000 | [diff] [blame] | 456 | char packname[GIT_MAX_HEXSZ + 6]; |
| 457 | const int len = the_hash_algo->hexsz + 6; |
Shawn O. Pearce | 106764e | 2007-09-14 03:31:16 -0400 | [diff] [blame] | 458 | |
| 459 | /* |
Junio C Hamano | 6e180cd | 2009-02-24 23:11:29 -0800 | [diff] [blame] | 460 | * The first thing we expect from index-pack's output |
Shawn O. Pearce | 106764e | 2007-09-14 03:31:16 -0400 | [diff] [blame] | 461 | * is "pack\t%40s\n" or "keep\t%40s\n" (46 bytes) where |
| 462 | * %40s is the newly created pack SHA1 name. In the "keep" |
| 463 | * case, we need it to remove the corresponding .keep file |
| 464 | * later on. If we don't get that then tough luck with it. |
| 465 | */ |
brian m. carlson | 81c58cd | 2018-02-01 02:18:44 +0000 | [diff] [blame] | 466 | if (read_in_full(ip_out, packname, len) == len && packname[len-1] == '\n') { |
René Scharfe | d773144 | 2014-08-30 11:47:19 +0200 | [diff] [blame] | 467 | const char *name; |
Jonathan Tan | 5476e1e | 2021-02-22 11:20:09 -0800 | [diff] [blame] | 468 | |
| 469 | if (is_well_formed) |
| 470 | *is_well_formed = 1; |
brian m. carlson | 81c58cd | 2018-02-01 02:18:44 +0000 | [diff] [blame] | 471 | packname[len-1] = 0; |
René Scharfe | d773144 | 2014-08-30 11:47:19 +0200 | [diff] [blame] | 472 | if (skip_prefix(packname, "keep\t", &name)) |
| 473 | return xstrfmt("%s/pack/pack-%s.keep", |
| 474 | get_object_directory(), name); |
Jonathan Tan | 5476e1e | 2021-02-22 11:20:09 -0800 | [diff] [blame] | 475 | return NULL; |
Shawn O. Pearce | 106764e | 2007-09-14 03:31:16 -0400 | [diff] [blame] | 476 | } |
Jonathan Tan | 5476e1e | 2021-02-22 11:20:09 -0800 | [diff] [blame] | 477 | if (is_well_formed) |
| 478 | *is_well_formed = 0; |
Shawn O. Pearce | 106764e | 2007-09-14 03:31:16 -0400 | [diff] [blame] | 479 | return NULL; |
| 480 | } |
Nicolas Pitre | f965c52 | 2010-02-23 15:02:37 -0500 | [diff] [blame] | 481 | |
| 482 | /* |
| 483 | * The per-object header is a pretty dense thing, which is |
| 484 | * - first byte: low four bits are "size", then three bits of "type", |
| 485 | * and the high bit is "size continues". |
| 486 | * - each byte afterwards: low seven bits are size continuation, |
| 487 | * with the high bit being "size continues" |
| 488 | */ |
Jeff King | 7202a6f | 2017-03-24 13:26:40 -0400 | [diff] [blame] | 489 | int encode_in_pack_object_header(unsigned char *hdr, int hdr_len, |
| 490 | enum object_type type, uintmax_t size) |
Nicolas Pitre | f965c52 | 2010-02-23 15:02:37 -0500 | [diff] [blame] | 491 | { |
| 492 | int n = 1; |
| 493 | unsigned char c; |
| 494 | |
| 495 | if (type < OBJ_COMMIT || type > OBJ_REF_DELTA) |
| 496 | die("bad type %d", type); |
| 497 | |
| 498 | c = (type << 4) | (size & 15); |
| 499 | size >>= 4; |
| 500 | while (size) { |
Jeff King | 7202a6f | 2017-03-24 13:26:40 -0400 | [diff] [blame] | 501 | if (n == hdr_len) |
| 502 | die("object size is too enormous to format"); |
Nicolas Pitre | f965c52 | 2010-02-23 15:02:37 -0500 | [diff] [blame] | 503 | *hdr++ = c | 0x80; |
| 504 | c = size & 0x7f; |
| 505 | size >>= 7; |
| 506 | n++; |
| 507 | } |
| 508 | *hdr = c; |
| 509 | return n; |
| 510 | } |
Junio C Hamano | cdf9db3 | 2011-10-28 11:52:14 -0700 | [diff] [blame] | 511 | |
brian m. carlson | 98a3bea | 2018-02-01 02:18:46 +0000 | [diff] [blame] | 512 | struct hashfile *create_tmp_packfile(char **pack_tmp_name) |
Junio C Hamano | cdf9db3 | 2011-10-28 11:52:14 -0700 | [diff] [blame] | 513 | { |
Jeff King | 594fa99 | 2017-03-28 15:45:43 -0400 | [diff] [blame] | 514 | struct strbuf tmpname = STRBUF_INIT; |
Junio C Hamano | cdf9db3 | 2011-10-28 11:52:14 -0700 | [diff] [blame] | 515 | int fd; |
| 516 | |
Jeff King | 594fa99 | 2017-03-28 15:45:43 -0400 | [diff] [blame] | 517 | fd = odb_mkstemp(&tmpname, "pack/tmp_pack_XXXXXX"); |
| 518 | *pack_tmp_name = strbuf_detach(&tmpname, NULL); |
brian m. carlson | 98a3bea | 2018-02-01 02:18:46 +0000 | [diff] [blame] | 519 | return hashfd(fd, *pack_tmp_name); |
Junio C Hamano | cdf9db3 | 2011-10-28 11:52:14 -0700 | [diff] [blame] | 520 | } |
Junio C Hamano | 0e99053 | 2011-10-28 12:34:09 -0700 | [diff] [blame] | 521 | |
Ævar Arnfjörð Bjarmason | 66833f0 | 2021-09-09 19:24:37 -0400 | [diff] [blame] | 522 | static void rename_tmp_packfile(struct strbuf *name_prefix, const char *source, |
| 523 | const char *ext) |
| 524 | { |
| 525 | size_t name_prefix_len = name_prefix->len; |
| 526 | |
| 527 | strbuf_addstr(name_prefix, ext); |
| 528 | if (rename(source, name_prefix->buf)) |
| 529 | die_errno("unable to rename temporary file to '%s'", |
| 530 | name_prefix->buf); |
| 531 | strbuf_setlen(name_prefix, name_prefix_len); |
| 532 | } |
| 533 | |
Ævar Arnfjörð Bjarmason | 2ec02dd | 2021-09-09 19:24:56 -0400 | [diff] [blame] | 534 | void rename_tmp_packfile_idx(struct strbuf *name_buffer, |
| 535 | char **idx_tmp_name) |
| 536 | { |
| 537 | rename_tmp_packfile(name_buffer, *idx_tmp_name, "idx"); |
| 538 | } |
| 539 | |
| 540 | void stage_tmp_packfiles(struct strbuf *name_buffer, |
Junio C Hamano | 0e99053 | 2011-10-28 12:34:09 -0700 | [diff] [blame] | 541 | const char *pack_tmp_name, |
| 542 | struct pack_idx_entry **written_list, |
| 543 | uint32_t nr_written, |
Taylor Blau | 1c573cd | 2022-05-20 19:17:38 -0400 | [diff] [blame] | 544 | struct packing_data *to_pack, |
Junio C Hamano | 0e99053 | 2011-10-28 12:34:09 -0700 | [diff] [blame] | 545 | struct pack_idx_option *pack_idx_opts, |
Ævar Arnfjörð Bjarmason | 2ec02dd | 2021-09-09 19:24:56 -0400 | [diff] [blame] | 546 | unsigned char hash[], |
| 547 | char **idx_tmp_name) |
Junio C Hamano | 0e99053 | 2011-10-28 12:34:09 -0700 | [diff] [blame] | 548 | { |
Ævar Arnfjörð Bjarmason | 2ec02dd | 2021-09-09 19:24:56 -0400 | [diff] [blame] | 549 | const char *rev_tmp_name = NULL; |
Taylor Blau | c412583 | 2023-04-18 16:40:32 -0400 | [diff] [blame] | 550 | char *mtimes_tmp_name = NULL; |
Junio C Hamano | 0e99053 | 2011-10-28 12:34:09 -0700 | [diff] [blame] | 551 | |
| 552 | if (adjust_shared_perm(pack_tmp_name)) |
| 553 | die_errno("unable to make temporary pack file readable"); |
| 554 | |
Ævar Arnfjörð Bjarmason | 2ec02dd | 2021-09-09 19:24:56 -0400 | [diff] [blame] | 555 | *idx_tmp_name = (char *)write_idx_file(NULL, written_list, nr_written, |
| 556 | pack_idx_opts, hash); |
| 557 | if (adjust_shared_perm(*idx_tmp_name)) |
Junio C Hamano | 0e99053 | 2011-10-28 12:34:09 -0700 | [diff] [blame] | 558 | die_errno("unable to make temporary index file readable"); |
| 559 | |
Taylor Blau | 8ef50d9 | 2021-01-25 18:37:18 -0500 | [diff] [blame] | 560 | rev_tmp_name = write_rev_file(NULL, written_list, nr_written, hash, |
| 561 | pack_idx_opts->flags); |
| 562 | |
Taylor Blau | 5dfaf49 | 2022-05-20 19:17:43 -0400 | [diff] [blame] | 563 | if (pack_idx_opts->flags & WRITE_MTIMES) { |
Derrick Stolee | 82db195 | 2022-06-16 13:13:49 +0000 | [diff] [blame] | 564 | mtimes_tmp_name = write_mtimes_file(to_pack, written_list, |
Taylor Blau | 5dfaf49 | 2022-05-20 19:17:43 -0400 | [diff] [blame] | 565 | nr_written, |
| 566 | hash); |
| 567 | } |
| 568 | |
Ævar Arnfjörð Bjarmason | 66833f0 | 2021-09-09 19:24:37 -0400 | [diff] [blame] | 569 | rename_tmp_packfile(name_buffer, pack_tmp_name, "pack"); |
Ævar Arnfjörð Bjarmason | 66833f0 | 2021-09-09 19:24:37 -0400 | [diff] [blame] | 570 | if (rev_tmp_name) |
| 571 | rename_tmp_packfile(name_buffer, rev_tmp_name, "rev"); |
Taylor Blau | 5dfaf49 | 2022-05-20 19:17:43 -0400 | [diff] [blame] | 572 | if (mtimes_tmp_name) |
| 573 | rename_tmp_packfile(name_buffer, mtimes_tmp_name, "mtimes"); |
Taylor Blau | 3969e6c | 2023-04-12 18:20:17 -0400 | [diff] [blame] | 574 | |
| 575 | free((char *)rev_tmp_name); |
Taylor Blau | c412583 | 2023-04-18 16:40:32 -0400 | [diff] [blame] | 576 | free(mtimes_tmp_name); |
Junio C Hamano | 0e99053 | 2011-10-28 12:34:09 -0700 | [diff] [blame] | 577 | } |
Christian Couder | 33add2a | 2021-01-12 09:21:59 +0100 | [diff] [blame] | 578 | |
| 579 | void write_promisor_file(const char *promisor_name, struct ref **sought, int nr_sought) |
| 580 | { |
Christian Couder | 7c99bc2 | 2021-01-14 16:50:16 +0100 | [diff] [blame] | 581 | int i, err; |
Christian Couder | 33add2a | 2021-01-12 09:21:59 +0100 | [diff] [blame] | 582 | FILE *output = xfopen(promisor_name, "w"); |
| 583 | |
| 584 | for (i = 0; i < nr_sought; i++) |
| 585 | fprintf(output, "%s %s\n", oid_to_hex(&sought[i]->old_oid), |
| 586 | sought[i]->name); |
Christian Couder | 7c99bc2 | 2021-01-14 16:50:16 +0100 | [diff] [blame] | 587 | |
| 588 | err = ferror(output); |
| 589 | err |= fclose(output); |
| 590 | if (err) |
| 591 | die(_("could not write '%s' promisor file"), promisor_name); |
Christian Couder | 33add2a | 2021-01-12 09:21:59 +0100 | [diff] [blame] | 592 | } |