]>
diplodocus.org Git - nmh/blob - sbr/lock_file.c
1 /* lock_file.c -- routines to lock/unlock files
3 * This code is Copyright (c) 2002, by the authors of nmh. See the
4 * COPYRIGHT file in the root directory of the nmh distribution for
5 * complete copyright information.
8 /* Modified by Ruud de Rooij to support Miquel van Smoorenburg's liblockfile
10 * Since liblockfile locking shares most of its code with dot locking, it
11 * is enabled by defining both DOT_LOCKING and HAVE_LIBLOCKFILE.
13 * Ruud de Rooij <ruud@debian.org> Sun, 28 Mar 1999 15:34:03 +0200
17 #include "context_find.h"
19 #include "h/signals.h"
22 #include "lock_file.h"
25 #ifdef HAVE_SYS_TIME_H
26 # include <sys/time.h>
31 # include <sys/file.h>
34 #if defined(HAVE_LIBLOCKFILE)
35 # include <lockfile.h>
39 char *lockdir
= LOCKDIR
;
42 /* struct for getting name of lock file to create */
45 #if !defined(HAVE_LIBLOCKFILE)
51 * Number of tries to retry locking
53 #define LOCK_RETRIES 60
56 * Amount of time to wait before
57 * updating ctime of lock file.
61 #if !defined(HAVE_LIBLOCKFILE)
63 * How old does a lock file need to be
64 * before we remove it.
67 #endif /* HAVE_LIBLOCKFILE */
69 /* struct for recording and updating locks */
76 enum locktype
{ FCNTL_LOCKING
, FLOCK_LOCKING
, LOCKF_LOCKING
, DOT_LOCKING
};
78 /* Our saved lock types. */
79 static enum locktype datalocktype
, spoollocktype
;
81 /* top of list containing all open locks */
82 static struct lock
*l_top
= NULL
;
84 static int lkopen(const char *, int, mode_t
, enum locktype
, int *);
85 static int str2accbits(const char *);
87 static int lkopen_fcntl (const char *, int, mode_t
, int *);
89 static int lkopen_lockf (const char *, int, mode_t
, int *);
90 #endif /* HAVE_LOCKF */
92 static int lkopen_flock (const char *, int, mode_t
, int *);
93 #endif /* HAVE_FLOCK */
95 static enum locktype
init_locktype(const char *) PURE
;
97 static int lkopen_dot (const char *, int, mode_t
, int *);
98 static void lkclose_dot (int, const char *);
99 static void lockname (const char *, struct lockinfo
*, int);
100 static void timerON (char *, int);
101 static void timerOFF (int);
102 static void alrmser (int);
104 #if !defined(HAVE_LIBLOCKFILE)
105 static int lockit (struct lockinfo
*);
109 * Base functions: determine the data type used to lock files and
110 * call the underlying function.
114 lkopendata(const char *file
, int access
, mode_t mode
, int *failed_to_lock
)
122 if ((dl
= context_find("datalocking"))) {
123 datalocktype
= init_locktype(dl
);
125 /* We default to fcntl locking for data files */
126 datalocktype
= FCNTL_LOCKING
;
130 return lkopen(file
, access
, mode
, datalocktype
, failed_to_lock
);
135 * Locking using the spool locking algorithm
139 lkopenspool(const char *file
, int access
, mode_t mode
, int *failed_to_lock
)
145 spoollocktype
= init_locktype(spoollocking
);
148 return lkopen(file
, access
, mode
, spoollocktype
, failed_to_lock
);
153 * Versions of lkopen that return a FILE *
157 lkfopendata(const char *file
, const char *mode
, int *failed_to_lock
)
160 int oflags
= str2accbits(mode
);
168 if ((fd
= lkopendata(file
, oflags
, 0666, failed_to_lock
)) == -1)
171 if ((fp
= fdopen (fd
, mode
)) == NULL
) {
180 lkfopenspool(const char *file
, const char *mode
)
183 int oflags
= str2accbits(mode
);
184 int failed_to_lock
= 0;
192 if ((fd
= lkopenspool(file
, oflags
, 0666, &failed_to_lock
)) == -1)
195 if ((fp
= fdopen (fd
, mode
)) == NULL
) {
205 * Corresponding close functions.
207 * A note here: All of the kernel locking functions terminate the lock
208 * when the descriptor is closed, so why write the code to explicitly
209 * unlock the file? We only need to do this in the dot-locking case.
213 lkclosedata(int fd
, const char *name
)
217 if (datalocktype
== DOT_LOCKING
)
218 lkclose_dot(fd
, name
);
224 lkfclosedata(FILE *f
, const char *name
)
234 if (datalocktype
== DOT_LOCKING
)
235 lkclose_dot(fd
, name
);
241 lkclosespool(int fd
, const char *name
)
245 if (spoollocktype
== DOT_LOCKING
)
246 lkclose_dot(fd
, name
);
252 lkfclosespool(FILE *f
, const char *name
)
262 if (spoollocktype
== DOT_LOCKING
)
263 lkclose_dot(fd
, name
);
270 * Convert fopen() mode argument to open() bits
274 str2accbits(const char *mode
)
276 if (strcmp (mode
, "r") == 0)
278 if (strcmp (mode
, "r+") == 0)
280 if (strcmp (mode
, "w") == 0)
281 return O_WRONLY
| O_CREAT
| O_TRUNC
;
282 if (strcmp (mode
, "w+") == 0)
283 return O_RDWR
| O_CREAT
| O_TRUNC
;
284 if (strcmp (mode
, "a") == 0)
285 return O_WRONLY
| O_CREAT
| O_APPEND
;
286 if (strcmp (mode
, "a+") == 0)
287 return O_RDWR
| O_CREAT
| O_APPEND
;
294 * Internal routine to switch between different locking types.
298 lkopen (const char *file
, int access
, mode_t mode
, enum locktype ltype
,
304 return lkopen_fcntl(file
, access
, mode
, failed_to_lock
);
307 return lkopen_dot(file
, access
, mode
, failed_to_lock
);
311 return lkopen_flock(file
, access
, mode
, failed_to_lock
);
312 #endif /* HAVE_FLOCK */
316 return lkopen_lockf(file
, access
, mode
, failed_to_lock
);
317 #endif /* HAVE_FLOCK */
320 die("Internal locking error: unsupported lock type used!");
328 * Routine to clean up the dot locking file
332 lkclose_dot (int fd
, const char *file
)
334 struct lockinfo lkinfo
;
336 lockname (file
, &lkinfo
, 0); /* get name of lock file */
337 #if !defined(HAVE_LIBLOCKFILE)
338 (void) m_unlink (lkinfo
.curlock
); /* remove lock file */
340 lockfile_remove(lkinfo
.curlock
);
341 #endif /* HAVE_LIBLOCKFILE */
342 timerOFF (fd
); /* turn off lock timer */
347 * Open and lock a file, using fcntl locking
351 lkopen_fcntl(const char *file
, int access
, mode_t mode
, int *failed_to_lock
)
353 int fd
, i
, saved_errno
;
357 * The assumption here is that if you open the file for writing, you
358 * need an exclusive lock.
361 for (i
= 0; i
< LOCK_RETRIES
; i
++) {
362 if ((fd
= open(file
, access
, mode
)) == -1)
367 flk
.l_type
= (access
& O_ACCMODE
) == O_RDONLY
? F_RDLCK
: F_WRLCK
;
368 flk
.l_whence
= SEEK_SET
;
370 if (fcntl(fd
, F_SETLK
, &flk
) != -1)
386 * Open and lock a file, using flock locking
390 lkopen_flock(const char *file
, int access
, mode_t mode
, int *failed_to_lock
)
392 int fd
, i
, saved_errno
, locktype
;
395 * The assumption here is that if you open the file for writing, you
396 * need an exclusive lock.
399 locktype
= (((access
& O_ACCMODE
) == O_RDONLY
) ? LOCK_SH
: LOCK_EX
) |
402 for (i
= 0; i
< LOCK_RETRIES
; i
++) {
403 if ((fd
= open(file
, access
, mode
)) == -1)
406 if (flock(fd
, locktype
) != -1)
418 #endif /* HAVE_FLOCK */
421 * Open and lock a file, using lockf locking
425 lkopen_lockf(const char *file
, int access
, mode_t mode
, int *failed_to_lock
)
427 int fd
, i
, saved_errno
, saved_access
;
432 * Because lockf locks start from the current offset, mask off O_APPEND
433 * and seek to the end of the file later if it was requested.
435 * lockf locks require write access to the file, so always add it
436 * even if it wasn't requested.
439 saved_access
= access
;
443 if ((access
& O_ACCMODE
) == O_RDONLY
) {
448 for (i
= 0; i
< LOCK_RETRIES
; i
++) {
449 if ((fd
= open(file
, access
, mode
)) == -1)
452 if (lockf(fd
, F_TLOCK
, 0) != -1) {
454 * Seek to end if requested
456 if (saved_access
& O_APPEND
) {
457 lseek(fd
, 0, SEEK_END
);
474 * open and lock a file, using dot locking
478 lkopen_dot (const char *file
, int access
, mode_t mode
, int *failed_to_lock
)
481 struct lockinfo lkinfo
;
484 if ((fd
= open (file
, access
, mode
)) == -1)
488 * Get the name of the eventual lock file, as well
489 * as a name for a temporary lock file.
491 lockname (file
, &lkinfo
, 1);
493 #if !defined(HAVE_LIBLOCKFILE)
496 for (i
= 0; i
< LOCK_RETRIES
; ++i
) {
499 /* attempt to create lock file */
500 if (lockit (&lkinfo
) == 0) {
501 /* if successful, turn on timer and return */
502 timerON (lkinfo
.curlock
, fd
);
507 * Abort locking, if we fail to lock after 5 attempts
508 * and are never able to stat the lock file. Or, if
509 * we can stat the lockfile but exceed LOCK_RETRIES
510 * seconds waiting for it (by falling out of the loop).
512 if (stat (lkinfo
.curlock
, &st
) == -1) {
519 /* check for stale lockfile, else sleep */
520 if (curtime
> st
.st_ctime
+ RSECS
)
521 (void) m_unlink (lkinfo
.curlock
);
525 lockname (file
, &lkinfo
, 1);
533 if (lockfile_create(lkinfo
.curlock
, 5, 0) == L_SUCCESS
) {
534 timerON(lkinfo
.curlock
, fd
);
541 #endif /* HAVE_LIBLOCKFILE */
544 #if !defined(HAVE_LIBLOCKFILE)
546 * Routine that actually tries to create
551 lockit (struct lockinfo
*li
)
554 char *curlock
, *tmpfile
;
560 curlock
= li
->curlock
;
562 if ((tmpfile
= m_mktemp(li
->tmplock
, &fd
, NULL
)) == NULL
) {
563 inform("unable to create temporary file in %s", li
->tmplock
);
568 /* write our process id into lock file */
569 snprintf (buffer
, sizeof(buffer
), "nmh lock: pid %d\n", (int) getpid());
570 write(fd
, buffer
, strlen(buffer
) + 1);
576 * Now try to create the real lock file
577 * by linking to the temporary file.
579 fd
= link(tmpfile
, curlock
);
580 (void) m_unlink(tmpfile
);
582 return fd
== -1 ? -1 : 0;
584 #endif /* HAVE_LIBLOCKFILE */
587 * Get name of lock file, and temporary lock file
591 lockname (const char *file
, struct lockinfo
*li
, int isnewlock
)
601 if ((cp
= strrchr (file
, '/')) == NULL
|| *++cp
== 0)
607 snprintf (bp
, sizeof(li
->curlock
), "%s/", lockdir
);
608 tmplen
= strlen (bp
);
613 snprintf (bp
, sizeof(li
->curlock
), "%.*s", (int)(cp
- file
), file
);
614 tmplen
= strlen (bp
);
622 * mmdf style dot locking. Currently not supported.
623 * If we start supporting mmdf style dot locking,
624 * we will need to change the return value of lockname
626 if (stat (file
, &st
) == -1)
629 snprintf (bp
, sizeof(li
->curlock
) - bplen
, "LCK%05d.%05d",
630 st
.st_dev
, st
.st_ino
);
633 snprintf (bp
, sizeof(li
->curlock
) - bplen
, "%s.lock", cp
);
635 #if defined(HAVE_LIBLOCKFILE)
636 NMH_UNUSED(isnewlock
);
639 * If this is for a new lock, create a name for
640 * the temporary lock file for lockit()
643 if ((cp
= strrchr (li
->curlock
, '/')) == NULL
|| *++cp
== 0)
644 strncpy (li
->tmplock
, ",LCK.XXXXXX", sizeof(li
->tmplock
));
646 snprintf (li
->tmplock
, sizeof(li
->tmplock
), "%.*s,LCK.XXXXXX",
647 (int)(cp
- li
->curlock
), li
->curlock
);
654 * Add new lockfile to the list of open lockfiles
655 * and start the lock file timer.
659 timerON (char *curlock
, int fd
)
664 lp
->l_lock
= mh_xstrdup(curlock
);
669 /* perhaps SIGT{STP,TIN,TOU} ? */
670 SIGNAL (SIGALRM
, alrmser
);
678 * Search through the list of lockfiles for the
679 * current lockfile, and remove it from the list.
685 struct lock
*pp
, *lp
;
690 for (pp
= lp
= l_top
; lp
; pp
= lp
, lp
= lp
->l_next
) {
698 pp
->l_next
= lp
->l_next
;
705 /* if there are locks left, restart timer */
712 * If timer goes off, we update the ctime of all open
713 * lockfiles, so another command doesn't remove them.
723 /* update the ctime of all the lock files */
724 for (lp
= l_top
; lp
; lp
= lp
->l_next
) {
725 lockfile
= lp
->l_lock
;
726 #if !defined(HAVE_LIBLOCKFILE)
729 if (*lockfile
&& (j
= creat (lockfile
, 0600)) != -1)
733 lockfile_touch(lockfile
);
737 /* restart the alarm */
743 * Return a locking algorithm based on the string name
747 init_locktype(const char *lockname
)
749 if (strcasecmp(lockname
, "fcntl") == 0) {
750 return FCNTL_LOCKING
;
752 if (strcasecmp(lockname
, "lockf") == 0) {
754 return LOCKF_LOCKING
;
755 #else /* ! HAVE_LOCKF */
756 die("lockf not supported on this system");
757 #endif /* HAVE_LOCKF */
759 if (strcasecmp(lockname
, "flock") == 0) {
761 return FLOCK_LOCKING
;
762 #else /* ! HAVE_FLOCK */
763 die("flock not supported on this system");
764 #endif /* HAVE_FLOCK */
766 if (strcasecmp(lockname
, "dot") == 0) {
769 die("Unknown lock type: \"%s\"", lockname
);