Linus Torvalds | 908e531 | 2005-12-24 13:50:45 -0800 | [diff] [blame] | 1 | #include "cache.h" |
| 2 | #include "commit.h" |
Junio C Hamano | 635d413 | 2005-12-27 14:36:49 -0800 | [diff] [blame] | 3 | #include "tag.h" |
Linus Torvalds | 908e531 | 2005-12-24 13:50:45 -0800 | [diff] [blame] | 4 | #include "refs.h" |
| 5 | |
| 6 | #define SEEN (1u << 0) |
| 7 | |
Junio C Hamano | 2d9e7c9 | 2005-12-27 14:40:17 -0800 | [diff] [blame] | 8 | static const char describe_usage[] = |
| 9 | "git-describe [--all] [--tags] [--abbrev=<n>] <committish>*"; |
Linus Torvalds | 908e531 | 2005-12-24 13:50:45 -0800 | [diff] [blame] | 10 | |
David Rientjes | 96f1e58 | 2006-08-15 10:23:48 -0700 | [diff] [blame] | 11 | static int all; /* Default to annotated tags only */ |
| 12 | static int tags; /* But allow any tags if --tags is specified */ |
Junio C Hamano | 2d9e7c9 | 2005-12-27 14:40:17 -0800 | [diff] [blame] | 13 | |
Junio C Hamano | 2d9e7c9 | 2005-12-27 14:40:17 -0800 | [diff] [blame] | 14 | static int abbrev = DEFAULT_ABBREV; |
Linus Torvalds | 908e531 | 2005-12-24 13:50:45 -0800 | [diff] [blame] | 15 | |
David Rientjes | 96f1e58 | 2006-08-15 10:23:48 -0700 | [diff] [blame] | 16 | static int names, allocs; |
Linus Torvalds | 908e531 | 2005-12-24 13:50:45 -0800 | [diff] [blame] | 17 | static struct commit_name { |
| 18 | const struct commit *commit; |
Junio C Hamano | 64deb85 | 2005-12-27 16:09:37 -0800 | [diff] [blame] | 19 | int prio; /* annotated tag = 2, tag = 1, head = 0 */ |
Junio C Hamano | 5a2282d | 2006-01-08 14:22:19 -0800 | [diff] [blame] | 20 | char path[FLEX_ARRAY]; /* more */ |
Linus Torvalds | 908e531 | 2005-12-24 13:50:45 -0800 | [diff] [blame] | 21 | } **name_array = NULL; |
| 22 | |
| 23 | static struct commit_name *match(struct commit *cmit) |
| 24 | { |
| 25 | int i = names; |
| 26 | struct commit_name **p = name_array; |
| 27 | |
| 28 | while (i-- > 0) { |
| 29 | struct commit_name *n = *p++; |
| 30 | if (n->commit == cmit) |
| 31 | return n; |
| 32 | } |
| 33 | return NULL; |
| 34 | } |
| 35 | |
Junio C Hamano | 64deb85 | 2005-12-27 16:09:37 -0800 | [diff] [blame] | 36 | static void add_to_known_names(const char *path, |
| 37 | const struct commit *commit, |
| 38 | int prio) |
Linus Torvalds | 908e531 | 2005-12-24 13:50:45 -0800 | [diff] [blame] | 39 | { |
| 40 | int idx; |
| 41 | int len = strlen(path)+1; |
| 42 | struct commit_name *name = xmalloc(sizeof(struct commit_name) + len); |
| 43 | |
| 44 | name->commit = commit; |
Jonas Fonseca | f712226 | 2006-08-25 02:48:04 +0200 | [diff] [blame] | 45 | name->prio = prio; |
Linus Torvalds | 908e531 | 2005-12-24 13:50:45 -0800 | [diff] [blame] | 46 | memcpy(name->path, path, len); |
| 47 | idx = names; |
| 48 | if (idx >= allocs) { |
| 49 | allocs = (idx + 50) * 3 / 2; |
| 50 | name_array = xrealloc(name_array, allocs*sizeof(*name_array)); |
| 51 | } |
| 52 | name_array[idx] = name; |
| 53 | names = ++idx; |
| 54 | } |
| 55 | |
Junio C Hamano | 8da1977 | 2006-09-20 22:02:01 -0700 | [diff] [blame] | 56 | static int get_name(const char *path, const unsigned char *sha1, int flag, void *cb_data) |
Linus Torvalds | 908e531 | 2005-12-24 13:50:45 -0800 | [diff] [blame] | 57 | { |
| 58 | struct commit *commit = lookup_commit_reference_gently(sha1, 1); |
Junio C Hamano | 64deb85 | 2005-12-27 16:09:37 -0800 | [diff] [blame] | 59 | struct object *object; |
| 60 | int prio; |
| 61 | |
Linus Torvalds | 908e531 | 2005-12-24 13:50:45 -0800 | [diff] [blame] | 62 | if (!commit) |
| 63 | return 0; |
Junio C Hamano | 64deb85 | 2005-12-27 16:09:37 -0800 | [diff] [blame] | 64 | object = parse_object(sha1); |
Junio C Hamano | 2d9e7c9 | 2005-12-27 14:40:17 -0800 | [diff] [blame] | 65 | /* If --all, then any refs are used. |
| 66 | * If --tags, then any tags are used. |
| 67 | * Otherwise only annotated tags are used. |
| 68 | */ |
Junio C Hamano | 64deb85 | 2005-12-27 16:09:37 -0800 | [diff] [blame] | 69 | if (!strncmp(path, "refs/tags/", 10)) { |
Linus Torvalds | 1974632 | 2006-07-11 20:45:31 -0700 | [diff] [blame] | 70 | if (object->type == OBJ_TAG) |
Junio C Hamano | 64deb85 | 2005-12-27 16:09:37 -0800 | [diff] [blame] | 71 | prio = 2; |
| 72 | else |
| 73 | prio = 1; |
Junio C Hamano | 635d413 | 2005-12-27 14:36:49 -0800 | [diff] [blame] | 74 | } |
Junio C Hamano | 64deb85 | 2005-12-27 16:09:37 -0800 | [diff] [blame] | 75 | else |
| 76 | prio = 0; |
| 77 | |
| 78 | if (!all) { |
| 79 | if (!prio) |
| 80 | return 0; |
| 81 | if (!tags && prio < 2) |
| 82 | return 0; |
| 83 | } |
| 84 | add_to_known_names(all ? path + 5 : path + 10, commit, prio); |
Linus Torvalds | 908e531 | 2005-12-24 13:50:45 -0800 | [diff] [blame] | 85 | return 0; |
| 86 | } |
| 87 | |
| 88 | static int compare_names(const void *_a, const void *_b) |
| 89 | { |
| 90 | struct commit_name *a = *(struct commit_name **)_a; |
| 91 | struct commit_name *b = *(struct commit_name **)_b; |
| 92 | unsigned long a_date = a->commit->date; |
| 93 | unsigned long b_date = b->commit->date; |
Junio C Hamano | 64deb85 | 2005-12-27 16:09:37 -0800 | [diff] [blame] | 94 | |
| 95 | if (a->prio != b->prio) |
| 96 | return b->prio - a->prio; |
Linus Torvalds | 908e531 | 2005-12-24 13:50:45 -0800 | [diff] [blame] | 97 | return (a_date > b_date) ? -1 : (a_date == b_date) ? 0 : 1; |
| 98 | } |
| 99 | |
Timo Hirvonen | 554fe20 | 2006-06-28 12:04:39 +0300 | [diff] [blame] | 100 | static void describe(const char *arg, int last_one) |
Linus Torvalds | 908e531 | 2005-12-24 13:50:45 -0800 | [diff] [blame] | 101 | { |
Junio C Hamano | 4c34a2c | 2006-01-11 13:57:42 -0800 | [diff] [blame] | 102 | unsigned char sha1[20]; |
| 103 | struct commit *cmit; |
Linus Torvalds | 908e531 | 2005-12-24 13:50:45 -0800 | [diff] [blame] | 104 | struct commit_list *list; |
| 105 | static int initialized = 0; |
| 106 | struct commit_name *n; |
| 107 | |
Dmitry V. Levin | 31fff30 | 2006-05-09 01:43:38 +0400 | [diff] [blame] | 108 | if (get_sha1(arg, sha1)) |
| 109 | die("Not a valid object name %s", arg); |
Junio C Hamano | 4c34a2c | 2006-01-11 13:57:42 -0800 | [diff] [blame] | 110 | cmit = lookup_commit_reference(sha1); |
| 111 | if (!cmit) |
Dmitry V. Levin | 31fff30 | 2006-05-09 01:43:38 +0400 | [diff] [blame] | 112 | die("%s is not a valid '%s' object", arg, commit_type); |
Junio C Hamano | 4c34a2c | 2006-01-11 13:57:42 -0800 | [diff] [blame] | 113 | |
Linus Torvalds | 908e531 | 2005-12-24 13:50:45 -0800 | [diff] [blame] | 114 | if (!initialized) { |
| 115 | initialized = 1; |
Junio C Hamano | cb5d709 | 2006-09-20 21:47:42 -0700 | [diff] [blame] | 116 | for_each_ref(get_name, NULL); |
Linus Torvalds | 908e531 | 2005-12-24 13:50:45 -0800 | [diff] [blame] | 117 | qsort(name_array, names, sizeof(*name_array), compare_names); |
| 118 | } |
| 119 | |
| 120 | n = match(cmit); |
| 121 | if (n) { |
| 122 | printf("%s\n", n->path); |
| 123 | return; |
| 124 | } |
| 125 | |
| 126 | list = NULL; |
| 127 | commit_list_insert(cmit, &list); |
| 128 | while (list) { |
| 129 | struct commit *c = pop_most_recent_commit(&list, SEEN); |
| 130 | n = match(c); |
| 131 | if (n) { |
Junio C Hamano | 4cdf78b | 2005-12-27 14:49:22 -0800 | [diff] [blame] | 132 | printf("%s-g%s\n", n->path, |
Junio C Hamano | 2d9e7c9 | 2005-12-27 14:40:17 -0800 | [diff] [blame] | 133 | find_unique_abbrev(cmit->object.sha1, abbrev)); |
Junio C Hamano | 181dc77 | 2006-01-15 22:15:37 -0800 | [diff] [blame] | 134 | if (!last_one) |
| 135 | clear_commit_marks(cmit, SEEN); |
Junio C Hamano | 8c23b6f | 2006-01-11 13:41:25 -0800 | [diff] [blame] | 136 | return; |
Linus Torvalds | 908e531 | 2005-12-24 13:50:45 -0800 | [diff] [blame] | 137 | } |
| 138 | } |
Junio C Hamano | 8c23b6f | 2006-01-11 13:41:25 -0800 | [diff] [blame] | 139 | die("cannot describe '%s'", sha1_to_hex(cmit->object.sha1)); |
Linus Torvalds | 908e531 | 2005-12-24 13:50:45 -0800 | [diff] [blame] | 140 | } |
| 141 | |
| 142 | int main(int argc, char **argv) |
| 143 | { |
| 144 | int i; |
| 145 | |
| 146 | for (i = 1; i < argc; i++) { |
| 147 | const char *arg = argv[i]; |
Linus Torvalds | 908e531 | 2005-12-24 13:50:45 -0800 | [diff] [blame] | 148 | |
Junio C Hamano | 4c34a2c | 2006-01-11 13:57:42 -0800 | [diff] [blame] | 149 | if (*arg != '-') |
| 150 | break; |
| 151 | else if (!strcmp(arg, "--all")) |
Linus Torvalds | 908e531 | 2005-12-24 13:50:45 -0800 | [diff] [blame] | 152 | all = 1; |
Junio C Hamano | 4c34a2c | 2006-01-11 13:57:42 -0800 | [diff] [blame] | 153 | else if (!strcmp(arg, "--tags")) |
Junio C Hamano | 2d9e7c9 | 2005-12-27 14:40:17 -0800 | [diff] [blame] | 154 | tags = 1; |
Junio C Hamano | 4c34a2c | 2006-01-11 13:57:42 -0800 | [diff] [blame] | 155 | else if (!strncmp(arg, "--abbrev=", 9)) { |
Junio C Hamano | 2d9e7c9 | 2005-12-27 14:40:17 -0800 | [diff] [blame] | 156 | abbrev = strtoul(arg + 9, NULL, 10); |
Jonas Fonseca | f712226 | 2006-08-25 02:48:04 +0200 | [diff] [blame] | 157 | if (abbrev < MINIMUM_ABBREV || 40 < abbrev) |
Junio C Hamano | 2d9e7c9 | 2005-12-27 14:40:17 -0800 | [diff] [blame] | 158 | abbrev = DEFAULT_ABBREV; |
Junio C Hamano | 2d9e7c9 | 2005-12-27 14:40:17 -0800 | [diff] [blame] | 159 | } |
Junio C Hamano | 4c34a2c | 2006-01-11 13:57:42 -0800 | [diff] [blame] | 160 | else |
Linus Torvalds | 908e531 | 2005-12-24 13:50:45 -0800 | [diff] [blame] | 161 | usage(describe_usage); |
Linus Torvalds | 908e531 | 2005-12-24 13:50:45 -0800 | [diff] [blame] | 162 | } |
Junio C Hamano | 4c34a2c | 2006-01-11 13:57:42 -0800 | [diff] [blame] | 163 | |
Dmitry V. Levin | 8112894 | 2006-09-14 05:03:59 +0400 | [diff] [blame] | 164 | setup_git_directory(); |
| 165 | |
Dmitry V. Levin | 5b6df8e | 2006-09-14 05:04:09 +0400 | [diff] [blame] | 166 | if (argc <= i) |
Junio C Hamano | fec9ebf | 2006-01-15 22:25:35 -0800 | [diff] [blame] | 167 | describe("HEAD", 1); |
Junio C Hamano | 4c34a2c | 2006-01-11 13:57:42 -0800 | [diff] [blame] | 168 | else |
Junio C Hamano | fec9ebf | 2006-01-15 22:25:35 -0800 | [diff] [blame] | 169 | while (i < argc) { |
| 170 | describe(argv[i], (i == argc - 1)); |
| 171 | i++; |
| 172 | } |
Junio C Hamano | 4c34a2c | 2006-01-11 13:57:42 -0800 | [diff] [blame] | 173 | |
Linus Torvalds | 908e531 | 2005-12-24 13:50:45 -0800 | [diff] [blame] | 174 | return 0; |
| 175 | } |