
2002-11-11 Jeffrey Stedfast <fejj@ximian.com> * providers/local/camel-spool-summary.c (spool_summary_sync_full): Use g_strerror when setting an exception string (we need it to be in UTF-8). (spool_summary_check): Here too. * providers/local/camel-spool-store.c (construct): Use g_strerror when setting an exception string (we need it to be in UTF-8). (get_folder): Same. (scan_dir): Here too. * providers/local/camel-spool-folder.c (spool_lock): Use g_strerror when setting an exception string (we need it to be in UTF-8). * providers/local/camel-mh-summary.c (mh_summary_check): Use g_strerror when setting an exception string (we need it to be in UTF-8). * providers/local/camel-mh-store.c (delete_folder): Use g_strerror when setting an exception string (we need it to be in UTF-8). * providers/local/camel-mbox-summary.c (summary_update): Use g_strerror when setting an exception string (we need it to be in UTF-8). (mbox_summary_sync_full): Here too. (mbox_summary_sync_quick): Same. (mbox_summary_sync): Also here. (camel_mbox_summary_sync_mbox): Again here. * providers/local/camel-mbox-folder.c (mbox_lock): Use g_strerror when setting an exception string (we need it to be in UTF-8). (mbox_append_message): Same. (mbox_get_message): Here too. * providers/local/camel-maildir-summary.c (maildir_summary_load): Use g_strerror when setting an exception string (we need it to be in UTF-8). (maildir_summary_check): Same. * providers/local/camel-maildir-store.c (get_folder): Use g_strerror when setting an exception string (we need it to be in UTF-8). (delete_folder): Same. (delete_folder): Here too. * providers/local/camel-local-summary.c (local_summary_sync): Use g_strerror when setting an exception string (we need it to be in UTF-8). * providers/local/camel-local-store.c (get_folder): Use g_strerror when setting an exception string (we need it to be in UTF-8). (create_folder): Same. (xrename): Here too. (rename_folder): And here. (delete_folder): Also here. * camel-provider.c (camel_provider_init): For debugging printfs, we want to use normal strerror (we want locale charset, not UTF-8). * camel-movemail.c (camel_movemail): Use g_strerror when setting an exception string (we need it to be in UTF-8). (movemail_external): Same. (camel_movemail_copy_file): Here too. (camel_movemail_solaris): Also here. * camel-mime-utils.c (rfc2047_decode_word): For debugging printfs, we want to use normal strerror (we want locale charset, not UTF-8). (header_encode_param): Same. * camel-mime-part-utils.c (convert_buffer): For debugging printfs, we want to use normal strerror (we want locale charset, not UTF-8). * camel-lock-client.c (camel_lock_helper_init): Use g_strerror when setting an exception string (we need it to be in UTF-8). * camel-data-cache.c (camel_data_cache_remove): Use g_strerror when setting an exception string (we need it to be in UTF-8). * camel-tcp-stream-raw.c (flaky_tcp_write): For debugging printfs, we want to use normal strerror (we want locale charset, not UTF-8). (flaky_tcp_read): Same. * camel-gpg-context.c (gpg_ctx_op_step): For debugging printfs, we want to use normal strerror (we want locale charset, not UTF-8). * camel-service.c (camel_gethostbyname): Use g_strerror when setting an exception string (we need it to be in UTF-8). * camel-lock.c (camel_lock_dot): Use g_strerror when setting an exception string (we need it to be in UTF-8). (camel_lock_fcntl): Same. svn path=/trunk/; revision=18689
426 lines
9.4 KiB
C
426 lines
9.4 KiB
C
/* -*- Mode: C; tab-width: 8; indent-tabs-mode: t; c-basic-offset: 8 -*-
|
|
*
|
|
* Author: Michael Zucchi <notzed@ximian.com>
|
|
*
|
|
* Copyright (C) 1999 Ximian (www.ximian.com/).
|
|
*
|
|
* This program is free software; you can redistribute it and/or
|
|
* modify it under the terms of version 2 of the GNU General Public
|
|
* License as published by the Free Software Foundation.
|
|
*
|
|
* This program is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with this program; if not, write to the Free Software
|
|
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307
|
|
* USA
|
|
*/
|
|
|
|
#ifdef HAVE_CONFIG_H
|
|
#include <config.h>
|
|
#endif
|
|
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
#include <string.h>
|
|
|
|
#include <sys/stat.h>
|
|
#include <unistd.h>
|
|
#include <errno.h>
|
|
#ifdef HAVE_ALLOCA_H
|
|
#include <alloca.h>
|
|
#endif
|
|
#include <time.h>
|
|
|
|
#ifdef USE_DOT
|
|
#include <sys/types.h>
|
|
#include <sys/stat.h>
|
|
#include <fcntl.h>
|
|
#endif
|
|
|
|
#ifdef USE_FCNTL
|
|
#include <unistd.h>
|
|
#include <fcntl.h>
|
|
#endif
|
|
|
|
#ifdef USE_FLOCK
|
|
#include <sys/file.h>
|
|
#endif
|
|
|
|
#include <glib.h>
|
|
|
|
#include "camel-lock.h"
|
|
#include "camel-i18n.h"
|
|
|
|
#define d(x) /*(printf("%s(%d): ", __FILE__, __LINE__),(x))*/
|
|
|
|
/**
|
|
* camel_lock_dot:
|
|
* @path:
|
|
* @ex:
|
|
*
|
|
* Create an exclusive lock using .lock semantics.
|
|
* All locks are equivalent to write locks (exclusive).
|
|
*
|
|
* Return value: -1 on error, sets @ex appropriately.
|
|
**/
|
|
int
|
|
camel_lock_dot(const char *path, CamelException *ex)
|
|
{
|
|
#ifdef USE_DOT
|
|
char *locktmp, *lock;
|
|
int retry = 0;
|
|
int fdtmp;
|
|
struct stat st;
|
|
|
|
/* TODO: Is there a reliable way to refresh the lock, if we're still busy with it?
|
|
Does it matter? We will normally also use fcntl too ... */
|
|
|
|
/* use alloca, save cleaning up afterwards */
|
|
lock = alloca(strlen(path) + strlen(".lock") + 1);
|
|
sprintf(lock, "%s.lock", path);
|
|
locktmp = alloca(strlen(path) + strlen("XXXXXX") + 1);
|
|
|
|
#ifndef HAVE_MKSTEMP
|
|
sprintf(locktmp, "%sXXXXXX", path);
|
|
if (mktemp(locktmp) == NULL) {
|
|
/* well, this is really only a programatic error */
|
|
camel_exception_setv (ex, CAMEL_EXCEPTION_SYSTEM,
|
|
_("Could not create lock file for %s: %s"),
|
|
path, g_strerror (errno));
|
|
return -1;
|
|
}
|
|
#endif
|
|
|
|
while (retry < CAMEL_LOCK_DOT_RETRY) {
|
|
|
|
d(printf("trying to lock '%s', attempt %d\n", lock, retry));
|
|
|
|
if (retry > 0)
|
|
sleep(CAMEL_LOCK_DOT_DELAY);
|
|
|
|
#ifdef HAVE_MKSTEMP
|
|
sprintf(locktmp, "%sXXXXXX", path);
|
|
fdtmp = mkstemp(locktmp);
|
|
#else
|
|
fdtmp = open(locktmp, O_RDWR|O_CREAT|O_EXCL, 0600);
|
|
#endif
|
|
if (fdtmp == -1) {
|
|
camel_exception_setv (ex, CAMEL_EXCEPTION_SYSTEM,
|
|
_("Could not create lock file for %s: %s"),
|
|
path, g_strerror (errno));
|
|
return -1;
|
|
}
|
|
close(fdtmp);
|
|
|
|
/* apparently return code from link can be unreliable for nfs (see link(2)), so we ignore it */
|
|
link(locktmp, lock);
|
|
|
|
/* but we check stat instead (again, see link(2)) */
|
|
if (stat(locktmp, &st) == -1) {
|
|
d(printf("Our lock file %s vanished!?\n", locktmp));
|
|
|
|
/* well that was unexpected, try cleanup/retry */
|
|
unlink(locktmp);
|
|
unlink(lock);
|
|
} else {
|
|
d(printf("tmp lock created, link count is %d\n", st.st_nlink));
|
|
|
|
unlink(locktmp);
|
|
|
|
/* if we had 2 links, we have created the .lock, return ok, otherwise we need to keep trying */
|
|
if (st.st_nlink == 2)
|
|
return 0;
|
|
}
|
|
|
|
/* check for stale lock, kill it */
|
|
if (stat(lock, &st) == 0) {
|
|
time_t now = time(0);
|
|
(printf("There is an existing lock %ld seconds old\n", now-st.st_ctime));
|
|
if (st.st_ctime < now - CAMEL_LOCK_DOT_STALE) {
|
|
d(printf("Removing it now\n"));
|
|
unlink(lock);
|
|
}
|
|
}
|
|
|
|
retry++;
|
|
}
|
|
|
|
d(printf("failed to get lock after %d retries\n", retry));
|
|
|
|
camel_exception_setv(ex, CAMEL_EXCEPTION_SYSTEM, _("Timed out trying to get lock file on %s. Try again later."), path);
|
|
return -1;
|
|
#else /* ! USE_DOT */
|
|
return 0;
|
|
#endif
|
|
}
|
|
|
|
/**
|
|
* camel_unlock_dot:
|
|
* @path:
|
|
*
|
|
* Attempt to unlock a .lock lock.
|
|
**/
|
|
void
|
|
camel_unlock_dot(const char *path)
|
|
{
|
|
#ifdef USE_DOT
|
|
char *lock;
|
|
|
|
lock = alloca(strlen(path) + strlen(".lock") + 1);
|
|
sprintf(lock, "%s.lock", path);
|
|
d(printf("unlocking %s\n", lock));
|
|
(void)unlink(lock);
|
|
#endif
|
|
}
|
|
|
|
/**
|
|
* camel_lock_fcntl:
|
|
* @fd:
|
|
* @type:
|
|
* @ex:
|
|
*
|
|
* Create a lock using fcntl(2).
|
|
*
|
|
* @type is CAMEL_LOCK_WRITE or CAMEL_LOCK_READ,
|
|
* to create exclusive or shared read locks
|
|
*
|
|
* Return value: -1 on error.
|
|
**/
|
|
int
|
|
camel_lock_fcntl(int fd, CamelLockType type, CamelException *ex)
|
|
{
|
|
#ifdef USE_FCNTL
|
|
struct flock lock;
|
|
|
|
d(printf("fcntl locking %d\n", fd));
|
|
|
|
memset(&lock, 0, sizeof(lock));
|
|
lock.l_type = type==CAMEL_LOCK_READ?F_RDLCK:F_WRLCK;
|
|
if (fcntl(fd, F_SETLK, &lock) == -1) {
|
|
/* If we get a 'locking not vailable' type error,
|
|
we assume the filesystem doesn't support fcntl() locking */
|
|
/* this is somewhat system-dependent */
|
|
if (errno != EINVAL && errno != ENOLCK) {
|
|
camel_exception_setv (ex, CAMEL_EXCEPTION_SYSTEM,
|
|
_("Failed to get lock using fcntl(2): %s"),
|
|
g_strerror (errno));
|
|
return -1;
|
|
} else {
|
|
static int failed = 0;
|
|
|
|
if (failed == 0)
|
|
fprintf(stderr, "fcntl(2) locking appears not to work on this filesystem");
|
|
failed++;
|
|
}
|
|
}
|
|
#endif
|
|
return 0;
|
|
}
|
|
|
|
/**
|
|
* camel_unlock_fcntl:
|
|
* @fd:
|
|
*
|
|
* Unlock an fcntl lock.
|
|
**/
|
|
void
|
|
camel_unlock_fcntl(int fd)
|
|
{
|
|
#ifdef USE_FCNTL
|
|
struct flock lock;
|
|
|
|
d(printf("fcntl unlocking %d\n", fd));
|
|
|
|
memset(&lock, 0, sizeof(lock));
|
|
lock.l_type = F_UNLCK;
|
|
fcntl(fd, F_SETLK, &lock);
|
|
#endif
|
|
}
|
|
|
|
/**
|
|
* camel_lock_flock:
|
|
* @fd:
|
|
* @type:
|
|
* @ex:
|
|
*
|
|
* Create a lock using flock(2).
|
|
*
|
|
* @type is CAMEL_LOCK_WRITE or CAMEL_LOCK_READ,
|
|
* to create exclusive or shared read locks
|
|
*
|
|
* Return value: -1 on error.
|
|
**/
|
|
int
|
|
camel_lock_flock(int fd, CamelLockType type, CamelException *ex)
|
|
{
|
|
#ifdef USE_FLOCK
|
|
int op;
|
|
|
|
d(printf("flock locking %d\n", fd));
|
|
|
|
if (type == CAMEL_LOCK_READ)
|
|
op = LOCK_SH|LOCK_NB;
|
|
else
|
|
op = LOCK_EX|LOCK_NB;
|
|
|
|
if (flock(fd, op) == -1) {
|
|
camel_exception_setv (ex, CAMEL_EXCEPTION_SYSTEM,
|
|
_("Failed to get lock using flock(2): %s"),
|
|
g_strerror (errno));
|
|
return -1;
|
|
}
|
|
#endif
|
|
return 0;
|
|
}
|
|
|
|
/**
|
|
* camel_unlock_flock:
|
|
* @fd:
|
|
*
|
|
* Unlock an flock lock.
|
|
**/
|
|
void
|
|
camel_unlock_flock(int fd)
|
|
{
|
|
#ifdef USE_FLOCK
|
|
d(printf("flock unlocking %d\n", fd));
|
|
|
|
(void)flock(fd, LOCK_UN);
|
|
#endif
|
|
}
|
|
|
|
/**
|
|
* camel_lock_folder:
|
|
* @path: Path to the file to lock (used for .locking only).
|
|
* @fd: Open file descriptor of the right type to lock.
|
|
* @type: Type of lock, CAMEL_LOCK_READ or CAMEL_LOCK_WRITE.
|
|
* @ex:
|
|
*
|
|
* Attempt to lock a folder, multiple attempts will be made using all
|
|
* locking strategies available.
|
|
*
|
|
* Return value: -1 on error, @ex will describe the locking system that failed.
|
|
**/
|
|
int
|
|
camel_lock_folder(const char *path, int fd, CamelLockType type, CamelException *ex)
|
|
{
|
|
int retry = 0;
|
|
|
|
while (retry < CAMEL_LOCK_RETRY) {
|
|
if (retry > 0)
|
|
sleep(CAMEL_LOCK_DELAY);
|
|
|
|
camel_exception_clear(ex);
|
|
|
|
if (camel_lock_fcntl(fd, type, ex) == 0) {
|
|
if (camel_lock_flock(fd, type, ex) == 0) {
|
|
if (camel_lock_dot(path, ex) == 0)
|
|
return 0;
|
|
camel_unlock_flock(fd);
|
|
}
|
|
camel_unlock_fcntl(fd);
|
|
}
|
|
retry++;
|
|
}
|
|
|
|
return -1;
|
|
}
|
|
|
|
/**
|
|
* camel_unlock_folder:
|
|
* @path: Filename of folder.
|
|
* @fd: Open descrptor on which locks were placed.
|
|
*
|
|
* Free a lock on a folder.
|
|
**/
|
|
void
|
|
camel_unlock_folder(const char *path, int fd)
|
|
{
|
|
camel_unlock_dot(path);
|
|
camel_unlock_flock(fd);
|
|
camel_unlock_fcntl(fd);
|
|
}
|
|
|
|
#if 0
|
|
int main(int argc, char **argv)
|
|
{
|
|
CamelException *ex;
|
|
int fd1, fd2;
|
|
|
|
ex = camel_exception_new();
|
|
|
|
#if 0
|
|
if (camel_lock_dot("mylock", ex) == 0) {
|
|
if (camel_lock_dot("mylock", ex) == 0) {
|
|
printf("Got lock twice?\n");
|
|
} else {
|
|
printf("failed to get lock 2: %s\n", camel_exception_get_description(ex));
|
|
}
|
|
camel_unlock_dot("mylock");
|
|
} else {
|
|
printf("failed to get lock 1: %s\n", camel_exception_get_description(ex));
|
|
}
|
|
|
|
camel_exception_clear(ex);
|
|
#endif
|
|
|
|
fd1 = open("mylock", O_RDWR);
|
|
fd2 = open("mylock", O_RDWR);
|
|
|
|
if (camel_lock_fcntl(fd1, CAMEL_LOCK_WRITE, ex) == 0) {
|
|
printf("got fcntl write lock once\n");
|
|
sleep(5);
|
|
if (camel_lock_fcntl(fd2, CAMEL_LOCK_WRITE, ex) == 0) {
|
|
printf("got fcntl write lock twice!\n");
|
|
} else {
|
|
printf("failed to get write lock: %s\n", camel_exception_get_description(ex));
|
|
}
|
|
|
|
camel_exception_clear(ex);
|
|
|
|
if (camel_lock_fcntl(fd2, CAMEL_LOCK_READ, ex) == 0) {
|
|
printf("got fcntl read lock as well?\n");
|
|
camel_unlock_fcntl(fd2);
|
|
} else {
|
|
printf("failed to get read lock: %s\n", camel_exception_get_description(ex));
|
|
}
|
|
|
|
camel_exception_clear(ex);
|
|
camel_unlock_fcntl(fd1);
|
|
} else {
|
|
printf("failed to get write lock at all: %s\n", camel_exception_get_description(ex));
|
|
}
|
|
|
|
if (camel_lock_fcntl(fd1, CAMEL_LOCK_READ, ex) == 0) {
|
|
printf("got fcntl read lock once\n");
|
|
sleep(5);
|
|
if (camel_lock_fcntl(fd2, CAMEL_LOCK_WRITE, ex) == 0) {
|
|
printf("got fcntl write lock too?!\n");
|
|
} else {
|
|
printf("failed to get write lock: %s\n", camel_exception_get_description(ex));
|
|
}
|
|
|
|
camel_exception_clear(ex);
|
|
|
|
if (camel_lock_fcntl(fd2, CAMEL_LOCK_READ, ex) == 0) {
|
|
printf("got fcntl read lock twice\n");
|
|
camel_unlock_fcntl(fd2);
|
|
} else {
|
|
printf("failed to get read lock: %s\n", camel_exception_get_description(ex));
|
|
}
|
|
|
|
camel_exception_clear(ex);
|
|
camel_unlock_fcntl(fd1);
|
|
}
|
|
|
|
close(fd1);
|
|
close(fd2);
|
|
|
|
return 0;
|
|
}
|
|
#endif
|