8023 Panic destroying a metaslab deferred range tree
[unleashed.git] / usr / src / uts / common / fs / zfs / bqueue.c
blob1ddc697b5424a67b51886142764f1f03ea58ae1d
1 /*
2 * CDDL HEADER START
4 * This file and its contents are supplied under the terms of the
5 * Common Development and Distribution License ("CDDL"), version 1.0.
6 * You may only use this file in accordance with the terms of version
7 * 1.0 of the CDDL.
9 * A full copy of the text of the CDDL should have accompanied this
10 * source. A copy of the CDDL is also available via the Internet at
11 * http://www.illumos.org/license/CDDL.
13 * CDDL HEADER END
16 * Copyright (c) 2014 by Delphix. All rights reserved.
19 #include <sys/bqueue.h>
20 #include <sys/zfs_context.h>
22 static inline bqueue_node_t *
23 obj2node(bqueue_t *q, void *data)
25 return ((bqueue_node_t *)((char *)data + q->bq_node_offset));
29 * Initialize a blocking queue The maximum capacity of the queue is set to
30 * size. Types that want to be stored in a bqueue must contain a bqueue_node_t,
31 * and offset should give its offset from the start of the struct. Return 0 on
32 * success, or -1 on failure.
34 int
35 bqueue_init(bqueue_t *q, uint64_t size, size_t node_offset)
37 list_create(&q->bq_list, node_offset + sizeof (bqueue_node_t),
38 node_offset + offsetof(bqueue_node_t, bqn_node));
39 cv_init(&q->bq_add_cv, NULL, CV_DEFAULT, NULL);
40 cv_init(&q->bq_pop_cv, NULL, CV_DEFAULT, NULL);
41 mutex_init(&q->bq_lock, NULL, MUTEX_DEFAULT, NULL);
42 q->bq_node_offset = node_offset;
43 q->bq_size = 0;
44 q->bq_maxsize = size;
45 return (0);
49 * Destroy a blocking queue. This function asserts that there are no
50 * elements in the queue, and no one is blocked on the condition
51 * variables.
53 void
54 bqueue_destroy(bqueue_t *q)
56 ASSERT0(q->bq_size);
57 cv_destroy(&q->bq_add_cv);
58 cv_destroy(&q->bq_pop_cv);
59 mutex_destroy(&q->bq_lock);
60 list_destroy(&q->bq_list);
64 * Add data to q, consuming size units of capacity. If there is insufficient
65 * capacity to consume size units, block until capacity exists. Asserts size is
66 * > 0.
68 void
69 bqueue_enqueue(bqueue_t *q, void *data, uint64_t item_size)
71 ASSERT3U(item_size, >, 0);
72 ASSERT3U(item_size, <, q->bq_maxsize);
73 mutex_enter(&q->bq_lock);
74 obj2node(q, data)->bqn_size = item_size;
75 while (q->bq_size + item_size > q->bq_maxsize) {
76 cv_wait(&q->bq_add_cv, &q->bq_lock);
78 q->bq_size += item_size;
79 list_insert_tail(&q->bq_list, data);
80 cv_signal(&q->bq_pop_cv);
81 mutex_exit(&q->bq_lock);
84 * Take the first element off of q. If there are no elements on the queue, wait
85 * until one is put there. Return the removed element.
87 void *
88 bqueue_dequeue(bqueue_t *q)
90 void *ret;
91 uint64_t item_size;
92 mutex_enter(&q->bq_lock);
93 while (q->bq_size == 0) {
94 cv_wait(&q->bq_pop_cv, &q->bq_lock);
96 ret = list_remove_head(&q->bq_list);
97 item_size = obj2node(q, ret)->bqn_size;
98 q->bq_size -= item_size;
99 mutex_exit(&q->bq_lock);
100 cv_signal(&q->bq_add_cv);
101 return (ret);
105 * Returns true if the space used is 0.
107 boolean_t
108 bqueue_empty(bqueue_t *q)
110 return (q->bq_size == 0);