2006-06-06 23:51:49 +04:00
|
|
|
/*
|
|
|
|
* Copyright (c) 2005, Junio C Hamano
|
|
|
|
*/
|
|
|
|
#include "cache.h"
|
|
|
|
|
|
|
|
static struct lock_file *lock_file_list;
|
|
|
|
|
|
|
|
static void remove_lock_file(void)
|
|
|
|
{
|
|
|
|
while (lock_file_list) {
|
|
|
|
if (lock_file_list->filename[0])
|
|
|
|
unlink(lock_file_list->filename);
|
|
|
|
lock_file_list = lock_file_list->next;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void remove_lock_file_on_signal(int signo)
|
|
|
|
{
|
|
|
|
remove_lock_file();
|
|
|
|
signal(SIGINT, SIG_DFL);
|
|
|
|
raise(signo);
|
|
|
|
}
|
|
|
|
|
2006-08-12 12:03:47 +04:00
|
|
|
static int lock_file(struct lock_file *lk, const char *path)
|
2006-06-06 23:51:49 +04:00
|
|
|
{
|
|
|
|
int fd;
|
|
|
|
sprintf(lk->filename, "%s.lock", path);
|
|
|
|
fd = open(lk->filename, O_RDWR | O_CREAT | O_EXCL, 0666);
|
2006-06-10 09:07:23 +04:00
|
|
|
if (0 <= fd) {
|
2007-01-02 22:19:05 +03:00
|
|
|
if (!lk->on_list) {
|
2006-06-10 09:07:23 +04:00
|
|
|
lk->next = lock_file_list;
|
|
|
|
lock_file_list = lk;
|
2007-01-02 22:19:05 +03:00
|
|
|
lk->on_list = 1;
|
|
|
|
}
|
|
|
|
if (lock_file_list) {
|
2006-06-10 09:07:23 +04:00
|
|
|
signal(SIGINT, remove_lock_file_on_signal);
|
|
|
|
atexit(remove_lock_file);
|
|
|
|
}
|
|
|
|
if (adjust_shared_perm(lk->filename))
|
|
|
|
return error("cannot fix permission bits on %s",
|
|
|
|
lk->filename);
|
2006-06-06 23:51:49 +04:00
|
|
|
}
|
2007-01-02 22:19:05 +03:00
|
|
|
else
|
|
|
|
lk->filename[0] = 0;
|
2006-06-06 23:51:49 +04:00
|
|
|
return fd;
|
|
|
|
}
|
|
|
|
|
2006-08-12 12:03:47 +04:00
|
|
|
int hold_lock_file_for_update(struct lock_file *lk, const char *path, int die_on_error)
|
|
|
|
{
|
|
|
|
int fd = lock_file(lk, path);
|
|
|
|
if (fd < 0 && die_on_error)
|
2007-01-06 06:14:04 +03:00
|
|
|
die("unable to create '%s.lock': %s", path, strerror(errno));
|
2006-08-12 12:03:47 +04:00
|
|
|
return fd;
|
|
|
|
}
|
|
|
|
|
2006-06-06 23:51:49 +04:00
|
|
|
int commit_lock_file(struct lock_file *lk)
|
|
|
|
{
|
|
|
|
char result_file[PATH_MAX];
|
|
|
|
int i;
|
|
|
|
strcpy(result_file, lk->filename);
|
|
|
|
i = strlen(result_file) - 5; /* .lock */
|
|
|
|
result_file[i] = 0;
|
|
|
|
i = rename(lk->filename, result_file);
|
|
|
|
lk->filename[0] = 0;
|
|
|
|
return i;
|
|
|
|
}
|
|
|
|
|
_GIT_INDEX_OUTPUT: allow plumbing to output to an alternative index file.
When defined, this allows plumbing commands that update the
index (add, apply, checkout-index, merge-recursive, mv,
read-tree, rm, update-index, and write-tree) to write their
resulting index to an alternative index file while holding a
lock to the original index file. With this, git-commit that
jumps the index does not have to make an extra copy of the index
file, and more importantly, it can do the update while holding
the lock on the index.
However, I think the interface to let an environment variable
specify the output is a mistake, as shown in the documentation.
If a curious user has the environment variable set to something
other than the file GIT_INDEX_FILE points at, almost everything
will break. This should instead be a command line parameter to
tell these plumbing commands to write the result in the named
file, to prevent stupid mistakes.
Signed-off-by: Junio C Hamano <junkio@cox.net>
2007-04-01 10:09:02 +04:00
|
|
|
int hold_locked_index(struct lock_file *lk, int die_on_error)
|
|
|
|
{
|
|
|
|
return hold_lock_file_for_update(lk, get_index_file(), die_on_error);
|
|
|
|
}
|
|
|
|
|
|
|
|
int commit_locked_index(struct lock_file *lk)
|
|
|
|
{
|
|
|
|
char *output = getenv(INDEX_OUTPUT_ENVIRONMENT);
|
|
|
|
if (output && *output) {
|
|
|
|
int result = rename(lk->filename, output);
|
|
|
|
lk->filename[0] = 0;
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
return commit_lock_file(lk);
|
|
|
|
}
|
|
|
|
|
2006-06-06 23:51:49 +04:00
|
|
|
void rollback_lock_file(struct lock_file *lk)
|
|
|
|
{
|
|
|
|
if (lk->filename[0])
|
|
|
|
unlink(lk->filename);
|
|
|
|
lk->filename[0] = 0;
|
|
|
|
}
|
|
|
|
|