maillock.c revision 7257d1b4d25bfac0c802847390e98a464fd787ac
/*
* CDDL HEADER START
*
* The contents of this file are subject to the terms of the
* Common Development and Distribution License (the "License").
* You may not use this file except in compliance with the License.
*
* You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
* or http://www.opensolaris.org/os/licensing.
* See the License for the specific language governing permissions
* and limitations under the License.
*
* When distributing Covered Code, include this CDDL HEADER in each
* file and include the License file at usr/src/OPENSOLARIS.LICENSE.
* If applicable, add the following below this CDDL HEADER, with the
* fields enclosed by brackets "[]" replaced with your own identifying
* information: Portions Copyright [yyyy] [name of copyright owner]
*
* CDDL HEADER END
*/
/*
* Copyright 2008 Sun Microsystems, Inc. All rights reserved.
* Use is subject to license terms.
*/
/* Copyright (c) 1984, 1986, 1987, 1988, 1989 AT&T */
/* All Rights Reserved */
#pragma ident "%Z%%M% %I% %E% SMI"
#include "maillock.h"
#include <sys/types.h>
#include <fcntl.h>
#include <stdio.h>
#include <string.h>
#include <unistd.h>
#include <stdlib.h>
#include <utime.h>
#include <sys/stat.h>
static char *lockext = ".lock"; /* Lock suffix for mailname */
static char curlock[PATHSIZE]; /* Last used name of lock */
static int locked; /* To note that we locked it */
static time_t locktime; /* time lock file was touched */
static time_t lock1(char *, char *);
/*
* Lock the specified mail file by setting the file mailfile.lock.
* We must, of course, be careful to remove the lock file by a call
* to unlock before we stop. The algorithm used here is to see if
* the lock exists, and if it does, to check its modify time. If it
* is older than 5 minutes, we assume error and set our own file.
* Otherwise, we wait for 5 seconds and try again.
*/
/*ARGSUSED*/
int
maillock(char *user, int retrycnt)
{
time_t t;
struct stat sbuf;
int statfailed;
char locktmp[PATHSIZE]; /* Usable lock temporary */
char file[PATHSIZE];
if (locked)
return (0);
(void) strcpy(file, MAILDIR);
(void) strcat(file, user);
(void) strcpy(curlock, file);
(void) strcat(curlock, lockext);
(void) strcpy(locktmp, file);
(void) strcat(locktmp, "XXXXXX");
(void) mktemp(locktmp);
(void) remove(locktmp);
statfailed = 0;
for (;;) {
t = lock1(locktmp, curlock);
if (t == (time_t)0) {
locked = 1;
locktime = time(0);
return (0);
}
if (stat(curlock, &sbuf) < 0) {
if (statfailed++ > 5)
return (-1);
(void) sleep(5);
continue;
}
statfailed = 0;
/*
* Compare the time of the temp file with the time
* of the lock file, rather than with the current
* time of day, since the files may reside on
* another machine whose time of day differs from
* ours. If the lock file is less than 5 minutes
* old, keep trying.
*/
if (t < sbuf.st_ctime + 300) {
(void) sleep(5);
continue;
}
(void) remove(curlock);
}
}
/*
* Remove the mail lock, and note that we no longer
* have it locked.
*/
void
mailunlock(void)
{
(void) remove(curlock);
locked = 0;
}
/*
* Attempt to set the lock by creating the temporary file,
* then doing a link/unlink. If it succeeds, return 0,
* else return a guess of the current time on the machine
* holding the file.
*/
static time_t
lock1(char tempfile[], char name[])
{
int fd;
struct stat sbuf;
fd = open(tempfile, O_RDWR|O_CREAT|O_EXCL, 0600);
if (fd < 0)
return (time(0));
(void) fstat(fd, &sbuf);
/*
* Write the string "0" into the lock file to give us some
* interoperability with SVR4 mailers. SVR4 mailers expect
* a process ID to be written into the lock file and then
* use kill() to see if the process is alive or not. We write
* 0 into it so that SVR4 mailers will always think our lock file
* is valid.
*/
(void) write(fd, "0", 2);
(void) close(fd);
if (link(tempfile, name) < 0) {
(void) remove(tempfile);
return (sbuf.st_ctime);
}
(void) remove(tempfile);
return ((time_t)0);
}
/*
* Update the change time on the lock file so
* others will know we're still using it.
*/
void
touchlock(void)
{
struct stat sbuf;
time_t t;
struct utimbuf tp;
if (!locked)
return;
/* if it hasn't been at least 3 minutes, don't bother */
if (time(&t) < locktime + 180)
return;
locktime = t;
if (stat(curlock, &sbuf) < 0)
return;
/*
* Don't actually change the times, we just want the
* side effect that utime causes st_ctime to be set
* to the current time.
*/
tp.actime = sbuf.st_atime;
tp.modtime = sbuf.st_mtime;
(void) utime(curlock, &tp);
}