Skip to content
Navigation Menu
Toggle navigation
Sign in
In this repository
All GitHub Enterprise
↵
Jump to
↵
No suggested jump to results
In this repository
All GitHub Enterprise
↵
Jump to
↵
In this organization
All GitHub Enterprise
↵
Jump to
↵
In this repository
All GitHub Enterprise
↵
Jump to
↵
Sign in
Reseting focus
You signed in with another tab or window.
Reload
to refresh your session.
You signed out in another tab or window.
Reload
to refresh your session.
You switched accounts on another tab or window.
Reload
to refresh your session.
Dismiss alert
{{ message }}
git-mirror
/
git
Public
Notifications
You must be signed in to change notification settings
Fork
0
Star
0
Code
Issues
0
Pull requests
0
Actions
Projects
0
Security
Insights
Additional navigation options
Code
Issues
Pull requests
Actions
Projects
Security
Insights
Files
c976d41
Documentation
arm
compat
contrib
gitweb
mozilla-sha1
perl
ppc
t
templates
xdiff
.gitignore
COPYING
GIT-VERSION-GEN
INSTALL
Makefile
README
alloc.c
archive-tar.c
archive-zip.c
archive.h
base85.c
blob.c
blob.h
builtin-add.c
builtin-annotate.c
builtin-apply.c
builtin-archive.c
builtin-blame.c
builtin-branch.c
builtin-cat-file.c
builtin-check-ref-format.c
builtin-checkout-index.c
builtin-commit-tree.c
builtin-count-objects.c
builtin-diff-files.c
builtin-diff-index.c
builtin-diff-stages.c
builtin-diff-tree.c
builtin-diff.c
builtin-fmt-merge-msg.c
builtin-for-each-ref.c
builtin-grep.c
builtin-init-db.c
builtin-log.c
builtin-ls-files.c
builtin-ls-tree.c
builtin-mailinfo.c
builtin-mailsplit.c
builtin-mv.c
builtin-name-rev.c
builtin-pack-objects.c
builtin-pack-refs.c
builtin-prune-packed.c
builtin-prune.c
builtin-push.c
builtin-read-tree.c
builtin-repo-config.c
builtin-rev-list.c
builtin-rev-parse.c
builtin-rm.c
builtin-runstatus.c
builtin-shortlog.c
builtin-show-branch.c
builtin-show-ref.c
builtin-stripspace.c
builtin-symbolic-ref.c
builtin-tar-tree.c
builtin-unpack-objects.c
builtin-update-index.c
builtin-update-ref.c
builtin-upload-archive.c
builtin-verify-pack.c
builtin-write-tree.c
builtin.h
cache-tree.c
cache-tree.h
cache.h
check-builtins.sh
check-racy.c
color.c
color.h
combine-diff.c
commit.c
commit.h
config.c
config.mak.in
configure.ac
connect.c
convert-objects.c
copy.c
csum-file.c
csum-file.h
ctype.c
daemon.c
date.c
delta.h
describe.c
diff-delta.c
diff-lib.c
diff.c
diff.h
diffcore-break.c
diffcore-delta.c
diffcore-order.c
diffcore-pickaxe.c
diffcore-rename.c
diffcore.h
dir.c
dir.h
dump-cache-tree.c
entry.c
environment.c
exec_cmd.c
exec_cmd.h
fetch-pack.c
fetch.c
fetch.h
fsck-objects.c
generate-cmdlist.sh
git-am.sh
git-applymbox.sh
git-applypatch.sh
git-archimport.perl
git-bisect.sh
git-checkout.sh
git-clean.sh
git-clone.sh
git-commit.sh
git-compat-util.h
git-cvsexportcommit.perl
git-cvsimport.perl
git-cvsserver.perl
git-fetch.sh
git-instaweb.sh
git-lost-found.sh
git-ls-remote.sh
git-merge-octopus.sh
git-merge-one-file.sh
git-merge-ours.sh
git-merge-resolve.sh
git-merge-stupid.sh
git-merge.sh
git-p4import.py
git-parse-remote.sh
git-pull.sh
git-quiltimport.sh
git-rebase.sh
git-relink.perl
git-repack.sh
git-request-pull.sh
git-rerere.perl
git-reset.sh
git-resolve.sh
git-revert.sh
git-send-email.perl
git-sh-setup.sh
git-svn.perl
git-svnimport.perl
git-tag.sh
git-verify-tag.sh
git.c
git.spec.in
gitk
grep.c
grep.h
hash-object.c
help.c
http-fetch.c
http-push.c
http.c
http.h
ident.c
imap-send.c
index-pack.c
interpolate.c
interpolate.h
list-objects.c
list-objects.h
local-fetch.c
lockfile.c
log-tree.c
log-tree.h
merge-base.c
merge-file.c
merge-index.c
merge-recursive.c
merge-tree.c
mktag.c
mktree.c
object-refs.c
object.c
object.h
pack-check.c
pack-redundant.c
pack.h
pager.c
patch-delta.c
patch-id.c
path-list.c
path-list.h
path.c
peek-remote.c
pkt-line.c
pkt-line.h
quote.c
quote.h
read-cache.c
receive-pack.c
refs.c
refs.h
revision.c
revision.h
rsh.c
rsh.h
run-command.c
run-command.h
send-pack.c
server-info.c
setup.c
sha1_file.c
sha1_name.c
shell.c
show-index.c
sideband.c
sideband.h
ssh-fetch.c
ssh-pull.c
ssh-push.c
ssh-upload.c
strbuf.c
strbuf.h
tag.c
tag.h
tar.h
test-date.c
test-delta.c
test-sha1.c
test-sha1.sh
trace.c
tree-diff.c
tree-walk.c
tree-walk.h
tree.c
tree.h
unpack-file.c
unpack-trees.c
unpack-trees.h
update-server-info.c
upload-pack.c
usage.c
var.c
write_or_die.c
wt-status.c
wt-status.h
xdiff-interface.c
xdiff-interface.h
Breadcrumbs
git
/
builtin-branch.c
Blame
Blame
Latest commit
History
History
367 lines (306 loc) · 8.64 KB
Breadcrumbs
git
/
builtin-branch.c
Top
File metadata and controls
Code
Blame
367 lines (306 loc) · 8.64 KB
Raw
/* * Builtin "git branch" * * Copyright (c) 2006 Kristian Høgsberg <krh@redhat.com> * Based on git-branch.sh by Junio C Hamano. */ #include "cache.h" #include "refs.h" #include "commit.h" #include "builtin.h" static const char builtin_branch_usage[] = "git-branch (-d | -D) <branchname> | [-l] [-f] <branchname> [<start-point>] | (-m | -M) [<oldbranch>] <newbranch> | [-r | -a] [-v [--abbrev=<length>]]"; static const char *head; static unsigned char head_sha1[20]; static int in_merge_bases(const unsigned char *sha1, struct commit *rev1, struct commit *rev2) { struct commit_list *bases, *b; int ret = 0; bases = get_merge_bases(rev1, rev2, 1); for (b = bases; b; b = b->next) { if (!hashcmp(sha1, b->item->object.sha1)) { ret = 1; break; } } free_commit_list(bases); return ret; } static void delete_branches(int argc, const char **argv, int force) { struct commit *rev, *head_rev = head_rev; unsigned char sha1[20]; char *name; int i; if (!force) { head_rev = lookup_commit_reference(head_sha1); if (!head_rev) die("Couldn't look up commit object for HEAD"); } for (i = 0; i < argc; i++) { if (!strcmp(head, argv[i])) die("Cannot delete the branch you are currently on."); name = xstrdup(mkpath("refs/heads/%s", argv[i])); if (!resolve_ref(name, sha1, 1, NULL)) die("Branch '%s' not found.", argv[i]); rev = lookup_commit_reference(sha1); if (!rev) die("Couldn't look up commit object for '%s'", name); /* This checks whether the merge bases of branch and * HEAD contains branch -- which means that the HEAD * contains everything in both. */ if (!force && !in_merge_bases(sha1, rev, head_rev)) { fprintf(stderr, "The branch '%s' is not a strict subset of your current HEAD.\n" "If you are sure you want to delete it, run 'git branch -D %s'.\n", argv[i], argv[i]); exit(1); } if (delete_ref(name, sha1)) printf("Error deleting branch '%s'\n", argv[i]); else printf("Deleted branch %s.\n", argv[i]); free(name); } } #define REF_UNKNOWN_TYPE 0x00 #define REF_LOCAL_BRANCH 0x01 #define REF_REMOTE_BRANCH 0x02 #define REF_TAG 0x04 struct ref_item { char *name; unsigned int kind; unsigned char sha1[20]; }; struct ref_list { int index, alloc, maxwidth; struct ref_item *list; int kinds; }; static int append_ref(const char *refname, const unsigned char *sha1, int flags, void *cb_data) { struct ref_list *ref_list = (struct ref_list*)(cb_data); struct ref_item *newitem; int kind = REF_UNKNOWN_TYPE; int len; /* Detect kind */ if (!strncmp(refname, "refs/heads/", 11)) { kind = REF_LOCAL_BRANCH; refname += 11; } else if (!strncmp(refname, "refs/remotes/", 13)) { kind = REF_REMOTE_BRANCH; refname += 13; } else if (!strncmp(refname, "refs/tags/", 10)) { kind = REF_TAG; refname += 10; } /* Don't add types the caller doesn't want */ if ((kind & ref_list->kinds) == 0) return 0; /* Resize buffer */ if (ref_list->index >= ref_list->alloc) { ref_list->alloc = alloc_nr(ref_list->alloc); ref_list->list = xrealloc(ref_list->list, ref_list->alloc * sizeof(struct ref_item)); } /* Record the new item */ newitem = &(ref_list->list[ref_list->index++]); newitem->name = xstrdup(refname); newitem->kind = kind; hashcpy(newitem->sha1, sha1); len = strlen(newitem->name); if (len > ref_list->maxwidth) ref_list->maxwidth = len; return 0; } static void free_ref_list(struct ref_list *ref_list) { int i; for (i = 0; i < ref_list->index; i++) free(ref_list->list[i].name); free(ref_list->list); } static int ref_cmp(const void *r1, const void *r2) { struct ref_item *c1 = (struct ref_item *)(r1); struct ref_item *c2 = (struct ref_item *)(r2); if (c1->kind != c2->kind) return c1->kind - c2->kind; return strcmp(c1->name, c2->name); } static void print_ref_info(const unsigned char *sha1, int abbrev) { struct commit *commit; char subject[256]; commit = lookup_commit(sha1); if (commit && !parse_commit(commit)) pretty_print_commit(CMIT_FMT_ONELINE, commit, ~0, subject, sizeof(subject), 0, NULL, NULL, 0); else strcpy(subject, " **** invalid ref ****"); printf(" %s %s\n", find_unique_abbrev(sha1, abbrev), subject); } static void print_ref_list(int kinds, int verbose, int abbrev) { int i; char c; struct ref_list ref_list; memset(&ref_list, 0, sizeof(ref_list)); ref_list.kinds = kinds; for_each_ref(append_ref, &ref_list); qsort(ref_list.list, ref_list.index, sizeof(struct ref_item), ref_cmp); for (i = 0; i < ref_list.index; i++) { c = ' '; if (ref_list.list[i].kind == REF_LOCAL_BRANCH && !strcmp(ref_list.list[i].name, head)) c = '*'; if (verbose) { printf("%c %-*s", c, ref_list.maxwidth, ref_list.list[i].name); print_ref_info(ref_list.list[i].sha1, abbrev); } else printf("%c %s\n", c, ref_list.list[i].name); } free_ref_list(&ref_list); } static void create_branch(const char *name, const char *start, int force, int reflog) { struct ref_lock *lock; struct commit *commit; unsigned char sha1[20]; char ref[PATH_MAX], msg[PATH_MAX + 20]; snprintf(ref, sizeof ref, "refs/heads/%s", name); if (check_ref_format(ref)) die("'%s' is not a valid branch name.", name); if (resolve_ref(ref, sha1, 1, NULL)) { if (!force) die("A branch named '%s' already exists.", name); else if (!strcmp(head, name)) die("Cannot force update the current branch."); } if (get_sha1(start, sha1) || (commit = lookup_commit_reference(sha1)) == NULL) die("Not a valid branch point: '%s'.", start); hashcpy(sha1, commit->object.sha1); lock = lock_any_ref_for_update(ref, NULL); if (!lock) die("Failed to lock ref for update: %s.", strerror(errno)); if (reflog) { log_all_ref_updates = 1; snprintf(msg, sizeof msg, "branch: Created from %s", start); } if (write_ref_sha1(lock, sha1, msg) < 0) die("Failed to write ref: %s.", strerror(errno)); } static void rename_branch(const char *oldname, const char *newname, int force) { char oldref[PATH_MAX], newref[PATH_MAX]; unsigned char sha1[20]; if (snprintf(oldref, sizeof(oldref), "refs/heads/%s", oldname) > sizeof(oldref)) die("Old branchname too long"); if (check_ref_format(oldref)) die("Invalid branch name: %s", oldref); if (snprintf(newref, sizeof(newref), "refs/heads/%s", newname) > sizeof(newref)) die("New branchname too long"); if (check_ref_format(newref)) die("Invalid branch name: %s", newref); if (resolve_ref(newref, sha1, 1, NULL) && !force) die("A branch named '%s' already exists.", newname); if (rename_ref(oldref, newref)) die("Branch rename failed"); if (!strcmp(oldname, head) && create_symref("HEAD", newref)) die("Branch renamed to %s, but HEAD is not updated!", newname); } int cmd_branch(int argc, const char **argv, const char *prefix) { int delete = 0, force_delete = 0, force_create = 0; int rename = 0, force_rename = 0; int verbose = 0, abbrev = DEFAULT_ABBREV; int reflog = 0; int kinds = REF_LOCAL_BRANCH; int i; git_config(git_default_config); for (i = 1; i < argc; i++) { const char *arg = argv[i]; if (arg[0] != '-') break; if (!strcmp(arg, "--")) { i++; break; } if (!strcmp(arg, "-d")) { delete = 1; continue; } if (!strcmp(arg, "-D")) { delete = 1; force_delete = 1; continue; } if (!strcmp(arg, "-f")) { force_create = 1; continue; } if (!strcmp(arg, "-m")) { rename = 1; continue; } if (!strcmp(arg, "-M")) { rename = 1; force_rename = 1; continue; } if (!strcmp(arg, "-r")) { kinds = REF_REMOTE_BRANCH; continue; } if (!strcmp(arg, "-a")) { kinds = REF_REMOTE_BRANCH | REF_LOCAL_BRANCH; continue; } if (!strcmp(arg, "-l")) { reflog = 1; continue; } if (!strncmp(arg, "--abbrev=", 9)) { abbrev = atoi(arg+9); continue; } if (!strcmp(arg, "-v")) { verbose = 1; continue; } usage(builtin_branch_usage); } if ((delete && rename) || (delete && force_create) || (rename && force_create)) usage(builtin_branch_usage); head = xstrdup(resolve_ref("HEAD", head_sha1, 0, NULL)); if (!head) die("Failed to resolve HEAD as a valid ref."); if (strncmp(head, "refs/heads/", 11)) die("HEAD not found below refs/heads!"); head += 11; if (delete) delete_branches(argc - i, argv + i, force_delete); else if (i == argc) print_ref_list(kinds, verbose, abbrev); else if (rename && (i == argc - 1)) rename_branch(head, argv[i], force_rename); else if (rename && (i == argc - 2)) rename_branch(argv[i], argv[i + 1], force_rename); else if (i == argc - 1) create_branch(argv[i], head, force_create, reflog); else if (i == argc - 2) create_branch(argv[i], argv[i + 1], force_create, reflog); else usage(builtin_branch_usage); return 0; }
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
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
You can’t perform that action at this time.