|
|
|
#include "cache.h"
|
|
|
|
#include "blob.h"
|
|
|
|
#include "dir.h"
|
|
|
|
|
|
|
|
static void create_directories(const char *path, const struct checkout *state)
|
|
|
|
{
|
|
|
|
int len = strlen(path);
|
|
|
|
char *buf = xmalloc(len + 1);
|
|
|
|
const char *slash = path;
|
|
|
|
|
|
|
|
while ((slash = strchr(slash+1, '/')) != NULL) {
|
Do not expect unlink(2) to fail on a directory.
When "git checkout-index" checks out path A/B/C, it makes sure A
and A/B are truly directories; if there is a regular file or
symlink at A, we prefer to remove it.
We used to do this by catching an error return from mkdir(2),
and on EEXIST did unlink(2), and when it succeeded, tried
another mkdir(2).
Thomas Glanzmann found out the above does not work on Solaris
for a root user, as unlink(2) was so old fashioned there that it
allowed to unlink a directory.
As pointed out, this still doesn't guarantee that git won't call
"unlink()" on a directory (race conditions etc), but that's
fundamentally true (there is no "funlink()" like there is
"fstat()"), and besides, that is in no way git-specific (ie it's
true of any application that gets run as root).
Acked-by: Linus Torvalds <torvalds@linux-foundation.org>
Signed-off-by: Junio C Hamano <gitster@pobox.com>
18 years ago
|
|
|
struct stat st;
|
|
|
|
int stat_status;
|
|
|
|
|
|
|
|
len = slash - path;
|
|
|
|
memcpy(buf, path, len);
|
|
|
|
buf[len] = 0;
|
Do not expect unlink(2) to fail on a directory.
When "git checkout-index" checks out path A/B/C, it makes sure A
and A/B are truly directories; if there is a regular file or
symlink at A, we prefer to remove it.
We used to do this by catching an error return from mkdir(2),
and on EEXIST did unlink(2), and when it succeeded, tried
another mkdir(2).
Thomas Glanzmann found out the above does not work on Solaris
for a root user, as unlink(2) was so old fashioned there that it
allowed to unlink a directory.
As pointed out, this still doesn't guarantee that git won't call
"unlink()" on a directory (race conditions etc), but that's
fundamentally true (there is no "funlink()" like there is
"fstat()"), and besides, that is in no way git-specific (ie it's
true of any application that gets run as root).
Acked-by: Linus Torvalds <torvalds@linux-foundation.org>
Signed-off-by: Junio C Hamano <gitster@pobox.com>
18 years ago
|
|
|
|
|
|
|
if (len <= state->base_dir_len)
|
|
|
|
/*
|
|
|
|
* checkout-index --prefix=<dir>; <dir> is
|
|
|
|
* allowed to be a symlink to an existing
|
|
|
|
* directory.
|
|
|
|
*/
|
|
|
|
stat_status = stat(buf, &st);
|
|
|
|
else
|
|
|
|
/*
|
|
|
|
* if there currently is a symlink, we would
|
|
|
|
* want to replace it with a real directory.
|
|
|
|
*/
|
|
|
|
stat_status = lstat(buf, &st);
|
|
|
|
|
|
|
|
if (!stat_status && S_ISDIR(st.st_mode))
|
|
|
|
continue; /* ok, it is already a directory. */
|
|
|
|
|
|
|
|
/*
|
|
|
|
* We know stat_status == 0 means something exists
|
|
|
|
* there and this mkdir would fail, but that is an
|
|
|
|
* error codepath; we do not care, as we unlink and
|
|
|
|
* mkdir again in such a case.
|
|
|
|
*/
|
|
|
|
if (mkdir(buf, 0777)) {
|
Do not expect unlink(2) to fail on a directory.
When "git checkout-index" checks out path A/B/C, it makes sure A
and A/B are truly directories; if there is a regular file or
symlink at A, we prefer to remove it.
We used to do this by catching an error return from mkdir(2),
and on EEXIST did unlink(2), and when it succeeded, tried
another mkdir(2).
Thomas Glanzmann found out the above does not work on Solaris
for a root user, as unlink(2) was so old fashioned there that it
allowed to unlink a directory.
As pointed out, this still doesn't guarantee that git won't call
"unlink()" on a directory (race conditions etc), but that's
fundamentally true (there is no "funlink()" like there is
"fstat()"), and besides, that is in no way git-specific (ie it's
true of any application that gets run as root).
Acked-by: Linus Torvalds <torvalds@linux-foundation.org>
Signed-off-by: Junio C Hamano <gitster@pobox.com>
18 years ago
|
|
|
if (errno == EEXIST && state->force &&
|
|
|
|
!unlink(buf) && !mkdir(buf, 0777))
|
|
|
|
continue;
|
|
|
|
die("cannot create directory at %s", buf);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
free(buf);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void remove_subtree(const char *path)
|
|
|
|
{
|
|
|
|
DIR *dir = opendir(path);
|
|
|
|
struct dirent *de;
|
|
|
|
char pathbuf[PATH_MAX];
|
|
|
|
char *name;
|
|
|
|
|
|
|
|
if (!dir)
|
|
|
|
die("cannot opendir %s (%s)", path, strerror(errno));
|
|
|
|
strcpy(pathbuf, path);
|
|
|
|
name = pathbuf + strlen(path);
|
|
|
|
*name++ = '/';
|
|
|
|
while ((de = readdir(dir)) != NULL) {
|
|
|
|
struct stat st;
|
|
|
|
if (is_dot_or_dotdot(de->d_name))
|
|
|
|
continue;
|
|
|
|
strcpy(name, de->d_name);
|
|
|
|
if (lstat(pathbuf, &st))
|
|
|
|
die("cannot lstat %s (%s)", pathbuf, strerror(errno));
|
|
|
|
if (S_ISDIR(st.st_mode))
|
|
|
|
remove_subtree(pathbuf);
|
|
|
|
else if (unlink(pathbuf))
|
|
|
|
die("cannot unlink %s (%s)", pathbuf, strerror(errno));
|
|
|
|
}
|
|
|
|
closedir(dir);
|
|
|
|
if (rmdir(path))
|
|
|
|
die("cannot rmdir %s (%s)", path, strerror(errno));
|
|
|
|
}
|
|
|
|
|
|
|
|
static int create_file(const char *path, unsigned int mode)
|
|
|
|
{
|
|
|
|
mode = (mode & 0100) ? 0777 : 0666;
|
|
|
|
return open(path, O_WRONLY | O_CREAT | O_EXCL, mode);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void *read_blob_entry(struct cache_entry *ce, const char *path, unsigned long *size)
|
|
|
|
{
|
|
|
|
enum object_type type;
|
|
|
|
void *new = read_sha1_file(ce->sha1, &type, size);
|
|
|
|
|
|
|
|
if (new) {
|
|
|
|
if (type == OBJ_BLOB)
|
|
|
|
return new;
|
|
|
|
free(new);
|
|
|
|
}
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int write_entry(struct cache_entry *ce, char *path, const struct checkout *state, int to_tempfile)
|
|
|
|
{
|
|
|
|
int fd;
|
|
|
|
long wrote;
|
|
|
|
|
|
|
|
switch (ce->ce_mode & S_IFMT) {
|
Rewrite convert_to_{git,working_tree} to use strbuf's.
* Now, those functions take an "out" strbuf argument, where they store their
result if any. In that case, it also returns 1, else it returns 0.
* those functions support "in place" editing, in the sense that it's OK to
call them this way:
convert_to_git(path, sb->buf, sb->len, sb);
When doable, conversions are done in place for real, else the strbuf
content is just replaced with the new one, transparentely for the caller.
If you want to create a new filter working this way, being the accumulation
of filter1, filter2, ... filtern, then your meta_filter would be:
int meta_filter(..., const char *src, size_t len, struct strbuf *sb)
{
int ret = 0;
ret |= filter1(...., src, len, sb);
if (ret) {
src = sb->buf;
len = sb->len;
}
ret |= filter2(...., src, len, sb);
if (ret) {
src = sb->buf;
len = sb->len;
}
....
return ret | filtern(..., src, len, sb);
}
That's why subfilters the convert_to_* functions called were also rewritten
to work this way.
Signed-off-by: Pierre Habouzit <madcoder@debian.org>
Acked-by: Linus Torvalds <torvalds@linux-foundation.org>
Signed-off-by: Junio C Hamano <gitster@pobox.com>
18 years ago
|
|
|
char *new;
|
|
|
|
struct strbuf buf;
|
|
|
|
unsigned long size;
|
Lazy man's auto-CRLF
It currently does NOT know about file attributes, so it does its
conversion purely based on content. Maybe that is more in the "git
philosophy" anyway, since content is king, but I think we should try to do
the file attributes to turn it off on demand.
Anyway, BY DEFAULT it is off regardless, because it requires a
[core]
AutoCRLF = true
in your config file to be enabled. We could make that the default for
Windows, of course, the same way we do some other things (filemode etc).
But you can actually enable it on UNIX, and it will cause:
- "git update-index" will write blobs without CRLF
- "git diff" will diff working tree files without CRLF
- "git checkout" will write files to the working tree _with_ CRLF
and things work fine.
Funnily, it actually shows an odd file in git itself:
git clone -n git test-crlf
cd test-crlf
git config core.autocrlf true
git checkout
git diff
shows a diff for "Documentation/docbook-xsl.css". Why? Because we have
actually checked in that file *with* CRLF! So when "core.autocrlf" is
true, we'll always generate a *different* hash for it in the index,
because the index hash will be for the content _without_ CRLF.
Is this complete? I dunno. It seems to work for me. It doesn't use the
filename at all right now, and that's probably a deficiency (we could
certainly make the "is_binary()" heuristics also take standard filename
heuristics into account).
I don't pass in the filename at all for the "index_fd()" case
(git-update-index), so that would need to be passed around, but this
actually works fine.
NOTE NOTE NOTE! The "is_binary()" heuristics are totally made-up by yours
truly. I will not guarantee that they work at all reasonable. Caveat
emptor. But it _is_ simple, and it _is_ safe, since it's all off by
default.
The patch is pretty simple - the biggest part is the new "convert.c" file,
but even that is really just basic stuff that anybody can write in
"Teaching C 101" as a final project for their first class in programming.
Not to say that it's bug-free, of course - but at least we're not talking
about rocket surgery here.
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
Signed-off-by: Junio C Hamano <junkio@cox.net>
18 years ago
|
|
|
|
|
|
|
case S_IFREG:
|
|
|
|
new = read_blob_entry(ce, path, &size);
|
|
|
|
if (!new)
|
|
|
|
return error("git checkout-index: unable to read sha1 file of %s (%s)",
|
|
|
|
path, sha1_to_hex(ce->sha1));
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Convert from git internal format to working tree format
|
|
|
|
*/
|
Rewrite convert_to_{git,working_tree} to use strbuf's.
* Now, those functions take an "out" strbuf argument, where they store their
result if any. In that case, it also returns 1, else it returns 0.
* those functions support "in place" editing, in the sense that it's OK to
call them this way:
convert_to_git(path, sb->buf, sb->len, sb);
When doable, conversions are done in place for real, else the strbuf
content is just replaced with the new one, transparentely for the caller.
If you want to create a new filter working this way, being the accumulation
of filter1, filter2, ... filtern, then your meta_filter would be:
int meta_filter(..., const char *src, size_t len, struct strbuf *sb)
{
int ret = 0;
ret |= filter1(...., src, len, sb);
if (ret) {
src = sb->buf;
len = sb->len;
}
ret |= filter2(...., src, len, sb);
if (ret) {
src = sb->buf;
len = sb->len;
}
....
return ret | filtern(..., src, len, sb);
}
That's why subfilters the convert_to_* functions called were also rewritten
to work this way.
Signed-off-by: Pierre Habouzit <madcoder@debian.org>
Acked-by: Linus Torvalds <torvalds@linux-foundation.org>
Signed-off-by: Junio C Hamano <gitster@pobox.com>
18 years ago
|
|
|
strbuf_init(&buf, 0);
|
|
|
|
if (convert_to_working_tree(ce->name, new, size, &buf)) {
|
|
|
|
size_t newsize = 0;
|
|
|
|
free(new);
|
|
|
|
new = strbuf_detach(&buf, &newsize);
|
|
|
|
size = newsize;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (to_tempfile) {
|
|
|
|
strcpy(path, ".merge_file_XXXXXX");
|
|
|
|
fd = mkstemp(path);
|
|
|
|
} else
|
|
|
|
fd = create_file(path, ce->ce_mode);
|
|
|
|
if (fd < 0) {
|
|
|
|
free(new);
|
|
|
|
return error("git checkout-index: unable to create file %s (%s)",
|
|
|
|
path, strerror(errno));
|
|
|
|
}
|
Lazy man's auto-CRLF
It currently does NOT know about file attributes, so it does its
conversion purely based on content. Maybe that is more in the "git
philosophy" anyway, since content is king, but I think we should try to do
the file attributes to turn it off on demand.
Anyway, BY DEFAULT it is off regardless, because it requires a
[core]
AutoCRLF = true
in your config file to be enabled. We could make that the default for
Windows, of course, the same way we do some other things (filemode etc).
But you can actually enable it on UNIX, and it will cause:
- "git update-index" will write blobs without CRLF
- "git diff" will diff working tree files without CRLF
- "git checkout" will write files to the working tree _with_ CRLF
and things work fine.
Funnily, it actually shows an odd file in git itself:
git clone -n git test-crlf
cd test-crlf
git config core.autocrlf true
git checkout
git diff
shows a diff for "Documentation/docbook-xsl.css". Why? Because we have
actually checked in that file *with* CRLF! So when "core.autocrlf" is
true, we'll always generate a *different* hash for it in the index,
because the index hash will be for the content _without_ CRLF.
Is this complete? I dunno. It seems to work for me. It doesn't use the
filename at all right now, and that's probably a deficiency (we could
certainly make the "is_binary()" heuristics also take standard filename
heuristics into account).
I don't pass in the filename at all for the "index_fd()" case
(git-update-index), so that would need to be passed around, but this
actually works fine.
NOTE NOTE NOTE! The "is_binary()" heuristics are totally made-up by yours
truly. I will not guarantee that they work at all reasonable. Caveat
emptor. But it _is_ simple, and it _is_ safe, since it's all off by
default.
The patch is pretty simple - the biggest part is the new "convert.c" file,
but even that is really just basic stuff that anybody can write in
"Teaching C 101" as a final project for their first class in programming.
Not to say that it's bug-free, of course - but at least we're not talking
about rocket surgery here.
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
Signed-off-by: Junio C Hamano <junkio@cox.net>
18 years ago
|
|
|
|
|
|
|
wrote = write_in_full(fd, new, size);
|
|
|
|
close(fd);
|
|
|
|
free(new);
|
|
|
|
if (wrote != size)
|
|
|
|
return error("git checkout-index: unable to write file %s", path);
|
|
|
|
break;
|
|
|
|
case S_IFLNK:
|
|
|
|
new = read_blob_entry(ce, path, &size);
|
|
|
|
if (!new)
|
|
|
|
return error("git checkout-index: unable to read sha1 file of %s (%s)",
|
|
|
|
path, sha1_to_hex(ce->sha1));
|
|
|
|
if (to_tempfile || !has_symlinks) {
|
|
|
|
if (to_tempfile) {
|
|
|
|
strcpy(path, ".merge_link_XXXXXX");
|
|
|
|
fd = mkstemp(path);
|
|
|
|
} else
|
|
|
|
fd = create_file(path, 0666);
|
|
|
|
if (fd < 0) {
|
|
|
|
free(new);
|
|
|
|
return error("git checkout-index: unable to create "
|
|
|
|
"file %s (%s)", path, strerror(errno));
|
|
|
|
}
|
|
|
|
wrote = write_in_full(fd, new, size);
|
|
|
|
close(fd);
|
|
|
|
free(new);
|
|
|
|
if (wrote != size)
|
|
|
|
return error("git checkout-index: unable to write file %s",
|
|
|
|
path);
|
|
|
|
} else {
|
|
|
|
wrote = symlink(new, path);
|
|
|
|
free(new);
|
|
|
|
if (wrote)
|
|
|
|
return error("git checkout-index: unable to create "
|
|
|
|
"symlink %s (%s)", path, strerror(errno));
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case S_IFGITLINK:
|
|
|
|
if (to_tempfile)
|
|
|
|
return error("git checkout-index: cannot create temporary subproject %s", path);
|
|
|
|
if (mkdir(path, 0777) < 0)
|
|
|
|
return error("git checkout-index: cannot create subproject directory %s", path);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
return error("git checkout-index: unknown file mode for %s", path);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (state->refresh_cache) {
|
|
|
|
struct stat st;
|
|
|
|
lstat(ce->name, &st);
|
|
|
|
fill_stat_cache_info(ce, &st);
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
int checkout_entry(struct cache_entry *ce, const struct checkout *state, char *topath)
|
|
|
|
{
|
|
|
|
static char path[PATH_MAX + 1];
|
|
|
|
struct stat st;
|
|
|
|
int len = state->base_dir_len;
|
|
|
|
|
|
|
|
if (topath)
|
|
|
|
return write_entry(ce, topath, state, 1);
|
|
|
|
|
|
|
|
memcpy(path, state->base_dir, len);
|
|
|
|
strcpy(path + len, ce->name);
|
|
|
|
|
|
|
|
if (!lstat(path, &st)) {
|
|
|
|
unsigned changed = ce_match_stat(ce, &st, CE_MATCH_IGNORE_VALID);
|
|
|
|
if (!changed)
|
|
|
|
return 0;
|
|
|
|
if (!state->force) {
|
|
|
|
if (!state->quiet)
|
|
|
|
fprintf(stderr, "git-checkout-index: %s already exists\n", path);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* We unlink the old file, to get the new one with the
|
|
|
|
* right permissions (including umask, which is nasty
|
|
|
|
* to emulate by hand - much easier to let the system
|
|
|
|
* just do the right thing)
|
|
|
|
*/
|
|
|
|
if (S_ISDIR(st.st_mode)) {
|
|
|
|
/* If it is a gitlink, leave it alone! */
|
|
|
|
if (S_ISGITLINK(ce->ce_mode))
|
|
|
|
return 0;
|
|
|
|
if (!state->force)
|
|
|
|
return error("%s is a directory", path);
|
|
|
|
remove_subtree(path);
|
|
|
|
} else if (unlink(path))
|
|
|
|
return error("unable to unlink old '%s' (%s)", path, strerror(errno));
|
|
|
|
} else if (state->not_new)
|
|
|
|
return 0;
|
|
|
|
create_directories(path, state);
|
|
|
|
return write_entry(ce, path, state, 0);
|
|
|
|
}
|