import less(1)
[unleashed/tickless.git] / usr / src / lib / libc / port / gen / lckpwdf.c
blob030660d107f9bd3e03998a2728d2ffb100e33ec5
1 /*
2 * CDDL HEADER START
4 * The contents of this file are subject to the terms of the
5 * Common Development and Distribution License (the "License").
6 * You may not use this file except in compliance with the License.
8 * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
9 * or http://www.opensolaris.org/os/licensing.
10 * See the License for the specific language governing permissions
11 * and limitations under the License.
13 * When distributing Covered Code, include this CDDL HEADER in each
14 * file and include the License file at usr/src/OPENSOLARIS.LICENSE.
15 * If applicable, add the following below this CDDL HEADER, with the
16 * fields enclosed by brackets "[]" replaced with your own identifying
17 * information: Portions Copyright [yyyy] [name of copyright owner]
19 * CDDL HEADER END
23 * Copyright 2008 Sun Microsystems, Inc. All rights reserved.
24 * Use is subject to license terms.
27 /* Copyright (c) 1988 AT&T */
28 /* All Rights Reserved */
30 #pragma ident "%Z%%M% %I% %E% SMI"
32 #include "lint.h"
33 #include <sys/types.h>
34 #include <stdio.h>
35 #include <signal.h>
36 #include <fcntl.h>
37 #include <unistd.h>
38 #include <shadow.h>
39 #include <errno.h>
40 #include <thread.h>
41 #include "mtlib.h"
43 #define LOCKFILE "/etc/.pwd.lock"
44 #define S_WAITTIME 15
46 static struct flock flock = {
47 0, /* l_type */
48 0, /* l_whence */
49 0, /* l_start */
50 0, /* l_len */
51 0, /* l_sysid */
52 0 /* l_pid */
56 * lckpwdf() returns a 0 for a successful lock within W_WAITTIME
57 * seconds and -1 otherwise. We stand on our head to make it MT-safe.
60 static pid_t lck_pid = 0; /* process's pid at last lock */
61 static thread_t lck_tid = 0; /* thread that holds the lock */
62 static int fildes = -1;
63 static mutex_t lck_lock = DEFAULTMUTEX;
65 int
66 lckpwdf(void)
68 int seconds = 0;
70 lmutex_lock(&lck_lock);
71 for (;;) {
72 if (lck_pid != 0 && lck_pid != getpid()) {
73 /* somebody forked */
74 lck_pid = 0;
75 lck_tid = 0;
77 if (lck_tid == 0) {
78 if ((fildes = creat(LOCKFILE, 0600)) == -1)
79 break;
80 flock.l_type = F_WRLCK;
81 if (fcntl(fildes, F_SETLK, &flock) != -1) {
82 lck_pid = getpid();
83 lck_tid = thr_self();
84 lmutex_unlock(&lck_lock);
85 return (0);
87 (void) close(fildes);
88 fildes = -1;
90 if (seconds++ >= S_WAITTIME) {
92 * For compatibility with the past, pretend
93 * that we were interrupted by SIGALRM.
95 errno = EINTR;
96 break;
98 lmutex_unlock(&lck_lock);
99 (void) sleep(1);
100 lmutex_lock(&lck_lock);
102 lmutex_unlock(&lck_lock);
103 return (-1);
107 * ulckpwdf() returns 0 for a successful unlock and -1 otherwise
110 ulckpwdf(void)
112 lmutex_lock(&lck_lock);
113 if (lck_tid == thr_self() && fildes >= 0) {
114 flock.l_type = F_UNLCK;
115 (void) fcntl(fildes, F_SETLK, &flock);
116 (void) close(fildes);
117 fildes = -1;
118 lck_pid = 0;
119 lck_tid = 0;
120 lmutex_unlock(&lck_lock);
121 return (0);
123 lmutex_unlock(&lck_lock);
124 return (-1);