usr/src/uts/common/fs/zfs/rprwlock.c
author ck153898
Mon, 29 Oct 2007 22:45:33 -0700
changeset 5378 111aa1baa84a
parent 4787 602d3f97842c
permissions -rw-r--r--
PSARC 2007/555 zfs fs-only quotas and reservations 6431277 want filesystem-only quotas 6483677 need immediate reservation
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
4787
602d3f97842c 6393351 unique_* could be improved
ahrens
parents:
diff changeset
     1
/*
602d3f97842c 6393351 unique_* could be improved
ahrens
parents:
diff changeset
     2
 * CDDL HEADER START
602d3f97842c 6393351 unique_* could be improved
ahrens
parents:
diff changeset
     3
 *
602d3f97842c 6393351 unique_* could be improved
ahrens
parents:
diff changeset
     4
 * The contents of this file are subject to the terms of the
602d3f97842c 6393351 unique_* could be improved
ahrens
parents:
diff changeset
     5
 * Common Development and Distribution License (the "License").
602d3f97842c 6393351 unique_* could be improved
ahrens
parents:
diff changeset
     6
 * You may not use this file except in compliance with the License.
602d3f97842c 6393351 unique_* could be improved
ahrens
parents:
diff changeset
     7
 *
602d3f97842c 6393351 unique_* could be improved
ahrens
parents:
diff changeset
     8
 * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
602d3f97842c 6393351 unique_* could be improved
ahrens
parents:
diff changeset
     9
 * or http://www.opensolaris.org/os/licensing.
602d3f97842c 6393351 unique_* could be improved
ahrens
parents:
diff changeset
    10
 * See the License for the specific language governing permissions
602d3f97842c 6393351 unique_* could be improved
ahrens
parents:
diff changeset
    11
 * and limitations under the License.
602d3f97842c 6393351 unique_* could be improved
ahrens
parents:
diff changeset
    12
 *
602d3f97842c 6393351 unique_* could be improved
ahrens
parents:
diff changeset
    13
 * When distributing Covered Code, include this CDDL HEADER in each
602d3f97842c 6393351 unique_* could be improved
ahrens
parents:
diff changeset
    14
 * file and include the License file at usr/src/OPENSOLARIS.LICENSE.
602d3f97842c 6393351 unique_* could be improved
ahrens
parents:
diff changeset
    15
 * If applicable, add the following below this CDDL HEADER, with the
602d3f97842c 6393351 unique_* could be improved
ahrens
parents:
diff changeset
    16
 * fields enclosed by brackets "[]" replaced with your own identifying
602d3f97842c 6393351 unique_* could be improved
ahrens
parents:
diff changeset
    17
 * information: Portions Copyright [yyyy] [name of copyright owner]
602d3f97842c 6393351 unique_* could be improved
ahrens
parents:
diff changeset
    18
 *
602d3f97842c 6393351 unique_* could be improved
ahrens
parents:
diff changeset
    19
 * CDDL HEADER END
602d3f97842c 6393351 unique_* could be improved
ahrens
parents:
diff changeset
    20
 */
602d3f97842c 6393351 unique_* could be improved
ahrens
parents:
diff changeset
    21
/*
602d3f97842c 6393351 unique_* could be improved
ahrens
parents:
diff changeset
    22
 * Copyright 2007 Sun Microsystems, Inc.  All rights reserved.
602d3f97842c 6393351 unique_* could be improved
ahrens
parents:
diff changeset
    23
 * Use is subject to license terms.
602d3f97842c 6393351 unique_* could be improved
ahrens
parents:
diff changeset
    24
 */
602d3f97842c 6393351 unique_* could be improved
ahrens
parents:
diff changeset
    25
602d3f97842c 6393351 unique_* could be improved
ahrens
parents:
diff changeset
    26
#pragma ident	"%Z%%M%	%I%	%E% SMI"
602d3f97842c 6393351 unique_* could be improved
ahrens
parents:
diff changeset
    27
602d3f97842c 6393351 unique_* could be improved
ahrens
parents:
diff changeset
    28
#include <sys/zfs_context.h>
602d3f97842c 6393351 unique_* could be improved
ahrens
parents:
diff changeset
    29
#include <sys/refcount.h>
602d3f97842c 6393351 unique_* could be improved
ahrens
parents:
diff changeset
    30
#include <sys/rprwlock.h>
602d3f97842c 6393351 unique_* could be improved
ahrens
parents:
diff changeset
    31
602d3f97842c 6393351 unique_* could be improved
ahrens
parents:
diff changeset
    32
void
602d3f97842c 6393351 unique_* could be improved
ahrens
parents:
diff changeset
    33
rprw_init(rprwlock_t *rwl)
602d3f97842c 6393351 unique_* could be improved
ahrens
parents:
diff changeset
    34
{
602d3f97842c 6393351 unique_* could be improved
ahrens
parents:
diff changeset
    35
	mutex_init(&rwl->rw_lock, NULL, MUTEX_DEFAULT, NULL);
602d3f97842c 6393351 unique_* could be improved
ahrens
parents:
diff changeset
    36
	rwl->rw_writer = NULL;
602d3f97842c 6393351 unique_* could be improved
ahrens
parents:
diff changeset
    37
	cv_init(&rwl->rw_cv, NULL, CV_DEFAULT, NULL);
602d3f97842c 6393351 unique_* could be improved
ahrens
parents:
diff changeset
    38
	refcount_create(&rwl->rw_count);
602d3f97842c 6393351 unique_* could be improved
ahrens
parents:
diff changeset
    39
}
602d3f97842c 6393351 unique_* could be improved
ahrens
parents:
diff changeset
    40
602d3f97842c 6393351 unique_* could be improved
ahrens
parents:
diff changeset
    41
void
602d3f97842c 6393351 unique_* could be improved
ahrens
parents:
diff changeset
    42
rprw_destroy(rprwlock_t *rwl)
602d3f97842c 6393351 unique_* could be improved
ahrens
parents:
diff changeset
    43
{
602d3f97842c 6393351 unique_* could be improved
ahrens
parents:
diff changeset
    44
	mutex_destroy(&rwl->rw_lock);
602d3f97842c 6393351 unique_* could be improved
ahrens
parents:
diff changeset
    45
	ASSERT(rwl->rw_writer == NULL);
602d3f97842c 6393351 unique_* could be improved
ahrens
parents:
diff changeset
    46
	cv_destroy(&rwl->rw_cv);
602d3f97842c 6393351 unique_* could be improved
ahrens
parents:
diff changeset
    47
	refcount_destroy(&rwl->rw_count);
602d3f97842c 6393351 unique_* could be improved
ahrens
parents:
diff changeset
    48
}
602d3f97842c 6393351 unique_* could be improved
ahrens
parents:
diff changeset
    49
602d3f97842c 6393351 unique_* could be improved
ahrens
parents:
diff changeset
    50
void
602d3f97842c 6393351 unique_* could be improved
ahrens
parents:
diff changeset
    51
rprw_enter_read(rprwlock_t *rwl, void *tag)
602d3f97842c 6393351 unique_* could be improved
ahrens
parents:
diff changeset
    52
{
602d3f97842c 6393351 unique_* could be improved
ahrens
parents:
diff changeset
    53
	mutex_enter(&rwl->rw_lock);
602d3f97842c 6393351 unique_* could be improved
ahrens
parents:
diff changeset
    54
602d3f97842c 6393351 unique_* could be improved
ahrens
parents:
diff changeset
    55
	if (rwl->rw_writer != curthread) {
602d3f97842c 6393351 unique_* could be improved
ahrens
parents:
diff changeset
    56
		while (rwl->rw_writer != NULL)
602d3f97842c 6393351 unique_* could be improved
ahrens
parents:
diff changeset
    57
			cv_wait(&rwl->rw_cv, &rwl->rw_lock);
602d3f97842c 6393351 unique_* could be improved
ahrens
parents:
diff changeset
    58
	}
602d3f97842c 6393351 unique_* could be improved
ahrens
parents:
diff changeset
    59
602d3f97842c 6393351 unique_* could be improved
ahrens
parents:
diff changeset
    60
	(void) refcount_add(&rwl->rw_count, tag);
602d3f97842c 6393351 unique_* could be improved
ahrens
parents:
diff changeset
    61
602d3f97842c 6393351 unique_* could be improved
ahrens
parents:
diff changeset
    62
	mutex_exit(&rwl->rw_lock);
602d3f97842c 6393351 unique_* could be improved
ahrens
parents:
diff changeset
    63
}
602d3f97842c 6393351 unique_* could be improved
ahrens
parents:
diff changeset
    64
602d3f97842c 6393351 unique_* could be improved
ahrens
parents:
diff changeset
    65
void
602d3f97842c 6393351 unique_* could be improved
ahrens
parents:
diff changeset
    66
rprw_enter_write(rprwlock_t *rwl, void *tag)
602d3f97842c 6393351 unique_* could be improved
ahrens
parents:
diff changeset
    67
{
602d3f97842c 6393351 unique_* could be improved
ahrens
parents:
diff changeset
    68
	mutex_enter(&rwl->rw_lock);
602d3f97842c 6393351 unique_* could be improved
ahrens
parents:
diff changeset
    69
602d3f97842c 6393351 unique_* could be improved
ahrens
parents:
diff changeset
    70
	if (rwl->rw_writer != curthread) {
602d3f97842c 6393351 unique_* could be improved
ahrens
parents:
diff changeset
    71
		while (!refcount_is_zero(&rwl->rw_count))
602d3f97842c 6393351 unique_* could be improved
ahrens
parents:
diff changeset
    72
			cv_wait(&rwl->rw_cv, &rwl->rw_lock);
602d3f97842c 6393351 unique_* could be improved
ahrens
parents:
diff changeset
    73
		rwl->rw_writer = curthread;
602d3f97842c 6393351 unique_* could be improved
ahrens
parents:
diff changeset
    74
	}
602d3f97842c 6393351 unique_* could be improved
ahrens
parents:
diff changeset
    75
602d3f97842c 6393351 unique_* could be improved
ahrens
parents:
diff changeset
    76
	(void) refcount_add(&rwl->rw_count, tag);
602d3f97842c 6393351 unique_* could be improved
ahrens
parents:
diff changeset
    77
602d3f97842c 6393351 unique_* could be improved
ahrens
parents:
diff changeset
    78
	mutex_exit(&rwl->rw_lock);
602d3f97842c 6393351 unique_* could be improved
ahrens
parents:
diff changeset
    79
}
602d3f97842c 6393351 unique_* could be improved
ahrens
parents:
diff changeset
    80
602d3f97842c 6393351 unique_* could be improved
ahrens
parents:
diff changeset
    81
void
602d3f97842c 6393351 unique_* could be improved
ahrens
parents:
diff changeset
    82
rprw_enter(rprwlock_t *rwl, krw_t rw, void *tag)
602d3f97842c 6393351 unique_* could be improved
ahrens
parents:
diff changeset
    83
{
602d3f97842c 6393351 unique_* could be improved
ahrens
parents:
diff changeset
    84
	if (rw == RW_READER)
602d3f97842c 6393351 unique_* could be improved
ahrens
parents:
diff changeset
    85
		rprw_enter_read(rwl, tag);
602d3f97842c 6393351 unique_* could be improved
ahrens
parents:
diff changeset
    86
	else
602d3f97842c 6393351 unique_* could be improved
ahrens
parents:
diff changeset
    87
		rprw_enter_write(rwl, tag);
602d3f97842c 6393351 unique_* could be improved
ahrens
parents:
diff changeset
    88
}
602d3f97842c 6393351 unique_* could be improved
ahrens
parents:
diff changeset
    89
602d3f97842c 6393351 unique_* could be improved
ahrens
parents:
diff changeset
    90
void
602d3f97842c 6393351 unique_* could be improved
ahrens
parents:
diff changeset
    91
rprw_exit(rprwlock_t *rwl, void *tag)
602d3f97842c 6393351 unique_* could be improved
ahrens
parents:
diff changeset
    92
{
602d3f97842c 6393351 unique_* could be improved
ahrens
parents:
diff changeset
    93
	mutex_enter(&rwl->rw_lock);
602d3f97842c 6393351 unique_* could be improved
ahrens
parents:
diff changeset
    94
602d3f97842c 6393351 unique_* could be improved
ahrens
parents:
diff changeset
    95
	ASSERT(!refcount_is_zero(&rwl->rw_count));
602d3f97842c 6393351 unique_* could be improved
ahrens
parents:
diff changeset
    96
	ASSERT(rwl->rw_writer == NULL || curthread == rwl->rw_writer);
602d3f97842c 6393351 unique_* could be improved
ahrens
parents:
diff changeset
    97
	if (refcount_remove(&rwl->rw_count, tag) == 0) {
602d3f97842c 6393351 unique_* could be improved
ahrens
parents:
diff changeset
    98
		cv_broadcast(&rwl->rw_cv);
602d3f97842c 6393351 unique_* could be improved
ahrens
parents:
diff changeset
    99
		rwl->rw_writer = NULL;  /* OK in either case */
602d3f97842c 6393351 unique_* could be improved
ahrens
parents:
diff changeset
   100
	}
602d3f97842c 6393351 unique_* could be improved
ahrens
parents:
diff changeset
   101
602d3f97842c 6393351 unique_* could be improved
ahrens
parents:
diff changeset
   102
	mutex_exit(&rwl->rw_lock);
602d3f97842c 6393351 unique_* could be improved
ahrens
parents:
diff changeset
   103
}
602d3f97842c 6393351 unique_* could be improved
ahrens
parents:
diff changeset
   104
602d3f97842c 6393351 unique_* could be improved
ahrens
parents:
diff changeset
   105
boolean_t
602d3f97842c 6393351 unique_* could be improved
ahrens
parents:
diff changeset
   106
rprw_held(rprwlock_t *rwl, krw_t rw)
602d3f97842c 6393351 unique_* could be improved
ahrens
parents:
diff changeset
   107
{
602d3f97842c 6393351 unique_* could be improved
ahrens
parents:
diff changeset
   108
	boolean_t held;
602d3f97842c 6393351 unique_* could be improved
ahrens
parents:
diff changeset
   109
602d3f97842c 6393351 unique_* could be improved
ahrens
parents:
diff changeset
   110
	mutex_enter(&rwl->rw_lock);
602d3f97842c 6393351 unique_* could be improved
ahrens
parents:
diff changeset
   111
	if (rw == RW_WRITER)
602d3f97842c 6393351 unique_* could be improved
ahrens
parents:
diff changeset
   112
		held = (rwl->rw_writer == curthread);
602d3f97842c 6393351 unique_* could be improved
ahrens
parents:
diff changeset
   113
	else
602d3f97842c 6393351 unique_* could be improved
ahrens
parents:
diff changeset
   114
		held = !rwl->rw_writer && !refcount_is_zero(&rwl->rw_count);
602d3f97842c 6393351 unique_* could be improved
ahrens
parents:
diff changeset
   115
	mutex_exit(&rwl->rw_lock);
602d3f97842c 6393351 unique_* could be improved
ahrens
parents:
diff changeset
   116
602d3f97842c 6393351 unique_* could be improved
ahrens
parents:
diff changeset
   117
	return (held);
602d3f97842c 6393351 unique_* could be improved
ahrens
parents:
diff changeset
   118
}