-
Notifications
You must be signed in to change notification settings - Fork 0
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
* master: (40 commits) Clean up sha1 file writing Builtin git-cat-file builtin format-patch: squelch content-type for 7-bit ASCII CMIT_FMT_EMAIL: Q-encode Subject: and display-name part of From: fields. add more informative error messages to git-mktag remove the artificial restriction tagsize < 8kb git-rebase: use canonical A..B syntax to format-patch git-format-patch: now built-in. fmt-patch: Support --attach fmt-patch: understand old <his> notation Teach fmt-patch about --keep-subject Teach fmt-patch about --numbered fmt-patch: implement -o <dir> fmt-patch: output file names to stdout Teach fmt-patch to write individual files. Use RFC2822 dates from "git fmt-patch". git-fmt-patch: thinkofix to show [PATCH] properly. rename internal format-patch wip Minor tweak on subject line in --pretty=email Tentative built-in format-patch. ...
- Loading branch information
Showing
41 changed files
with
1,698 additions
and
1,065 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,187 @@ | ||
/* | ||
* "git add" builtin command | ||
* | ||
* Copyright (C) 2006 Linus Torvalds | ||
*/ | ||
#include <fnmatch.h> | ||
|
||
#include "cache.h" | ||
#include "builtin.h" | ||
#include "dir.h" | ||
|
||
static const char builtin_add_usage[] = | ||
"git-add [-n] [-v] <filepattern>..."; | ||
|
||
static void prune_directory(struct dir_struct *dir, const char **pathspec, int prefix) | ||
{ | ||
char *seen; | ||
int i, specs; | ||
struct dir_entry **src, **dst; | ||
|
||
for (specs = 0; pathspec[specs]; specs++) | ||
/* nothing */; | ||
seen = xmalloc(specs); | ||
memset(seen, 0, specs); | ||
|
||
src = dst = dir->entries; | ||
i = dir->nr; | ||
while (--i >= 0) { | ||
struct dir_entry *entry = *src++; | ||
if (!match_pathspec(pathspec, entry->name, entry->len, prefix, seen)) { | ||
free(entry); | ||
continue; | ||
} | ||
*dst++ = entry; | ||
} | ||
dir->nr = dst - dir->entries; | ||
|
||
for (i = 0; i < specs; i++) { | ||
struct stat st; | ||
const char *match; | ||
if (seen[i]) | ||
continue; | ||
|
||
/* Existing file? We must have ignored it */ | ||
match = pathspec[i]; | ||
if (!match[0] || !lstat(match, &st)) | ||
continue; | ||
die("pathspec '%s' did not match any files", match); | ||
} | ||
} | ||
|
||
static void fill_directory(struct dir_struct *dir, const char **pathspec) | ||
{ | ||
const char *path, *base; | ||
int baselen; | ||
|
||
/* Set up the default git porcelain excludes */ | ||
memset(dir, 0, sizeof(*dir)); | ||
dir->exclude_per_dir = ".gitignore"; | ||
path = git_path("info/exclude"); | ||
if (!access(path, R_OK)) | ||
add_excludes_from_file(dir, path); | ||
|
||
/* | ||
* Calculate common prefix for the pathspec, and | ||
* use that to optimize the directory walk | ||
*/ | ||
baselen = common_prefix(pathspec); | ||
path = "."; | ||
base = ""; | ||
if (baselen) { | ||
char *common = xmalloc(baselen + 1); | ||
common = xmalloc(baselen + 1); | ||
memcpy(common, *pathspec, baselen); | ||
common[baselen] = 0; | ||
path = base = common; | ||
} | ||
|
||
/* Read the directory and prune it */ | ||
read_directory(dir, path, base, baselen); | ||
if (pathspec) | ||
prune_directory(dir, pathspec, baselen); | ||
} | ||
|
||
static int add_file_to_index(const char *path, int verbose) | ||
{ | ||
int size, namelen; | ||
struct stat st; | ||
struct cache_entry *ce; | ||
|
||
if (lstat(path, &st)) | ||
die("%s: unable to stat (%s)", path, strerror(errno)); | ||
|
||
if (!S_ISREG(st.st_mode) && !S_ISLNK(st.st_mode)) | ||
die("%s: can only add regular files or symbolic links", path); | ||
|
||
namelen = strlen(path); | ||
size = cache_entry_size(namelen); | ||
ce = xcalloc(1, size); | ||
memcpy(ce->name, path, namelen); | ||
ce->ce_flags = htons(namelen); | ||
fill_stat_cache_info(ce, &st); | ||
|
||
ce->ce_mode = create_ce_mode(st.st_mode); | ||
if (!trust_executable_bit) { | ||
/* If there is an existing entry, pick the mode bits | ||
* from it. | ||
*/ | ||
int pos = cache_name_pos(path, namelen); | ||
if (pos >= 0) | ||
ce->ce_mode = active_cache[pos]->ce_mode; | ||
} | ||
|
||
if (index_path(ce->sha1, path, &st, 1)) | ||
die("unable to index file %s", path); | ||
if (add_cache_entry(ce, ADD_CACHE_OK_TO_ADD)) | ||
die("unable to add %s to index",path); | ||
if (verbose) | ||
printf("add '%s'\n", path); | ||
return 0; | ||
} | ||
|
||
static struct cache_file cache_file; | ||
|
||
int cmd_add(int argc, const char **argv, char **envp) | ||
{ | ||
int i, newfd; | ||
int verbose = 0, show_only = 0; | ||
const char *prefix = setup_git_directory(); | ||
const char **pathspec; | ||
struct dir_struct dir; | ||
|
||
git_config(git_default_config); | ||
|
||
newfd = hold_index_file_for_update(&cache_file, get_index_file()); | ||
if (newfd < 0) | ||
die("unable to create new cachefile"); | ||
|
||
if (read_cache() < 0) | ||
die("index file corrupt"); | ||
|
||
for (i = 1; i < argc; i++) { | ||
const char *arg = argv[i]; | ||
|
||
if (arg[0] != '-') | ||
break; | ||
if (!strcmp(arg, "--")) { | ||
i++; | ||
break; | ||
} | ||
if (!strcmp(arg, "-n")) { | ||
show_only = 1; | ||
continue; | ||
} | ||
if (!strcmp(arg, "-v")) { | ||
verbose = 1; | ||
continue; | ||
} | ||
die(builtin_add_usage); | ||
} | ||
git_config(git_default_config); | ||
pathspec = get_pathspec(prefix, argv + i); | ||
|
||
fill_directory(&dir, pathspec); | ||
|
||
if (show_only) { | ||
const char *sep = "", *eof = ""; | ||
for (i = 0; i < dir.nr; i++) { | ||
printf("%s%s", sep, dir.entries[i]->name); | ||
sep = " "; | ||
eof = "\n"; | ||
} | ||
fputs(eof, stdout); | ||
return 0; | ||
} | ||
|
||
for (i = 0; i < dir.nr; i++) | ||
add_file_to_index(dir.entries[i]->name, verbose); | ||
|
||
if (active_cache_changed) { | ||
if (write_cache(newfd, active_cache, active_nr) || | ||
commit_index_file(&cache_file)) | ||
die("Unable to write new index file"); | ||
} | ||
|
||
return 0; | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Oops, something went wrong.