usr/src/lib/libc/port/gen/lckpwdf.c
author eschrock
Fri, 19 Aug 2005 11:56:37 -0700
changeset 380 4ae8f505c115
parent 0 68f95e015346
child 6812 febeba71273d
permissions -rw-r--r--
6307489 getmntany() should not stat meaningless special devices
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
     1
/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
     2
 * CDDL HEADER START
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
     3
 *
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
     4
 * The contents of this file are subject to the terms of the
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
     5
 * Common Development and Distribution License, Version 1.0 only
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
     6
 * (the "License").  You may not use this file except in compliance
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
     7
 * with the License.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
     8
 *
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
     9
 * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    10
 * or http://www.opensolaris.org/os/licensing.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    11
 * See the License for the specific language governing permissions
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    12
 * and limitations under the License.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    13
 *
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    14
 * When distributing Covered Code, include this CDDL HEADER in each
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    15
 * file and include the License file at usr/src/OPENSOLARIS.LICENSE.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    16
 * If applicable, add the following below this CDDL HEADER, with the
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    17
 * fields enclosed by brackets "[]" replaced with your own identifying
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    18
 * information: Portions Copyright [yyyy] [name of copyright owner]
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    19
 *
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    20
 * CDDL HEADER END
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    21
 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    22
/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    23
 * Copyright 2004 Sun Microsystems, Inc.  All rights reserved.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    24
 * Use is subject to license terms.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    25
 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    26
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    27
#pragma ident	"%Z%%M%	%I%	%E% SMI"
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    28
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    29
/*	Copyright (c) 1988 AT&T	*/
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    30
/*	  All Rights Reserved  	*/
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    31
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    32
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    33
#pragma weak lckpwdf = _lckpwdf
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    34
#pragma weak ulckpwdf = _ulckpwdf
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    35
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    36
#include "synonyms.h"
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    37
#include <sys/types.h>
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    38
#include <stdio.h>
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    39
#include <signal.h>
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    40
#include <fcntl.h>
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    41
#include <unistd.h>
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    42
#include <shadow.h>
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    43
#include <errno.h>
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    44
#include <thread.h>
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    45
#include "mtlib.h"
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    46
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    47
#define	LOCKFILE	"/etc/.pwd.lock"
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    48
#define	S_WAITTIME	15
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    49
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    50
static struct flock flock =	{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    51
			0,	/* l_type */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    52
			0,	/* l_whence */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    53
			0,	/* l_start */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    54
			0,	/* l_len */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    55
			0,	/* l_sysid */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    56
			0	/* l_pid */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    57
			};
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    58
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    59
/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    60
 * lckpwdf() returns a 0 for a successful lock within W_WAITTIME
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    61
 * seconds and -1 otherwise.  We stand on our head to make it MT-safe.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    62
 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    63
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    64
static pid_t lck_pid = 0;	/* process's pid at last lock */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    65
static thread_t lck_tid = 0;	/* thread that holds the lock */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    66
static int fildes = -1;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    67
static mutex_t lck_lock = DEFAULTMUTEX;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    68
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    69
int
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    70
lckpwdf(void)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    71
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    72
	int seconds = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    73
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    74
	lmutex_lock(&lck_lock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    75
	for (;;) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    76
		if (lck_pid != 0 && lck_pid != getpid()) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    77
			/* somebody forked */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    78
			lck_pid = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    79
			lck_tid = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    80
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    81
		if (lck_tid == 0) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    82
			if ((fildes = creat(LOCKFILE, 0600)) == -1)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    83
				break;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    84
			flock.l_type = F_WRLCK;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    85
			if (fcntl(fildes, F_SETLK, &flock) != -1) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    86
				lck_pid = getpid();
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    87
				lck_tid = thr_self();
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    88
				lmutex_unlock(&lck_lock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    89
				return (0);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    90
			}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    91
			(void) close(fildes);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    92
			fildes = -1;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    93
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    94
		if (seconds++ >= S_WAITTIME) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    95
			/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    96
			 * For compatibility with the past, pretend
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    97
			 * that we were interrupted by SIGALRM.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    98
			 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    99
			errno = EINTR;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   100
			break;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   101
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   102
		lmutex_unlock(&lck_lock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   103
		(void) sleep(1);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   104
		lmutex_lock(&lck_lock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   105
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   106
	lmutex_unlock(&lck_lock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   107
	return (-1);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   108
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   109
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   110
/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   111
 * ulckpwdf() returns 0 for a successful unlock and -1 otherwise
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   112
 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   113
int
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   114
ulckpwdf(void)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   115
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   116
	lmutex_lock(&lck_lock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   117
	if (lck_tid == thr_self() && fildes >= 0) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   118
		flock.l_type = F_UNLCK;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   119
		(void) fcntl(fildes, F_SETLK, &flock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   120
		(void) close(fildes);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   121
		fildes = -1;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   122
		lck_pid = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   123
		lck_tid = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   124
		lmutex_unlock(&lck_lock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   125
		return (0);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   126
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   127
	lmutex_unlock(&lck_lock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   128
	return (-1);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   129
}