6 #include "parse-options.h"
9 struct ref_to_prune
*next
;
10 unsigned char sha1
[20];
11 char name
[FLEX_ARRAY
];
14 #define PACK_REFS_PRUNE 0x0001
15 #define PACK_REFS_ALL 0x0002
17 struct pack_refs_cb_data
{
19 struct ref_to_prune
*ref_to_prune
;
23 static int do_not_prune(int flags
)
25 /* If it is already packed or if it is a symref,
28 return (flags
& (REF_ISSYMREF
|REF_ISPACKED
));
31 static int handle_one_ref(const char *path
, const unsigned char *sha1
,
32 int flags
, void *cb_data
)
34 struct pack_refs_cb_data
*cb
= cb_data
;
37 /* Do not pack the symbolic refs */
38 if ((flags
& REF_ISSYMREF
))
40 is_tag_ref
= !prefixcmp(path
, "refs/tags/");
42 /* ALWAYS pack refs that were already packed or are tags */
43 if (!(cb
->flags
& PACK_REFS_ALL
) && !is_tag_ref
&& !(flags
& REF_ISPACKED
))
46 fprintf(cb
->refs_file
, "%s %s\n", sha1_to_hex(sha1
), path
);
48 struct object
*o
= parse_object(sha1
);
49 if (o
->type
== OBJ_TAG
) {
50 o
= deref_tag(o
, path
, 0);
52 fprintf(cb
->refs_file
, "^%s\n",
53 sha1_to_hex(o
->sha1
));
57 if ((cb
->flags
& PACK_REFS_PRUNE
) && !do_not_prune(flags
)) {
58 int namelen
= strlen(path
) + 1;
59 struct ref_to_prune
*n
= xcalloc(1, sizeof(*n
) + namelen
);
60 hashcpy(n
->sha1
, sha1
);
61 strcpy(n
->name
, path
);
62 n
->next
= cb
->ref_to_prune
;
68 /* make sure nobody touched the ref, and unlink */
69 static void prune_ref(struct ref_to_prune
*r
)
71 struct ref_lock
*lock
= lock_ref_sha1(r
->name
+ 5, r
->sha1
);
74 unlink(git_path("%s", r
->name
));
79 static void prune_refs(struct ref_to_prune
*r
)
87 static struct lock_file packed
;
89 static int pack_refs(unsigned int flags
)
92 struct pack_refs_cb_data cbdata
;
94 memset(&cbdata
, 0, sizeof(cbdata
));
97 fd
= hold_lock_file_for_update(&packed
, git_path("packed-refs"), 1);
98 cbdata
.refs_file
= fdopen(fd
, "w");
99 if (!cbdata
.refs_file
)
100 die("unable to create ref-pack file structure (%s)",
103 /* perhaps other traits later as well */
104 fprintf(cbdata
.refs_file
, "# pack-refs with: peeled \n");
106 for_each_ref(handle_one_ref
, &cbdata
);
107 if (ferror(cbdata
.refs_file
))
108 die("failed to write ref-pack file");
109 if (fflush(cbdata
.refs_file
) || fsync(fd
) || fclose(cbdata
.refs_file
))
110 die("failed to write ref-pack file (%s)", strerror(errno
));
112 * Since the lock file was fdopen()'ed and then fclose()'ed above,
113 * assign -1 to the lock file descriptor so that commit_lock_file()
114 * won't try to close() it.
117 if (commit_lock_file(&packed
) < 0)
118 die("unable to overwrite old ref-pack file (%s)", strerror(errno
));
119 if (cbdata
.flags
& PACK_REFS_PRUNE
)
120 prune_refs(cbdata
.ref_to_prune
);
124 static char const * const pack_refs_usage
[] = {
125 "git-pack-refs [options]",
129 int cmd_pack_refs(int argc
, const char **argv
, const char *prefix
)
131 unsigned int flags
= PACK_REFS_PRUNE
;
132 struct option opts
[] = {
133 OPT_BIT(0, "all", &flags
, "pack everything", PACK_REFS_ALL
),
134 OPT_BIT(0, "prune", &flags
, "prune loose refs (default)", PACK_REFS_PRUNE
),
137 if (parse_options(argc
, argv
, opts
, pack_refs_usage
, 0))
138 usage_with_options(pack_refs_usage
, opts
);
139 return pack_refs(flags
);