1dd84a43cSPoul-Henning Kamp /*-
23728855aSPedro F. Giffuni * SPDX-License-Identifier: BSD-3-Clause
33728855aSPedro F. Giffuni *
4dd84a43cSPoul-Henning Kamp * Copyright (c) 2002 Poul-Henning Kamp
5dd84a43cSPoul-Henning Kamp * Copyright (c) 2002 Networks Associates Technology, Inc.
6ee75e7deSKonstantin Belousov * Copyright (c) 2013 The FreeBSD Foundation
7dd84a43cSPoul-Henning Kamp * All rights reserved.
8dd84a43cSPoul-Henning Kamp *
9dd84a43cSPoul-Henning Kamp * This software was developed for the FreeBSD Project by Poul-Henning Kamp
10dd84a43cSPoul-Henning Kamp * and NAI Labs, the Security Research Division of Network Associates, Inc.
11dd84a43cSPoul-Henning Kamp * under DARPA/SPAWAR contract N66001-01-C-8035 ("CBOSS"), as part of the
12dd84a43cSPoul-Henning Kamp * DARPA CHATS research program.
13dd84a43cSPoul-Henning Kamp *
14ee75e7deSKonstantin Belousov * Portions of this software were developed by Konstantin Belousov
15ee75e7deSKonstantin Belousov * under sponsorship from the FreeBSD Foundation.
16ee75e7deSKonstantin Belousov *
17dd84a43cSPoul-Henning Kamp * Redistribution and use in source and binary forms, with or without
18dd84a43cSPoul-Henning Kamp * modification, are permitted provided that the following conditions
19dd84a43cSPoul-Henning Kamp * are met:
20dd84a43cSPoul-Henning Kamp * 1. Redistributions of source code must retain the above copyright
21dd84a43cSPoul-Henning Kamp * notice, this list of conditions and the following disclaimer.
22dd84a43cSPoul-Henning Kamp * 2. Redistributions in binary form must reproduce the above copyright
23dd84a43cSPoul-Henning Kamp * notice, this list of conditions and the following disclaimer in the
24dd84a43cSPoul-Henning Kamp * documentation and/or other materials provided with the distribution.
25dd84a43cSPoul-Henning Kamp * 3. The names of the authors may not be used to endorse or promote
26dd84a43cSPoul-Henning Kamp * products derived from this software without specific prior written
27dd84a43cSPoul-Henning Kamp * permission.
28dd84a43cSPoul-Henning Kamp *
29dd84a43cSPoul-Henning Kamp * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
30dd84a43cSPoul-Henning Kamp * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
31dd84a43cSPoul-Henning Kamp * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
32dd84a43cSPoul-Henning Kamp * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
33dd84a43cSPoul-Henning Kamp * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
34dd84a43cSPoul-Henning Kamp * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
35dd84a43cSPoul-Henning Kamp * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
36dd84a43cSPoul-Henning Kamp * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
37dd84a43cSPoul-Henning Kamp * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
38dd84a43cSPoul-Henning Kamp * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
39dd84a43cSPoul-Henning Kamp * SUCH DAMAGE.
40dd84a43cSPoul-Henning Kamp */
41dd84a43cSPoul-Henning Kamp
42dd84a43cSPoul-Henning Kamp #include <sys/param.h>
43dd84a43cSPoul-Henning Kamp #include <sys/systm.h>
44dd84a43cSPoul-Henning Kamp #include <sys/kernel.h>
45dd84a43cSPoul-Henning Kamp #include <sys/malloc.h>
46dd84a43cSPoul-Henning Kamp #include <sys/bio.h>
4749dbb61dSRobert Watson #include <sys/ktr.h>
4851460da8SJohn Baldwin #include <sys/proc.h>
49ac03832eSConrad Meyer #include <sys/sbuf.h>
503b378147SPawel Jakub Dawidek #include <sys/stack.h>
51ee75e7deSKonstantin Belousov #include <sys/sysctl.h>
525f518366SJeff Roberson #include <sys/vmem.h>
532555f175SKonstantin Belousov #include <machine/stack.h>
54ac03832eSConrad Meyer #include <machine/stdarg.h>
55dd84a43cSPoul-Henning Kamp
56dd84a43cSPoul-Henning Kamp #include <sys/errno.h>
57dd84a43cSPoul-Henning Kamp #include <geom/geom.h>
58b1876192SPoul-Henning Kamp #include <geom/geom_int.h>
59e24cbd90SPoul-Henning Kamp #include <sys/devicestat.h>
60dd84a43cSPoul-Henning Kamp
615ffb2c8bSPoul-Henning Kamp #include <vm/uma.h>
62ee75e7deSKonstantin Belousov #include <vm/vm.h>
63ee75e7deSKonstantin Belousov #include <vm/vm_param.h>
64ee75e7deSKonstantin Belousov #include <vm/vm_kern.h>
65ee75e7deSKonstantin Belousov #include <vm/vm_page.h>
66ee75e7deSKonstantin Belousov #include <vm/vm_object.h>
67ee75e7deSKonstantin Belousov #include <vm/vm_extern.h>
68ee75e7deSKonstantin Belousov #include <vm/vm_map.h>
695ffb2c8bSPoul-Henning Kamp
70479d224eSDimitry Andric #define KTR_GEOM_ENABLED \
71479d224eSDimitry Andric ((KTR_COMPILE & KTR_GEOM) != 0 && (ktr_mask & KTR_GEOM) != 0)
72479d224eSDimitry Andric
7340ea77a0SAlexander Motin static int g_io_transient_map_bio(struct bio *bp);
7440ea77a0SAlexander Motin
75dd84a43cSPoul-Henning Kamp static struct g_bioq g_bio_run_down;
76dd84a43cSPoul-Henning Kamp static struct g_bioq g_bio_run_up;
77dd84a43cSPoul-Henning Kamp
7832f40fc9SWarner Losh static u_long nomem_count;
7932f40fc9SWarner Losh static u_long pause_count;
8032f40fc9SWarner Losh
813f2e5b85SWarner Losh /*
823f2e5b85SWarner Losh * Pace is a hint that we've had some trouble recently allocating
833f2e5b85SWarner Losh * bios, so we should back off trying to send I/O down the stack
843f2e5b85SWarner Losh * a bit to let the problem resolve. When pacing, we also turn
853f2e5b85SWarner Losh * off direct dispatch to also reduce memory pressure from I/Os
863f2e5b85SWarner Losh * there, at the expxense of some added latency while the memory
873f2e5b85SWarner Losh * pressures exist. See g_io_schedule_down() for more details
883f2e5b85SWarner Losh * and limitations.
893f2e5b85SWarner Losh */
9061322a0aSAlexander Motin static volatile u_int __read_mostly pace;
913f2e5b85SWarner Losh
9261322a0aSAlexander Motin static uma_zone_t __read_mostly biozone;
933432e4fdSPoul-Henning Kamp
94dd84a43cSPoul-Henning Kamp #include <machine/atomic.h>
95dd84a43cSPoul-Henning Kamp
96dd84a43cSPoul-Henning Kamp static void
g_bioq_lock(struct g_bioq * bq)97dd84a43cSPoul-Henning Kamp g_bioq_lock(struct g_bioq *bq)
98dd84a43cSPoul-Henning Kamp {
99dd84a43cSPoul-Henning Kamp
100dd84a43cSPoul-Henning Kamp mtx_lock(&bq->bio_queue_lock);
101dd84a43cSPoul-Henning Kamp }
102dd84a43cSPoul-Henning Kamp
103dd84a43cSPoul-Henning Kamp static void
g_bioq_unlock(struct g_bioq * bq)104dd84a43cSPoul-Henning Kamp g_bioq_unlock(struct g_bioq *bq)
105dd84a43cSPoul-Henning Kamp {
106dd84a43cSPoul-Henning Kamp
107dd84a43cSPoul-Henning Kamp mtx_unlock(&bq->bio_queue_lock);
108dd84a43cSPoul-Henning Kamp }
109dd84a43cSPoul-Henning Kamp
110dd84a43cSPoul-Henning Kamp #if 0
111dd84a43cSPoul-Henning Kamp static void
112dd84a43cSPoul-Henning Kamp g_bioq_destroy(struct g_bioq *bq)
113dd84a43cSPoul-Henning Kamp {
114dd84a43cSPoul-Henning Kamp
115dd84a43cSPoul-Henning Kamp mtx_destroy(&bq->bio_queue_lock);
116dd84a43cSPoul-Henning Kamp }
117dd84a43cSPoul-Henning Kamp #endif
118dd84a43cSPoul-Henning Kamp
119dd84a43cSPoul-Henning Kamp static void
g_bioq_init(struct g_bioq * bq)120dd84a43cSPoul-Henning Kamp g_bioq_init(struct g_bioq *bq)
121dd84a43cSPoul-Henning Kamp {
122dd84a43cSPoul-Henning Kamp
123dd84a43cSPoul-Henning Kamp TAILQ_INIT(&bq->bio_queue);
1246008862bSJohn Baldwin mtx_init(&bq->bio_queue_lock, "bio queue", NULL, MTX_DEF);
125dd84a43cSPoul-Henning Kamp }
126dd84a43cSPoul-Henning Kamp
127dd84a43cSPoul-Henning Kamp static struct bio *
g_bioq_first(struct g_bioq * bq)128dd84a43cSPoul-Henning Kamp g_bioq_first(struct g_bioq *bq)
129dd84a43cSPoul-Henning Kamp {
130dd84a43cSPoul-Henning Kamp struct bio *bp;
131dd84a43cSPoul-Henning Kamp
132dd84a43cSPoul-Henning Kamp bp = TAILQ_FIRST(&bq->bio_queue);
133dd84a43cSPoul-Henning Kamp if (bp != NULL) {
134dcbd0fe5SPoul-Henning Kamp KASSERT((bp->bio_flags & BIO_ONQUEUE),
135dcbd0fe5SPoul-Henning Kamp ("Bio not on queue bp=%p target %p", bp, bq));
136dcbd0fe5SPoul-Henning Kamp bp->bio_flags &= ~BIO_ONQUEUE;
137dd84a43cSPoul-Henning Kamp TAILQ_REMOVE(&bq->bio_queue, bp, bio_queue);
138dd84a43cSPoul-Henning Kamp bq->bio_queue_length--;
139dd84a43cSPoul-Henning Kamp }
140dd84a43cSPoul-Henning Kamp return (bp);
141dd84a43cSPoul-Henning Kamp }
142dd84a43cSPoul-Henning Kamp
143dd84a43cSPoul-Henning Kamp struct bio *
g_new_bio(void)144dd84a43cSPoul-Henning Kamp g_new_bio(void)
145dd84a43cSPoul-Henning Kamp {
146dd84a43cSPoul-Henning Kamp struct bio *bp;
147dd84a43cSPoul-Henning Kamp
1485ffb2c8bSPoul-Henning Kamp bp = uma_zalloc(biozone, M_NOWAIT | M_ZERO);
1493b378147SPawel Jakub Dawidek #ifdef KTR
150479d224eSDimitry Andric if (KTR_GEOM_ENABLED) {
1513b378147SPawel Jakub Dawidek struct stack st;
1523b378147SPawel Jakub Dawidek
1533b378147SPawel Jakub Dawidek CTR1(KTR_GEOM, "g_new_bio(): %p", bp);
1543b378147SPawel Jakub Dawidek stack_save(&st);
15554533f66SConrad Meyer CTRSTACK(KTR_GEOM, &st, 3);
1563b378147SPawel Jakub Dawidek }
1573b378147SPawel Jakub Dawidek #endif
158dd84a43cSPoul-Henning Kamp return (bp);
159dd84a43cSPoul-Henning Kamp }
160dd84a43cSPoul-Henning Kamp
161a2033c96SPoul-Henning Kamp struct bio *
g_alloc_bio(void)162a2033c96SPoul-Henning Kamp g_alloc_bio(void)
163a2033c96SPoul-Henning Kamp {
164a2033c96SPoul-Henning Kamp struct bio *bp;
165a2033c96SPoul-Henning Kamp
166a2033c96SPoul-Henning Kamp bp = uma_zalloc(biozone, M_WAITOK | M_ZERO);
1673b378147SPawel Jakub Dawidek #ifdef KTR
168479d224eSDimitry Andric if (KTR_GEOM_ENABLED) {
1693b378147SPawel Jakub Dawidek struct stack st;
1703b378147SPawel Jakub Dawidek
1713b378147SPawel Jakub Dawidek CTR1(KTR_GEOM, "g_alloc_bio(): %p", bp);
1723b378147SPawel Jakub Dawidek stack_save(&st);
17354533f66SConrad Meyer CTRSTACK(KTR_GEOM, &st, 3);
1743b378147SPawel Jakub Dawidek }
1753b378147SPawel Jakub Dawidek #endif
176a2033c96SPoul-Henning Kamp return (bp);
177a2033c96SPoul-Henning Kamp }
178a2033c96SPoul-Henning Kamp
179dd84a43cSPoul-Henning Kamp void
g_destroy_bio(struct bio * bp)180dd84a43cSPoul-Henning Kamp g_destroy_bio(struct bio *bp)
181dd84a43cSPoul-Henning Kamp {
1823b378147SPawel Jakub Dawidek #ifdef KTR
183479d224eSDimitry Andric if (KTR_GEOM_ENABLED) {
1843b378147SPawel Jakub Dawidek struct stack st;
185dd84a43cSPoul-Henning Kamp
1863b378147SPawel Jakub Dawidek CTR1(KTR_GEOM, "g_destroy_bio(): %p", bp);
1873b378147SPawel Jakub Dawidek stack_save(&st);
18854533f66SConrad Meyer CTRSTACK(KTR_GEOM, &st, 3);
1893b378147SPawel Jakub Dawidek }
1903b378147SPawel Jakub Dawidek #endif
1915ffb2c8bSPoul-Henning Kamp uma_zfree(biozone, bp);
192dd84a43cSPoul-Henning Kamp }
193dd84a43cSPoul-Henning Kamp
194dd84a43cSPoul-Henning Kamp struct bio *
g_clone_bio(struct bio * bp)195dd84a43cSPoul-Henning Kamp g_clone_bio(struct bio *bp)
196dd84a43cSPoul-Henning Kamp {
197dd84a43cSPoul-Henning Kamp struct bio *bp2;
198dd84a43cSPoul-Henning Kamp
1995ffb2c8bSPoul-Henning Kamp bp2 = uma_zalloc(biozone, M_NOWAIT | M_ZERO);
200a1bd3ee2SPoul-Henning Kamp if (bp2 != NULL) {
201936cc461SPoul-Henning Kamp bp2->bio_parent = bp;
202dd84a43cSPoul-Henning Kamp bp2->bio_cmd = bp->bio_cmd;
20382a6ae10SJim Harris /*
20482a6ae10SJim Harris * BIO_ORDERED flag may be used by disk drivers to enforce
20582a6ae10SJim Harris * ordering restrictions, so this flag needs to be cloned.
206c6213befSGleb Smirnoff * BIO_UNMAPPED, BIO_VLIST, and BIO_SWAP should be inherited,
207c6213befSGleb Smirnoff * to properly indicate which way the buffer is passed.
20882a6ae10SJim Harris * Other bio flags are not suitable for cloning.
20982a6ae10SJim Harris */
210a9934668SKenneth D. Merry bp2->bio_flags = bp->bio_flags &
211c6213befSGleb Smirnoff (BIO_ORDERED | BIO_UNMAPPED | BIO_VLIST | BIO_SWAP);
212dd84a43cSPoul-Henning Kamp bp2->bio_length = bp->bio_length;
213dd84a43cSPoul-Henning Kamp bp2->bio_offset = bp->bio_offset;
214dd84a43cSPoul-Henning Kamp bp2->bio_data = bp->bio_data;
215ee75e7deSKonstantin Belousov bp2->bio_ma = bp->bio_ma;
216ee75e7deSKonstantin Belousov bp2->bio_ma_n = bp->bio_ma_n;
217ee75e7deSKonstantin Belousov bp2->bio_ma_offset = bp->bio_ma_offset;
218dd84a43cSPoul-Henning Kamp bp2->bio_attribute = bp->bio_attribute;
2199a6844d5SKenneth D. Merry if (bp->bio_cmd == BIO_ZONE)
2209a6844d5SKenneth D. Merry bcopy(&bp->bio_zone, &bp2->bio_zone,
2219a6844d5SKenneth D. Merry sizeof(bp->bio_zone));
2228532d381SConrad Meyer #if defined(BUF_TRACKING) || defined(FULL_BUF_TRACKING)
2238532d381SConrad Meyer bp2->bio_track_bp = bp->bio_track_bp;
2248532d381SConrad Meyer #endif
225801bb689SPoul-Henning Kamp bp->bio_children++;
226a1bd3ee2SPoul-Henning Kamp }
2273b378147SPawel Jakub Dawidek #ifdef KTR
228479d224eSDimitry Andric if (KTR_GEOM_ENABLED) {
2293b378147SPawel Jakub Dawidek struct stack st;
2303b378147SPawel Jakub Dawidek
231ad572235SRuslan Ermilov CTR2(KTR_GEOM, "g_clone_bio(%p): %p", bp, bp2);
2323b378147SPawel Jakub Dawidek stack_save(&st);
23354533f66SConrad Meyer CTRSTACK(KTR_GEOM, &st, 3);
2343b378147SPawel Jakub Dawidek }
2353b378147SPawel Jakub Dawidek #endif
236dd84a43cSPoul-Henning Kamp return(bp2);
237dd84a43cSPoul-Henning Kamp }
238dd84a43cSPoul-Henning Kamp
2394bec0ff1SPawel Jakub Dawidek struct bio *
g_duplicate_bio(struct bio * bp)2404bec0ff1SPawel Jakub Dawidek g_duplicate_bio(struct bio *bp)
2414bec0ff1SPawel Jakub Dawidek {
2424bec0ff1SPawel Jakub Dawidek struct bio *bp2;
2434bec0ff1SPawel Jakub Dawidek
2444bec0ff1SPawel Jakub Dawidek bp2 = uma_zalloc(biozone, M_WAITOK | M_ZERO);
245c6213befSGleb Smirnoff bp2->bio_flags = bp->bio_flags & (BIO_UNMAPPED | BIO_VLIST | BIO_SWAP);
2464bec0ff1SPawel Jakub Dawidek bp2->bio_parent = bp;
2474bec0ff1SPawel Jakub Dawidek bp2->bio_cmd = bp->bio_cmd;
2484bec0ff1SPawel Jakub Dawidek bp2->bio_length = bp->bio_length;
2494bec0ff1SPawel Jakub Dawidek bp2->bio_offset = bp->bio_offset;
2504bec0ff1SPawel Jakub Dawidek bp2->bio_data = bp->bio_data;
251ee75e7deSKonstantin Belousov bp2->bio_ma = bp->bio_ma;
252ee75e7deSKonstantin Belousov bp2->bio_ma_n = bp->bio_ma_n;
253ee75e7deSKonstantin Belousov bp2->bio_ma_offset = bp->bio_ma_offset;
2544bec0ff1SPawel Jakub Dawidek bp2->bio_attribute = bp->bio_attribute;
2554bec0ff1SPawel Jakub Dawidek bp->bio_children++;
2564bec0ff1SPawel Jakub Dawidek #ifdef KTR
257479d224eSDimitry Andric if (KTR_GEOM_ENABLED) {
2584bec0ff1SPawel Jakub Dawidek struct stack st;
2594bec0ff1SPawel Jakub Dawidek
2604bec0ff1SPawel Jakub Dawidek CTR2(KTR_GEOM, "g_duplicate_bio(%p): %p", bp, bp2);
2614bec0ff1SPawel Jakub Dawidek stack_save(&st);
26254533f66SConrad Meyer CTRSTACK(KTR_GEOM, &st, 3);
2634bec0ff1SPawel Jakub Dawidek }
2644bec0ff1SPawel Jakub Dawidek #endif
2654bec0ff1SPawel Jakub Dawidek return(bp2);
2664bec0ff1SPawel Jakub Dawidek }
2674bec0ff1SPawel Jakub Dawidek
268dd84a43cSPoul-Henning Kamp void
g_reset_bio(struct bio * bp)269c55f5707SWarner Losh g_reset_bio(struct bio *bp)
270c55f5707SWarner Losh {
271c55f5707SWarner Losh
272bd4c1dd6SWarner Losh bzero(bp, sizeof(*bp));
273c55f5707SWarner Losh }
274c55f5707SWarner Losh
275c55f5707SWarner Losh void
g_io_init(void)276165a3212SDimitry Andric g_io_init(void)
277dd84a43cSPoul-Henning Kamp {
278dd84a43cSPoul-Henning Kamp
279dd84a43cSPoul-Henning Kamp g_bioq_init(&g_bio_run_down);
280dd84a43cSPoul-Henning Kamp g_bioq_init(&g_bio_run_up);
2815ffb2c8bSPoul-Henning Kamp biozone = uma_zcreate("g_bio", sizeof (struct bio),
2825ffb2c8bSPoul-Henning Kamp NULL, NULL,
2835ffb2c8bSPoul-Henning Kamp NULL, NULL,
2845ffb2c8bSPoul-Henning Kamp 0, 0);
285dd84a43cSPoul-Henning Kamp }
286dd84a43cSPoul-Henning Kamp
287dd84a43cSPoul-Henning Kamp int
g_io_getattr(const char * attr,struct g_consumer * cp,int * len,void * ptr)2880d3f37a8SPoul-Henning Kamp g_io_getattr(const char *attr, struct g_consumer *cp, int *len, void *ptr)
289dd84a43cSPoul-Henning Kamp {
290dd84a43cSPoul-Henning Kamp struct bio *bp;
291dd84a43cSPoul-Henning Kamp int error;
292dd84a43cSPoul-Henning Kamp
293dd84a43cSPoul-Henning Kamp g_trace(G_T_BIO, "bio_getattr(%s)", attr);
294a2033c96SPoul-Henning Kamp bp = g_alloc_bio();
295dd84a43cSPoul-Henning Kamp bp->bio_cmd = BIO_GETATTR;
296dd84a43cSPoul-Henning Kamp bp->bio_done = NULL;
297dd84a43cSPoul-Henning Kamp bp->bio_attribute = attr;
298dd84a43cSPoul-Henning Kamp bp->bio_length = *len;
299dd84a43cSPoul-Henning Kamp bp->bio_data = ptr;
300dd84a43cSPoul-Henning Kamp g_io_request(bp, cp);
30153706245SPoul-Henning Kamp error = biowait(bp, "ggetattr");
302dd84a43cSPoul-Henning Kamp *len = bp->bio_completed;
303dd84a43cSPoul-Henning Kamp g_destroy_bio(bp);
304dd84a43cSPoul-Henning Kamp return (error);
305dd84a43cSPoul-Henning Kamp }
306dd84a43cSPoul-Henning Kamp
307c3618c65SPawel Jakub Dawidek int
g_io_zonecmd(struct disk_zone_args * zone_args,struct g_consumer * cp)3089a6844d5SKenneth D. Merry g_io_zonecmd(struct disk_zone_args *zone_args, struct g_consumer *cp)
3099a6844d5SKenneth D. Merry {
3109a6844d5SKenneth D. Merry struct bio *bp;
3119a6844d5SKenneth D. Merry int error;
3129a6844d5SKenneth D. Merry
3139a6844d5SKenneth D. Merry g_trace(G_T_BIO, "bio_zone(%d)", zone_args->zone_cmd);
3149a6844d5SKenneth D. Merry bp = g_alloc_bio();
3159a6844d5SKenneth D. Merry bp->bio_cmd = BIO_ZONE;
3169a6844d5SKenneth D. Merry bp->bio_done = NULL;
3179a6844d5SKenneth D. Merry /*
3189a6844d5SKenneth D. Merry * XXX KDM need to handle report zone data.
3199a6844d5SKenneth D. Merry */
3209a6844d5SKenneth D. Merry bcopy(zone_args, &bp->bio_zone, sizeof(*zone_args));
3219a6844d5SKenneth D. Merry if (zone_args->zone_cmd == DISK_ZONE_REPORT_ZONES)
3229a6844d5SKenneth D. Merry bp->bio_length =
3239a6844d5SKenneth D. Merry zone_args->zone_params.report.entries_allocated *
3249a6844d5SKenneth D. Merry sizeof(struct disk_zone_rep_entry);
3259a6844d5SKenneth D. Merry else
3269a6844d5SKenneth D. Merry bp->bio_length = 0;
3279a6844d5SKenneth D. Merry
3289a6844d5SKenneth D. Merry g_io_request(bp, cp);
3299a6844d5SKenneth D. Merry error = biowait(bp, "gzone");
3309a6844d5SKenneth D. Merry bcopy(&bp->bio_zone, zone_args, sizeof(*zone_args));
3319a6844d5SKenneth D. Merry g_destroy_bio(bp);
3329a6844d5SKenneth D. Merry return (error);
3339a6844d5SKenneth D. Merry }
3349a6844d5SKenneth D. Merry
335b182c792SWarner Losh /*
336b182c792SWarner Losh * Send a BIO_SPEEDUP down the stack. This is used to tell the lower layers that
337b182c792SWarner Losh * the upper layers have detected a resource shortage. The lower layers are
338b182c792SWarner Losh * advised to stop delaying I/O that they might be holding for performance
339b182c792SWarner Losh * reasons and to schedule it (non-trims) or complete it successfully (trims) as
340b182c792SWarner Losh * quickly as it can. bio_length is the amount of the shortage. This call
341b182c792SWarner Losh * should be non-blocking. bio_resid is used to communicate back if the lower
342b182c792SWarner Losh * layers couldn't find bio_length worth of I/O to schedule or discard. A length
343b182c792SWarner Losh * of 0 means to do as much as you can (schedule the h/w queues full, discard
344b182c792SWarner Losh * all trims). flags are a hint from the upper layers to the lower layers what
345b182c792SWarner Losh * operation should be done.
346b182c792SWarner Losh */
347b182c792SWarner Losh int
g_io_speedup(off_t shortage,u_int flags,size_t * resid,struct g_consumer * cp)348f4499487SMark Johnston g_io_speedup(off_t shortage, u_int flags, size_t *resid, struct g_consumer *cp)
349b182c792SWarner Losh {
350b182c792SWarner Losh struct bio *bp;
351b182c792SWarner Losh int error;
352b182c792SWarner Losh
353b182c792SWarner Losh KASSERT((flags & (BIO_SPEEDUP_TRIM | BIO_SPEEDUP_WRITE)) != 0,
354b182c792SWarner Losh ("Invalid flags passed to g_io_speedup: %#x", flags));
355f4499487SMark Johnston g_trace(G_T_BIO, "bio_speedup(%s, %jd, %#x)", cp->provider->name,
356f4499487SMark Johnston (intmax_t)shortage, flags);
357b182c792SWarner Losh bp = g_new_bio();
358b182c792SWarner Losh if (bp == NULL)
359b182c792SWarner Losh return (ENOMEM);
360b182c792SWarner Losh bp->bio_cmd = BIO_SPEEDUP;
361b182c792SWarner Losh bp->bio_length = shortage;
362b182c792SWarner Losh bp->bio_done = NULL;
363b182c792SWarner Losh bp->bio_flags |= flags;
364b182c792SWarner Losh g_io_request(bp, cp);
365b182c792SWarner Losh error = biowait(bp, "gflush");
366b182c792SWarner Losh *resid = bp->bio_resid;
367b182c792SWarner Losh g_destroy_bio(bp);
368b182c792SWarner Losh return (error);
369b182c792SWarner Losh }
370b182c792SWarner Losh
3719a6844d5SKenneth D. Merry int
g_io_flush(struct g_consumer * cp)372c3618c65SPawel Jakub Dawidek g_io_flush(struct g_consumer *cp)
373c3618c65SPawel Jakub Dawidek {
374c3618c65SPawel Jakub Dawidek struct bio *bp;
375c3618c65SPawel Jakub Dawidek int error;
376c3618c65SPawel Jakub Dawidek
377c3618c65SPawel Jakub Dawidek g_trace(G_T_BIO, "bio_flush(%s)", cp->provider->name);
378c3618c65SPawel Jakub Dawidek bp = g_alloc_bio();
379c3618c65SPawel Jakub Dawidek bp->bio_cmd = BIO_FLUSH;
380f03f7a0cSJustin T. Gibbs bp->bio_flags |= BIO_ORDERED;
381c3618c65SPawel Jakub Dawidek bp->bio_done = NULL;
382c3618c65SPawel Jakub Dawidek bp->bio_attribute = NULL;
383c3618c65SPawel Jakub Dawidek bp->bio_offset = cp->provider->mediasize;
384c3618c65SPawel Jakub Dawidek bp->bio_length = 0;
385c3618c65SPawel Jakub Dawidek bp->bio_data = NULL;
386c3618c65SPawel Jakub Dawidek g_io_request(bp, cp);
387c3618c65SPawel Jakub Dawidek error = biowait(bp, "gflush");
388c3618c65SPawel Jakub Dawidek g_destroy_bio(bp);
389c3618c65SPawel Jakub Dawidek return (error);
390c3618c65SPawel Jakub Dawidek }
391c3618c65SPawel Jakub Dawidek
392e39d70d4SPoul-Henning Kamp static int
g_io_check(struct bio * bp)393e39d70d4SPoul-Henning Kamp g_io_check(struct bio *bp)
394e39d70d4SPoul-Henning Kamp {
395e39d70d4SPoul-Henning Kamp struct g_consumer *cp;
396e39d70d4SPoul-Henning Kamp struct g_provider *pp;
39740ea77a0SAlexander Motin off_t excess;
39840ea77a0SAlexander Motin int error;
399e39d70d4SPoul-Henning Kamp
4008532d381SConrad Meyer biotrack(bp, __func__);
4018532d381SConrad Meyer
402e39d70d4SPoul-Henning Kamp cp = bp->bio_from;
403e39d70d4SPoul-Henning Kamp pp = bp->bio_to;
404e39d70d4SPoul-Henning Kamp
405e39d70d4SPoul-Henning Kamp /* Fail if access counters dont allow the operation */
406e39d70d4SPoul-Henning Kamp switch(bp->bio_cmd) {
407e39d70d4SPoul-Henning Kamp case BIO_READ:
408e39d70d4SPoul-Henning Kamp case BIO_GETATTR:
409e39d70d4SPoul-Henning Kamp if (cp->acr == 0)
410e39d70d4SPoul-Henning Kamp return (EPERM);
411e39d70d4SPoul-Henning Kamp break;
412e39d70d4SPoul-Henning Kamp case BIO_WRITE:
413e39d70d4SPoul-Henning Kamp case BIO_DELETE:
4148b522bdaSWarner Losh case BIO_SPEEDUP:
415c3618c65SPawel Jakub Dawidek case BIO_FLUSH:
416e39d70d4SPoul-Henning Kamp if (cp->acw == 0)
417e39d70d4SPoul-Henning Kamp return (EPERM);
418e39d70d4SPoul-Henning Kamp break;
4199a6844d5SKenneth D. Merry case BIO_ZONE:
4209a6844d5SKenneth D. Merry if ((bp->bio_zone.zone_cmd == DISK_ZONE_REPORT_ZONES) ||
4219a6844d5SKenneth D. Merry (bp->bio_zone.zone_cmd == DISK_ZONE_GET_PARAMS)) {
4229a6844d5SKenneth D. Merry if (cp->acr == 0)
4239a6844d5SKenneth D. Merry return (EPERM);
4249a6844d5SKenneth D. Merry } else if (cp->acw == 0)
4259a6844d5SKenneth D. Merry return (EPERM);
4269a6844d5SKenneth D. Merry break;
427e39d70d4SPoul-Henning Kamp default:
428e39d70d4SPoul-Henning Kamp return (EPERM);
429e39d70d4SPoul-Henning Kamp }
430e39d70d4SPoul-Henning Kamp /* if provider is marked for error, don't disturb. */
431e39d70d4SPoul-Henning Kamp if (pp->error)
432e39d70d4SPoul-Henning Kamp return (pp->error);
4333631c638SAlexander Motin if (cp->flags & G_CF_ORPHAN)
4343631c638SAlexander Motin return (ENXIO);
435e39d70d4SPoul-Henning Kamp
436e39d70d4SPoul-Henning Kamp switch(bp->bio_cmd) {
437e39d70d4SPoul-Henning Kamp case BIO_READ:
438e39d70d4SPoul-Henning Kamp case BIO_WRITE:
439e39d70d4SPoul-Henning Kamp case BIO_DELETE:
4402a842317SAndriy Gapon /* Zero sectorsize or mediasize is probably a lack of media. */
4412a842317SAndriy Gapon if (pp->sectorsize == 0 || pp->mediasize == 0)
44243bff1a7SPoul-Henning Kamp return (ENXIO);
443e39d70d4SPoul-Henning Kamp /* Reject I/O not on sector boundary */
444e39d70d4SPoul-Henning Kamp if (bp->bio_offset % pp->sectorsize)
445e39d70d4SPoul-Henning Kamp return (EINVAL);
446e39d70d4SPoul-Henning Kamp /* Reject I/O not integral sector long */
447e39d70d4SPoul-Henning Kamp if (bp->bio_length % pp->sectorsize)
448e39d70d4SPoul-Henning Kamp return (EINVAL);
449d1b8bf47SPoul-Henning Kamp /* Reject requests before or past the end of media. */
450d1b8bf47SPoul-Henning Kamp if (bp->bio_offset < 0)
451d1b8bf47SPoul-Henning Kamp return (EIO);
452e39d70d4SPoul-Henning Kamp if (bp->bio_offset > pp->mediasize)
453e39d70d4SPoul-Henning Kamp return (EIO);
45440ea77a0SAlexander Motin
45540ea77a0SAlexander Motin /* Truncate requests to the end of providers media. */
45640ea77a0SAlexander Motin excess = bp->bio_offset + bp->bio_length;
45740ea77a0SAlexander Motin if (excess > bp->bio_to->mediasize) {
45840ea77a0SAlexander Motin KASSERT((bp->bio_flags & BIO_UNMAPPED) == 0 ||
45940ea77a0SAlexander Motin round_page(bp->bio_ma_offset +
46040ea77a0SAlexander Motin bp->bio_length) / PAGE_SIZE == bp->bio_ma_n,
46140ea77a0SAlexander Motin ("excess bio %p too short", bp));
46240ea77a0SAlexander Motin excess -= bp->bio_to->mediasize;
46340ea77a0SAlexander Motin bp->bio_length -= excess;
46440ea77a0SAlexander Motin if ((bp->bio_flags & BIO_UNMAPPED) != 0) {
46540ea77a0SAlexander Motin bp->bio_ma_n = round_page(bp->bio_ma_offset +
46640ea77a0SAlexander Motin bp->bio_length) / PAGE_SIZE;
46740ea77a0SAlexander Motin }
46840ea77a0SAlexander Motin if (excess > 0)
46940ea77a0SAlexander Motin CTR3(KTR_GEOM, "g_down truncated bio "
47040ea77a0SAlexander Motin "%p provider %s by %d", bp,
47140ea77a0SAlexander Motin bp->bio_to->name, excess);
47240ea77a0SAlexander Motin }
47340ea77a0SAlexander Motin
47440ea77a0SAlexander Motin /* Deliver zero length transfers right here. */
47540ea77a0SAlexander Motin if (bp->bio_length == 0) {
47640ea77a0SAlexander Motin CTR2(KTR_GEOM, "g_down terminated 0-length "
47740ea77a0SAlexander Motin "bp %p provider %s", bp, bp->bio_to->name);
47840ea77a0SAlexander Motin return (0);
47940ea77a0SAlexander Motin }
48040ea77a0SAlexander Motin
48140ea77a0SAlexander Motin if ((bp->bio_flags & BIO_UNMAPPED) != 0 &&
48240ea77a0SAlexander Motin (bp->bio_to->flags & G_PF_ACCEPT_UNMAPPED) == 0 &&
48340ea77a0SAlexander Motin (bp->bio_cmd == BIO_READ || bp->bio_cmd == BIO_WRITE)) {
48440ea77a0SAlexander Motin if ((error = g_io_transient_map_bio(bp)) >= 0)
48540ea77a0SAlexander Motin return (error);
48640ea77a0SAlexander Motin }
487e39d70d4SPoul-Henning Kamp break;
488e39d70d4SPoul-Henning Kamp default:
489e39d70d4SPoul-Henning Kamp break;
490e39d70d4SPoul-Henning Kamp }
49140ea77a0SAlexander Motin return (EJUSTRETURN);
492e39d70d4SPoul-Henning Kamp }
493e39d70d4SPoul-Henning Kamp
494dd84a43cSPoul-Henning Kamp void
g_io_request(struct bio * bp,struct g_consumer * cp)495dd84a43cSPoul-Henning Kamp g_io_request(struct bio *bp, struct g_consumer *cp)
496dd84a43cSPoul-Henning Kamp {
497801bb689SPoul-Henning Kamp struct g_provider *pp;
49840ea77a0SAlexander Motin int direct, error, first;
4998076d204SWarner Losh uint8_t cmd;
500dd84a43cSPoul-Henning Kamp
5018532d381SConrad Meyer biotrack(bp, __func__);
5028532d381SConrad Meyer
503d0e17c1bSPoul-Henning Kamp KASSERT(cp != NULL, ("NULL cp in g_io_request"));
504d0e17c1bSPoul-Henning Kamp KASSERT(bp != NULL, ("NULL bp in g_io_request"));
505e060b6bdSPoul-Henning Kamp pp = cp->provider;
506801bb689SPoul-Henning Kamp KASSERT(pp != NULL, ("consumer not attached in g_io_request"));
50792ee312dSPawel Jakub Dawidek #ifdef DIAGNOSTIC
50892ee312dSPawel Jakub Dawidek KASSERT(bp->bio_driver1 == NULL,
50992ee312dSPawel Jakub Dawidek ("bio_driver1 used by the consumer (geom %s)", cp->geom->name));
51092ee312dSPawel Jakub Dawidek KASSERT(bp->bio_driver2 == NULL,
51192ee312dSPawel Jakub Dawidek ("bio_driver2 used by the consumer (geom %s)", cp->geom->name));
51292ee312dSPawel Jakub Dawidek KASSERT(bp->bio_pflags == 0,
51392ee312dSPawel Jakub Dawidek ("bio_pflags used by the consumer (geom %s)", cp->geom->name));
51492ee312dSPawel Jakub Dawidek /*
51592ee312dSPawel Jakub Dawidek * Remember consumer's private fields, so we can detect if they were
51692ee312dSPawel Jakub Dawidek * modified by the provider.
51792ee312dSPawel Jakub Dawidek */
51892ee312dSPawel Jakub Dawidek bp->_bio_caller1 = bp->bio_caller1;
51992ee312dSPawel Jakub Dawidek bp->_bio_caller2 = bp->bio_caller2;
52092ee312dSPawel Jakub Dawidek bp->_bio_cflags = bp->bio_cflags;
52192ee312dSPawel Jakub Dawidek #endif
522801bb689SPoul-Henning Kamp
5238076d204SWarner Losh cmd = bp->bio_cmd;
5248076d204SWarner Losh if (cmd == BIO_READ || cmd == BIO_WRITE || cmd == BIO_GETATTR) {
525c3618c65SPawel Jakub Dawidek KASSERT(bp->bio_data != NULL,
5269a8fa125SWarner Losh ("NULL bp->data in g_io_request(cmd=%hu)", bp->bio_cmd));
5271ded77b2SPawel Jakub Dawidek }
52838da0c96SMark Johnston if (cmd == BIO_DELETE || cmd == BIO_FLUSH || cmd == BIO_SPEEDUP) {
5291ded77b2SPawel Jakub Dawidek KASSERT(bp->bio_data == NULL,
5309a8fa125SWarner Losh ("non-NULL bp->data in g_io_request(cmd=%hu)",
5311ded77b2SPawel Jakub Dawidek bp->bio_cmd));
532c3618c65SPawel Jakub Dawidek }
5338076d204SWarner Losh if (cmd == BIO_READ || cmd == BIO_WRITE || cmd == BIO_DELETE) {
534dcbd0fe5SPoul-Henning Kamp KASSERT(bp->bio_offset % cp->provider->sectorsize == 0,
535dcbd0fe5SPoul-Henning Kamp ("wrong offset %jd for sectorsize %u",
536dcbd0fe5SPoul-Henning Kamp bp->bio_offset, cp->provider->sectorsize));
537dcbd0fe5SPoul-Henning Kamp KASSERT(bp->bio_length % cp->provider->sectorsize == 0,
538dcbd0fe5SPoul-Henning Kamp ("wrong length %jd for sectorsize %u",
539dcbd0fe5SPoul-Henning Kamp bp->bio_length, cp->provider->sectorsize));
540dcbd0fe5SPoul-Henning Kamp }
541dcbd0fe5SPoul-Henning Kamp
542f7717523SStephan Uphoff g_trace(G_T_BIO, "bio_request(%p) from %p(%s) to %p(%s) cmd %d",
543f7717523SStephan Uphoff bp, cp, cp->geom->name, pp, pp->name, bp->bio_cmd);
544f7717523SStephan Uphoff
545dd84a43cSPoul-Henning Kamp bp->bio_from = cp;
546801bb689SPoul-Henning Kamp bp->bio_to = pp;
5472fccec19SPoul-Henning Kamp bp->bio_error = 0;
5482fccec19SPoul-Henning Kamp bp->bio_completed = 0;
549dd84a43cSPoul-Henning Kamp
55019fa21aaSPoul-Henning Kamp KASSERT(!(bp->bio_flags & BIO_ONQUEUE),
55119fa21aaSPoul-Henning Kamp ("Bio already on queue bp=%p", bp));
552024932aaSAlexander Motin
55340ea77a0SAlexander Motin if ((g_collectstats & G_STATS_CONSUMERS) != 0 ||
55440ea77a0SAlexander Motin ((g_collectstats & G_STATS_PROVIDERS) != 0 && pp->stat != NULL))
55519fa21aaSPoul-Henning Kamp binuptime(&bp->bio_t0);
556a5be8eb5SAlexander Motin else
557a5be8eb5SAlexander Motin getbinuptime(&bp->bio_t0);
558024932aaSAlexander Motin if (g_collectstats & G_STATS_CONSUMERS)
5598b220f89SAlexander Motin devstat_start_transaction_bio_t0(cp->stat, bp);
560024932aaSAlexander Motin if (g_collectstats & G_STATS_PROVIDERS)
5618b220f89SAlexander Motin devstat_start_transaction_bio_t0(pp->stat, bp);
562024932aaSAlexander Motin #ifdef INVARIANTS
563024932aaSAlexander Motin atomic_add_int(&cp->nstart, 1);
564024932aaSAlexander Motin #endif
5658827c821SPoul-Henning Kamp
566347e9d54SKonstantin Belousov direct = (cp->flags & G_CF_DIRECT_SEND) != 0 &&
567347e9d54SKonstantin Belousov (pp->flags & G_PF_DIRECT_RECEIVE) != 0 &&
568ffc1cc95SAlexander Motin curthread != g_down_td &&
5699b349650SKonstantin Belousov ((pp->flags & G_PF_ACCEPT_UNMAPPED) != 0 ||
5703f2e5b85SWarner Losh (bp->bio_flags & BIO_UNMAPPED) == 0 || THREAD_CAN_SLEEP()) &&
5713f2e5b85SWarner Losh pace == 0;
57240ea77a0SAlexander Motin if (direct) {
57340ea77a0SAlexander Motin /* Block direct execution if less then half of stack left. */
57440ea77a0SAlexander Motin size_t st, su;
57540ea77a0SAlexander Motin GET_STACK_USAGE(st, su);
57640ea77a0SAlexander Motin if (su * 2 > st)
57740ea77a0SAlexander Motin direct = 0;
57840ea77a0SAlexander Motin }
57940ea77a0SAlexander Motin
58040ea77a0SAlexander Motin if (direct) {
58140ea77a0SAlexander Motin error = g_io_check(bp);
58240ea77a0SAlexander Motin if (error >= 0) {
58340ea77a0SAlexander Motin CTR3(KTR_GEOM, "g_io_request g_io_check on bp %p "
58440ea77a0SAlexander Motin "provider %s returned %d", bp, bp->bio_to->name,
58540ea77a0SAlexander Motin error);
58640ea77a0SAlexander Motin g_io_deliver(bp, error);
58740ea77a0SAlexander Motin return;
58840ea77a0SAlexander Motin }
58940ea77a0SAlexander Motin bp->bio_to->geom->start(bp);
59040ea77a0SAlexander Motin } else {
59140ea77a0SAlexander Motin g_bioq_lock(&g_bio_run_down);
5920d883b11SAlexander Motin first = TAILQ_EMPTY(&g_bio_run_down.bio_queue);
59319fa21aaSPoul-Henning Kamp TAILQ_INSERT_TAIL(&g_bio_run_down.bio_queue, bp, bio_queue);
59440ea77a0SAlexander Motin bp->bio_flags |= BIO_ONQUEUE;
59519fa21aaSPoul-Henning Kamp g_bio_run_down.bio_queue_length++;
59619fa21aaSPoul-Henning Kamp g_bioq_unlock(&g_bio_run_down);
5972fccec19SPoul-Henning Kamp /* Pass it on down. */
5980d883b11SAlexander Motin if (first)
599dd84a43cSPoul-Henning Kamp wakeup(&g_wait_down);
600dd84a43cSPoul-Henning Kamp }
60140ea77a0SAlexander Motin }
602dd84a43cSPoul-Henning Kamp
603dd84a43cSPoul-Henning Kamp void
g_io_deliver(struct bio * bp,int error)60472840432SPoul-Henning Kamp g_io_deliver(struct bio *bp, int error)
605dd84a43cSPoul-Henning Kamp {
606e431d66cSAlexander Motin struct bintime now;
607801bb689SPoul-Henning Kamp struct g_consumer *cp;
608801bb689SPoul-Henning Kamp struct g_provider *pp;
60940ea77a0SAlexander Motin struct mtx *mtxp;
61040ea77a0SAlexander Motin int direct, first;
611dd84a43cSPoul-Henning Kamp
6128532d381SConrad Meyer biotrack(bp, __func__);
6138532d381SConrad Meyer
614e060b6bdSPoul-Henning Kamp KASSERT(bp != NULL, ("NULL bp in g_io_deliver"));
615801bb689SPoul-Henning Kamp pp = bp->bio_to;
616f7eeab17SPoul-Henning Kamp KASSERT(pp != NULL, ("NULL bio_to in g_io_deliver"));
617f7eeab17SPoul-Henning Kamp cp = bp->bio_from;
618f7eeab17SPoul-Henning Kamp if (cp == NULL) {
619f7eeab17SPoul-Henning Kamp bp->bio_error = error;
620f7eeab17SPoul-Henning Kamp bp->bio_done(bp);
621f7eeab17SPoul-Henning Kamp return;
622f7eeab17SPoul-Henning Kamp }
623801bb689SPoul-Henning Kamp KASSERT(cp != NULL, ("NULL bio_from in g_io_deliver"));
624801bb689SPoul-Henning Kamp KASSERT(cp->geom != NULL, ("NULL bio_from->geom in g_io_deliver"));
625fb231f36SEdward Tomasz Napierala #ifdef DIAGNOSTIC
626fb231f36SEdward Tomasz Napierala /*
627fb231f36SEdward Tomasz Napierala * Some classes - GJournal in particular - can modify bio's
628fb231f36SEdward Tomasz Napierala * private fields while the bio is in transit; G_GEOM_VOLATILE_BIO
629fb231f36SEdward Tomasz Napierala * flag means it's an expected behaviour for that particular geom.
630fb231f36SEdward Tomasz Napierala */
631fb231f36SEdward Tomasz Napierala if ((cp->geom->flags & G_GEOM_VOLATILE_BIO) == 0) {
632fb231f36SEdward Tomasz Napierala KASSERT(bp->bio_caller1 == bp->_bio_caller1,
633fb231f36SEdward Tomasz Napierala ("bio_caller1 used by the provider %s", pp->name));
634fb231f36SEdward Tomasz Napierala KASSERT(bp->bio_caller2 == bp->_bio_caller2,
635fb231f36SEdward Tomasz Napierala ("bio_caller2 used by the provider %s", pp->name));
636fb231f36SEdward Tomasz Napierala KASSERT(bp->bio_cflags == bp->_bio_cflags,
637fb231f36SEdward Tomasz Napierala ("bio_cflags used by the provider %s", pp->name));
638fb231f36SEdward Tomasz Napierala }
639fb231f36SEdward Tomasz Napierala #endif
64046aeebecSPawel Jakub Dawidek KASSERT(bp->bio_completed >= 0, ("bio_completed can't be less than 0"));
64146aeebecSPawel Jakub Dawidek KASSERT(bp->bio_completed <= bp->bio_length,
64246aeebecSPawel Jakub Dawidek ("bio_completed can't be greater than bio_length"));
6435ab413bfSPoul-Henning Kamp
644dd84a43cSPoul-Henning Kamp g_trace(G_T_BIO,
6450355b86eSPoul-Henning Kamp "g_io_deliver(%p) from %p(%s) to %p(%s) cmd %d error %d off %jd len %jd",
646801bb689SPoul-Henning Kamp bp, cp, cp->geom->name, pp, pp->name, bp->bio_cmd, error,
6470355b86eSPoul-Henning Kamp (intmax_t)bp->bio_offset, (intmax_t)bp->bio_length);
648801bb689SPoul-Henning Kamp
64919fa21aaSPoul-Henning Kamp KASSERT(!(bp->bio_flags & BIO_ONQUEUE),
65019fa21aaSPoul-Henning Kamp ("Bio already on queue bp=%p", bp));
65119fa21aaSPoul-Henning Kamp
652dcbd0fe5SPoul-Henning Kamp /*
653dcbd0fe5SPoul-Henning Kamp * XXX: next two doesn't belong here
654dcbd0fe5SPoul-Henning Kamp */
655e24cbd90SPoul-Henning Kamp bp->bio_bcount = bp->bio_length;
656e24cbd90SPoul-Henning Kamp bp->bio_resid = bp->bio_bcount - bp->bio_completed;
65719fa21aaSPoul-Henning Kamp
65840ea77a0SAlexander Motin direct = (pp->flags & G_PF_DIRECT_SEND) &&
65940ea77a0SAlexander Motin (cp->flags & G_CF_DIRECT_RECEIVE) &&
660ffc1cc95SAlexander Motin curthread != g_up_td;
66140ea77a0SAlexander Motin if (direct) {
66240ea77a0SAlexander Motin /* Block direct execution if less then half of stack left. */
66340ea77a0SAlexander Motin size_t st, su;
66440ea77a0SAlexander Motin GET_STACK_USAGE(st, su);
66540ea77a0SAlexander Motin if (su * 2 > st)
66640ea77a0SAlexander Motin direct = 0;
66740ea77a0SAlexander Motin }
66840ea77a0SAlexander Motin
6698827c821SPoul-Henning Kamp /*
6708827c821SPoul-Henning Kamp * The statistics collection is lockless, as such, but we
6718827c821SPoul-Henning Kamp * can not update one instance of the statistics from more
6728827c821SPoul-Henning Kamp * than one thread at a time, so grab the lock first.
6738827c821SPoul-Henning Kamp */
67440ea77a0SAlexander Motin if ((g_collectstats & G_STATS_CONSUMERS) != 0 ||
67540ea77a0SAlexander Motin ((g_collectstats & G_STATS_PROVIDERS) != 0 && pp->stat != NULL))
676e431d66cSAlexander Motin binuptime(&now);
67706bd74e1SAlexander Motin mtxp = mtx_pool_find(mtxpool_sleep, pp);
67840ea77a0SAlexander Motin mtx_lock(mtxp);
67940ea77a0SAlexander Motin if (g_collectstats & G_STATS_PROVIDERS)
680e431d66cSAlexander Motin devstat_end_transaction_bio_bt(pp->stat, bp, &now);
68140ea77a0SAlexander Motin if (g_collectstats & G_STATS_CONSUMERS)
682e431d66cSAlexander Motin devstat_end_transaction_bio_bt(cp->stat, bp, &now);
6839794a803SAlexander Motin #ifdef INVARIANTS
684c6ae9b5fSPoul-Henning Kamp cp->nend++;
6859794a803SAlexander Motin #endif
68640ea77a0SAlexander Motin mtx_unlock(mtxp);
68740ea77a0SAlexander Motin
68819fa21aaSPoul-Henning Kamp if (error != ENOMEM) {
68919fa21aaSPoul-Henning Kamp bp->bio_error = error;
69040ea77a0SAlexander Motin if (direct) {
69140ea77a0SAlexander Motin biodone(bp);
69240ea77a0SAlexander Motin } else {
69340ea77a0SAlexander Motin g_bioq_lock(&g_bio_run_up);
6940d883b11SAlexander Motin first = TAILQ_EMPTY(&g_bio_run_up.bio_queue);
69519fa21aaSPoul-Henning Kamp TAILQ_INSERT_TAIL(&g_bio_run_up.bio_queue, bp, bio_queue);
696276f72c5SPoul-Henning Kamp bp->bio_flags |= BIO_ONQUEUE;
69719fa21aaSPoul-Henning Kamp g_bio_run_up.bio_queue_length++;
69819fa21aaSPoul-Henning Kamp g_bioq_unlock(&g_bio_run_up);
6990d883b11SAlexander Motin if (first)
70019fa21aaSPoul-Henning Kamp wakeup(&g_wait_up);
70140ea77a0SAlexander Motin }
70219fa21aaSPoul-Henning Kamp return;
70319fa21aaSPoul-Henning Kamp }
704dd84a43cSPoul-Henning Kamp
7052cc9686eSPoul-Henning Kamp if (bootverbose)
706801bb689SPoul-Henning Kamp printf("ENOMEM %p on %p(%s)\n", bp, pp, pp->name);
70732f40fc9SWarner Losh atomic_add_long(&nomem_count, 1); /* Rare event, but no locks held */
7081b949c05SPawel Jakub Dawidek bp->bio_children = 0;
7091b949c05SPawel Jakub Dawidek bp->bio_inbed = 0;
71060114438SPawel Jakub Dawidek bp->bio_driver1 = NULL;
71160114438SPawel Jakub Dawidek bp->bio_driver2 = NULL;
71260114438SPawel Jakub Dawidek bp->bio_pflags = 0;
713801bb689SPoul-Henning Kamp g_io_request(bp, cp);
7143f2e5b85SWarner Losh pace = 1;
7153432e4fdSPoul-Henning Kamp return;
7163432e4fdSPoul-Henning Kamp }
717dd84a43cSPoul-Henning Kamp
718ee75e7deSKonstantin Belousov SYSCTL_DECL(_kern_geom);
719ee75e7deSKonstantin Belousov
720ee75e7deSKonstantin Belousov static long transient_maps;
721ee75e7deSKonstantin Belousov SYSCTL_LONG(_kern_geom, OID_AUTO, transient_maps, CTLFLAG_RD,
722ee75e7deSKonstantin Belousov &transient_maps, 0,
723ee75e7deSKonstantin Belousov "Total count of the transient mapping requests");
724ee75e7deSKonstantin Belousov u_int transient_map_retries = 10;
725ee75e7deSKonstantin Belousov SYSCTL_UINT(_kern_geom, OID_AUTO, transient_map_retries, CTLFLAG_RW,
726ee75e7deSKonstantin Belousov &transient_map_retries, 0,
727ee75e7deSKonstantin Belousov "Max count of retries used before giving up on creating transient map");
728ee75e7deSKonstantin Belousov int transient_map_hard_failures;
729ee75e7deSKonstantin Belousov SYSCTL_INT(_kern_geom, OID_AUTO, transient_map_hard_failures, CTLFLAG_RD,
730ee75e7deSKonstantin Belousov &transient_map_hard_failures, 0,
731ee75e7deSKonstantin Belousov "Failures to establish the transient mapping due to retry attempts "
732ee75e7deSKonstantin Belousov "exhausted");
733ee75e7deSKonstantin Belousov int transient_map_soft_failures;
734ee75e7deSKonstantin Belousov SYSCTL_INT(_kern_geom, OID_AUTO, transient_map_soft_failures, CTLFLAG_RD,
735ee75e7deSKonstantin Belousov &transient_map_soft_failures, 0,
736ee75e7deSKonstantin Belousov "Count of retried failures to establish the transient mapping");
737ee75e7deSKonstantin Belousov int inflight_transient_maps;
738ee75e7deSKonstantin Belousov SYSCTL_INT(_kern_geom, OID_AUTO, inflight_transient_maps, CTLFLAG_RD,
739ee75e7deSKonstantin Belousov &inflight_transient_maps, 0,
740ee75e7deSKonstantin Belousov "Current count of the active transient maps");
74132f40fc9SWarner Losh SYSCTL_ULONG(_kern_geom, OID_AUTO, nomem_count, CTLFLAG_RD,
74232f40fc9SWarner Losh &nomem_count, 0,
74332f40fc9SWarner Losh "Total count of requests completed with status of ENOMEM");
74432f40fc9SWarner Losh SYSCTL_ULONG(_kern_geom, OID_AUTO, pause_count, CTLFLAG_RD,
74532f40fc9SWarner Losh &pause_count, 0,
74632f40fc9SWarner Losh "Total count of requests stalled due to low memory in g_down");
747ee75e7deSKonstantin Belousov
748ee75e7deSKonstantin Belousov static int
g_io_transient_map_bio(struct bio * bp)749ee75e7deSKonstantin Belousov g_io_transient_map_bio(struct bio *bp)
750ee75e7deSKonstantin Belousov {
751ee75e7deSKonstantin Belousov vm_offset_t addr;
752ee75e7deSKonstantin Belousov long size;
753ee75e7deSKonstantin Belousov u_int retried;
754ee75e7deSKonstantin Belousov
7556c83fce3SKonstantin Belousov KASSERT(unmapped_buf_allowed, ("unmapped disabled"));
7566c83fce3SKonstantin Belousov
757ee75e7deSKonstantin Belousov size = round_page(bp->bio_ma_offset + bp->bio_length);
758ee75e7deSKonstantin Belousov KASSERT(size / PAGE_SIZE == bp->bio_ma_n, ("Bio too short %p", bp));
759ee75e7deSKonstantin Belousov addr = 0;
760ee75e7deSKonstantin Belousov retried = 0;
761ee75e7deSKonstantin Belousov atomic_add_long(&transient_maps, 1);
762ee75e7deSKonstantin Belousov retry:
7635f518366SJeff Roberson if (vmem_alloc(transient_arena, size, M_BESTFIT | M_NOWAIT, &addr)) {
764ee75e7deSKonstantin Belousov if (transient_map_retries != 0 &&
765ee75e7deSKonstantin Belousov retried >= transient_map_retries) {
766ee75e7deSKonstantin Belousov CTR2(KTR_GEOM, "g_down cannot map bp %p provider %s",
767ee75e7deSKonstantin Belousov bp, bp->bio_to->name);
768ee75e7deSKonstantin Belousov atomic_add_int(&transient_map_hard_failures, 1);
76940ea77a0SAlexander Motin return (EDEADLK/* XXXKIB */);
770ee75e7deSKonstantin Belousov } else {
771ee75e7deSKonstantin Belousov /*
772ee75e7deSKonstantin Belousov * Naive attempt to quisce the I/O to get more
773ee75e7deSKonstantin Belousov * in-flight requests completed and defragment
7745f518366SJeff Roberson * the transient_arena.
775ee75e7deSKonstantin Belousov */
776ee75e7deSKonstantin Belousov CTR3(KTR_GEOM, "g_down retrymap bp %p provider %s r %d",
777ee75e7deSKonstantin Belousov bp, bp->bio_to->name, retried);
778ee75e7deSKonstantin Belousov pause("g_d_tra", hz / 10);
779ee75e7deSKonstantin Belousov retried++;
780ee75e7deSKonstantin Belousov atomic_add_int(&transient_map_soft_failures, 1);
781ee75e7deSKonstantin Belousov goto retry;
782ee75e7deSKonstantin Belousov }
783ee75e7deSKonstantin Belousov }
784ee75e7deSKonstantin Belousov atomic_add_int(&inflight_transient_maps, 1);
785ee75e7deSKonstantin Belousov pmap_qenter((vm_offset_t)addr, bp->bio_ma, OFF_TO_IDX(size));
786ee75e7deSKonstantin Belousov bp->bio_data = (caddr_t)addr + bp->bio_ma_offset;
787ee75e7deSKonstantin Belousov bp->bio_flags |= BIO_TRANSIENT_MAPPING;
788ee75e7deSKonstantin Belousov bp->bio_flags &= ~BIO_UNMAPPED;
78940ea77a0SAlexander Motin return (EJUSTRETURN);
790ee75e7deSKonstantin Belousov }
791ee75e7deSKonstantin Belousov
792dd84a43cSPoul-Henning Kamp void
g_io_schedule_down(struct thread * tp __unused)793dd84a43cSPoul-Henning Kamp g_io_schedule_down(struct thread *tp __unused)
794dd84a43cSPoul-Henning Kamp {
795dd84a43cSPoul-Henning Kamp struct bio *bp;
796e39d70d4SPoul-Henning Kamp int error;
797dd84a43cSPoul-Henning Kamp
798dd84a43cSPoul-Henning Kamp for(;;) {
799f0e185d7SPoul-Henning Kamp g_bioq_lock(&g_bio_run_down);
800dd84a43cSPoul-Henning Kamp bp = g_bioq_first(&g_bio_run_down);
801f0e185d7SPoul-Henning Kamp if (bp == NULL) {
80249dbb61dSRobert Watson CTR0(KTR_GEOM, "g_down going to sleep");
803f0e185d7SPoul-Henning Kamp msleep(&g_wait_down, &g_bio_run_down.bio_queue_lock,
8047fc019afSAlexander Motin PRIBIO | PDROP, "-", 0);
805f0e185d7SPoul-Henning Kamp continue;
806f0e185d7SPoul-Henning Kamp }
80749dbb61dSRobert Watson CTR0(KTR_GEOM, "g_down has work to do");
808f0e185d7SPoul-Henning Kamp g_bioq_unlock(&g_bio_run_down);
8098532d381SConrad Meyer biotrack(bp, __func__);
8103f2e5b85SWarner Losh if (pace != 0) {
8113f2e5b85SWarner Losh /*
812*6d83b381SWarner Losh * There has been at least one memory allocation failure
813*6d83b381SWarner Losh * since the last I/O completed. Pause 1ms to give the
814*6d83b381SWarner Losh * system a chance to free up memory. Pause time is not
815*6d83b381SWarner Losh * scaled to the number of I/O failures since they tend
816*6d83b381SWarner Losh * to cluster and the number is not predictive of how
817*6d83b381SWarner Losh * long a pause is needed.
818*6d83b381SWarner Losh *
819*6d83b381SWarner Losh * Older versions had a longer pause, which limited the
820*6d83b381SWarner Losh * IOPS to 10, which prolonged memory shortages that could
821*6d83b381SWarner Losh * be alleviated by I/O completing since it eliminated
822*6d83b381SWarner Losh * direct dispatch as well.
8233f2e5b85SWarner Losh *
8243f2e5b85SWarner Losh * XXX This pacing is really lame. It needs to be solved
8253f2e5b85SWarner Losh * by other methods. This is OK only because the worst
8263f2e5b85SWarner Losh * case scenario is so rare. In the worst case scenario
8273f2e5b85SWarner Losh * all memory is tied up waiting for I/O to complete
8283f2e5b85SWarner Losh * which can never happen since we can't allocate bios
8293f2e5b85SWarner Losh * for that I/O.
8303f2e5b85SWarner Losh */
8313f2e5b85SWarner Losh CTR0(KTR_GEOM, "g_down pacing self");
83232f40fc9SWarner Losh pause_count++; /* g_down has only one thread */
833*6d83b381SWarner Losh pause_sbt("g_down", SBT_1MS, 0, 0);
8343f2e5b85SWarner Losh pace = 0;
835376ceb79SPoul-Henning Kamp }
83640ea77a0SAlexander Motin CTR2(KTR_GEOM, "g_down processing bp %p provider %s", bp,
83740ea77a0SAlexander Motin bp->bio_to->name);
838e39d70d4SPoul-Henning Kamp error = g_io_check(bp);
83940ea77a0SAlexander Motin if (error >= 0) {
84049dbb61dSRobert Watson CTR3(KTR_GEOM, "g_down g_io_check on bp %p provider "
84149dbb61dSRobert Watson "%s returned %d", bp, bp->bio_to->name, error);
842e39d70d4SPoul-Henning Kamp g_io_deliver(bp, error);
843e39d70d4SPoul-Henning Kamp continue;
844e39d70d4SPoul-Henning Kamp }
84551460da8SJohn Baldwin THREAD_NO_SLEEPING();
84649dbb61dSRobert Watson CTR4(KTR_GEOM, "g_down starting bp %p provider %s off %ld "
84749dbb61dSRobert Watson "len %ld", bp, bp->bio_to->name, bp->bio_offset,
84849dbb61dSRobert Watson bp->bio_length);
849dd84a43cSPoul-Henning Kamp bp->bio_to->geom->start(bp);
85051460da8SJohn Baldwin THREAD_SLEEPING_OK();
851dd84a43cSPoul-Henning Kamp }
852dd84a43cSPoul-Henning Kamp }
853dd84a43cSPoul-Henning Kamp
854dd84a43cSPoul-Henning Kamp void
g_io_schedule_up(struct thread * tp __unused)855dd84a43cSPoul-Henning Kamp g_io_schedule_up(struct thread *tp __unused)
856dd84a43cSPoul-Henning Kamp {
857dd84a43cSPoul-Henning Kamp struct bio *bp;
8580c4440c3SEdward Tomasz Napierala
859dd84a43cSPoul-Henning Kamp for(;;) {
860f0e185d7SPoul-Henning Kamp g_bioq_lock(&g_bio_run_up);
8610c4440c3SEdward Tomasz Napierala bp = g_bioq_first(&g_bio_run_up);
8620c4440c3SEdward Tomasz Napierala if (bp == NULL) {
8630c4440c3SEdward Tomasz Napierala CTR0(KTR_GEOM, "g_up going to sleep");
8640c4440c3SEdward Tomasz Napierala msleep(&g_wait_up, &g_bio_run_up.bio_queue_lock,
8650c4440c3SEdward Tomasz Napierala PRIBIO | PDROP, "-", 0);
8665fcf4e43SPoul-Henning Kamp continue;
8675fcf4e43SPoul-Henning Kamp }
868f0e185d7SPoul-Henning Kamp g_bioq_unlock(&g_bio_run_up);
86951460da8SJohn Baldwin THREAD_NO_SLEEPING();
87049dbb61dSRobert Watson CTR4(KTR_GEOM, "g_up biodone bp %p provider %s off "
871c4901b67SSean Bruno "%jd len %ld", bp, bp->bio_to->name,
87249dbb61dSRobert Watson bp->bio_offset, bp->bio_length);
87353706245SPoul-Henning Kamp biodone(bp);
87451460da8SJohn Baldwin THREAD_SLEEPING_OK();
875dd84a43cSPoul-Henning Kamp }
876dd84a43cSPoul-Henning Kamp }
877dd84a43cSPoul-Henning Kamp
878dd84a43cSPoul-Henning Kamp void *
g_read_data(struct g_consumer * cp,off_t offset,off_t length,int * error)879dd84a43cSPoul-Henning Kamp g_read_data(struct g_consumer *cp, off_t offset, off_t length, int *error)
880dd84a43cSPoul-Henning Kamp {
881dd84a43cSPoul-Henning Kamp struct bio *bp;
882dd84a43cSPoul-Henning Kamp void *ptr;
883dd84a43cSPoul-Henning Kamp int errorc;
884dd84a43cSPoul-Henning Kamp
8858dd5480dSPawel Jakub Dawidek KASSERT(length > 0 && length >= cp->provider->sectorsize &&
886cd853791SKonstantin Belousov length <= maxphys, ("g_read_data(): invalid length %jd",
8878dd5480dSPawel Jakub Dawidek (intmax_t)length));
8883eb6ffdfSPoul-Henning Kamp
889a2033c96SPoul-Henning Kamp bp = g_alloc_bio();
890dd84a43cSPoul-Henning Kamp bp->bio_cmd = BIO_READ;
891dd84a43cSPoul-Henning Kamp bp->bio_done = NULL;
892dd84a43cSPoul-Henning Kamp bp->bio_offset = offset;
893dd84a43cSPoul-Henning Kamp bp->bio_length = length;
894a163d034SWarner Losh ptr = g_malloc(length, M_WAITOK);
895dd84a43cSPoul-Henning Kamp bp->bio_data = ptr;
896dd84a43cSPoul-Henning Kamp g_io_request(bp, cp);
89753706245SPoul-Henning Kamp errorc = biowait(bp, "gread");
898d91d2b51SMark Johnston if (errorc == 0 && bp->bio_completed != length)
899d91d2b51SMark Johnston errorc = EIO;
900dd84a43cSPoul-Henning Kamp if (error != NULL)
901dd84a43cSPoul-Henning Kamp *error = errorc;
902dd84a43cSPoul-Henning Kamp g_destroy_bio(bp);
903dd84a43cSPoul-Henning Kamp if (errorc) {
904dd84a43cSPoul-Henning Kamp g_free(ptr);
905dd84a43cSPoul-Henning Kamp ptr = NULL;
906dd84a43cSPoul-Henning Kamp }
907dd84a43cSPoul-Henning Kamp return (ptr);
908dd84a43cSPoul-Henning Kamp }
90990b1cd56SPoul-Henning Kamp
910dffce215SKirk McKusick /*
911dffce215SKirk McKusick * A read function for use by ffs_sbget when used by GEOM-layer routines.
912dffce215SKirk McKusick */
913dffce215SKirk McKusick int
g_use_g_read_data(void * devfd,off_t loc,void ** bufp,int size)914dffce215SKirk McKusick g_use_g_read_data(void *devfd, off_t loc, void **bufp, int size)
915dffce215SKirk McKusick {
916dffce215SKirk McKusick struct g_consumer *cp;
917dffce215SKirk McKusick
918efbf3964SKirk McKusick KASSERT(*bufp == NULL,
919efbf3964SKirk McKusick ("g_use_g_read_data: non-NULL *bufp %p\n", *bufp));
920efbf3964SKirk McKusick
921dffce215SKirk McKusick cp = (struct g_consumer *)devfd;
922dffce215SKirk McKusick /*
923dffce215SKirk McKusick * Take care not to issue an invalid I/O request. The offset of
924dffce215SKirk McKusick * the superblock candidate must be multiples of the provider's
925dffce215SKirk McKusick * sector size, otherwise an FFS can't exist on the provider
926dffce215SKirk McKusick * anyway.
927dffce215SKirk McKusick */
928dffce215SKirk McKusick if (loc % cp->provider->sectorsize != 0)
929dffce215SKirk McKusick return (ENOENT);
930dffce215SKirk McKusick *bufp = g_read_data(cp, loc, size, NULL);
931dffce215SKirk McKusick if (*bufp == NULL)
932dffce215SKirk McKusick return (ENOENT);
933dffce215SKirk McKusick return (0);
934dffce215SKirk McKusick }
935dffce215SKirk McKusick
93690b1cd56SPoul-Henning Kamp int
g_write_data(struct g_consumer * cp,off_t offset,void * ptr,off_t length)93790b1cd56SPoul-Henning Kamp g_write_data(struct g_consumer *cp, off_t offset, void *ptr, off_t length)
93890b1cd56SPoul-Henning Kamp {
93990b1cd56SPoul-Henning Kamp struct bio *bp;
94090b1cd56SPoul-Henning Kamp int error;
94190b1cd56SPoul-Henning Kamp
9428dd5480dSPawel Jakub Dawidek KASSERT(length > 0 && length >= cp->provider->sectorsize &&
943cd853791SKonstantin Belousov length <= maxphys, ("g_write_data(): invalid length %jd",
9448dd5480dSPawel Jakub Dawidek (intmax_t)length));
9453eb6ffdfSPoul-Henning Kamp
946a2033c96SPoul-Henning Kamp bp = g_alloc_bio();
94790b1cd56SPoul-Henning Kamp bp->bio_cmd = BIO_WRITE;
94890b1cd56SPoul-Henning Kamp bp->bio_done = NULL;
94990b1cd56SPoul-Henning Kamp bp->bio_offset = offset;
95090b1cd56SPoul-Henning Kamp bp->bio_length = length;
95190b1cd56SPoul-Henning Kamp bp->bio_data = ptr;
95290b1cd56SPoul-Henning Kamp g_io_request(bp, cp);
95390b1cd56SPoul-Henning Kamp error = biowait(bp, "gwrite");
954d91d2b51SMark Johnston if (error == 0 && bp->bio_completed != length)
955d91d2b51SMark Johnston error = EIO;
95690b1cd56SPoul-Henning Kamp g_destroy_bio(bp);
95790b1cd56SPoul-Henning Kamp return (error);
95890b1cd56SPoul-Henning Kamp }
95972e33095SPawel Jakub Dawidek
960dffce215SKirk McKusick /*
961dffce215SKirk McKusick * A write function for use by ffs_sbput when used by GEOM-layer routines.
962dffce215SKirk McKusick */
963dffce215SKirk McKusick int
g_use_g_write_data(void * devfd,off_t loc,void * buf,int size)964dffce215SKirk McKusick g_use_g_write_data(void *devfd, off_t loc, void *buf, int size)
965dffce215SKirk McKusick {
966dffce215SKirk McKusick
967dffce215SKirk McKusick return (g_write_data((struct g_consumer *)devfd, loc, buf, size));
968dffce215SKirk McKusick }
969dffce215SKirk McKusick
9702b17fb95SPawel Jakub Dawidek int
g_delete_data(struct g_consumer * cp,off_t offset,off_t length)9712b17fb95SPawel Jakub Dawidek g_delete_data(struct g_consumer *cp, off_t offset, off_t length)
9722b17fb95SPawel Jakub Dawidek {
9732b17fb95SPawel Jakub Dawidek struct bio *bp;
9742b17fb95SPawel Jakub Dawidek int error;
9752b17fb95SPawel Jakub Dawidek
976eed6cda9SPoul-Henning Kamp KASSERT(length > 0 && length >= cp->provider->sectorsize,
977eed6cda9SPoul-Henning Kamp ("g_delete_data(): invalid length %jd", (intmax_t)length));
9782b17fb95SPawel Jakub Dawidek
9792b17fb95SPawel Jakub Dawidek bp = g_alloc_bio();
9802b17fb95SPawel Jakub Dawidek bp->bio_cmd = BIO_DELETE;
9812b17fb95SPawel Jakub Dawidek bp->bio_done = NULL;
9822b17fb95SPawel Jakub Dawidek bp->bio_offset = offset;
9832b17fb95SPawel Jakub Dawidek bp->bio_length = length;
9842b17fb95SPawel Jakub Dawidek bp->bio_data = NULL;
9852b17fb95SPawel Jakub Dawidek g_io_request(bp, cp);
9862b17fb95SPawel Jakub Dawidek error = biowait(bp, "gdelete");
987d91d2b51SMark Johnston if (error == 0 && bp->bio_completed != length)
988d91d2b51SMark Johnston error = EIO;
9892b17fb95SPawel Jakub Dawidek g_destroy_bio(bp);
9902b17fb95SPawel Jakub Dawidek return (error);
9912b17fb95SPawel Jakub Dawidek }
9922b17fb95SPawel Jakub Dawidek
99372e33095SPawel Jakub Dawidek void
g_print_bio(const char * prefix,const struct bio * bp,const char * fmtsuffix,...)994ac03832eSConrad Meyer g_print_bio(const char *prefix, const struct bio *bp, const char *fmtsuffix,
995ac03832eSConrad Meyer ...)
996ac03832eSConrad Meyer {
997ac03832eSConrad Meyer #ifndef PRINTF_BUFR_SIZE
998ac03832eSConrad Meyer #define PRINTF_BUFR_SIZE 64
999ac03832eSConrad Meyer #endif
1000ac03832eSConrad Meyer char bufr[PRINTF_BUFR_SIZE];
1001ac03832eSConrad Meyer struct sbuf sb, *sbp __unused;
1002ac03832eSConrad Meyer va_list ap;
1003ac03832eSConrad Meyer
1004ac03832eSConrad Meyer sbp = sbuf_new(&sb, bufr, sizeof(bufr), SBUF_FIXEDLEN);
1005ac03832eSConrad Meyer KASSERT(sbp != NULL, ("sbuf_new misused?"));
1006ac03832eSConrad Meyer
1007ac03832eSConrad Meyer sbuf_set_drain(&sb, sbuf_printf_drain, NULL);
1008ac03832eSConrad Meyer
1009ac03832eSConrad Meyer sbuf_cat(&sb, prefix);
1010ac03832eSConrad Meyer g_format_bio(&sb, bp);
1011ac03832eSConrad Meyer
1012ac03832eSConrad Meyer va_start(ap, fmtsuffix);
1013ac03832eSConrad Meyer sbuf_vprintf(&sb, fmtsuffix, ap);
1014ac03832eSConrad Meyer va_end(ap);
1015ac03832eSConrad Meyer
1016ac03832eSConrad Meyer sbuf_nl_terminate(&sb);
1017ac03832eSConrad Meyer
1018ac03832eSConrad Meyer sbuf_finish(&sb);
1019ac03832eSConrad Meyer sbuf_delete(&sb);
1020ac03832eSConrad Meyer }
1021ac03832eSConrad Meyer
1022ac03832eSConrad Meyer void
g_format_bio(struct sbuf * sb,const struct bio * bp)1023ac03832eSConrad Meyer g_format_bio(struct sbuf *sb, const struct bio *bp)
102472e33095SPawel Jakub Dawidek {
102572e33095SPawel Jakub Dawidek const char *pname, *cmd = NULL;
102672e33095SPawel Jakub Dawidek
102772e33095SPawel Jakub Dawidek if (bp->bio_to != NULL)
102872e33095SPawel Jakub Dawidek pname = bp->bio_to->name;
102985f7e9a4SKirk McKusick else if (bp->bio_parent != NULL && bp->bio_parent->bio_to != NULL)
103085f7e9a4SKirk McKusick pname = bp->bio_parent->bio_to->name;
103172e33095SPawel Jakub Dawidek else
103272e33095SPawel Jakub Dawidek pname = "[unknown]";
103372e33095SPawel Jakub Dawidek
103472e33095SPawel Jakub Dawidek switch (bp->bio_cmd) {
103572e33095SPawel Jakub Dawidek case BIO_GETATTR:
103672e33095SPawel Jakub Dawidek cmd = "GETATTR";
1037ac03832eSConrad Meyer sbuf_printf(sb, "%s[%s(attr=%s)]", pname, cmd,
1038ac03832eSConrad Meyer bp->bio_attribute);
103972e33095SPawel Jakub Dawidek return;
1040c3618c65SPawel Jakub Dawidek case BIO_FLUSH:
1041c3618c65SPawel Jakub Dawidek cmd = "FLUSH";
1042ac03832eSConrad Meyer sbuf_printf(sb, "%s[%s]", pname, cmd);
1043c3618c65SPawel Jakub Dawidek return;
10449a6844d5SKenneth D. Merry case BIO_ZONE: {
10459a6844d5SKenneth D. Merry char *subcmd = NULL;
10469a6844d5SKenneth D. Merry cmd = "ZONE";
10479a6844d5SKenneth D. Merry switch (bp->bio_zone.zone_cmd) {
10489a6844d5SKenneth D. Merry case DISK_ZONE_OPEN:
10499a6844d5SKenneth D. Merry subcmd = "OPEN";
10509a6844d5SKenneth D. Merry break;
10519a6844d5SKenneth D. Merry case DISK_ZONE_CLOSE:
10529a6844d5SKenneth D. Merry subcmd = "CLOSE";
10539a6844d5SKenneth D. Merry break;
10549a6844d5SKenneth D. Merry case DISK_ZONE_FINISH:
10559a6844d5SKenneth D. Merry subcmd = "FINISH";
10569a6844d5SKenneth D. Merry break;
10579a6844d5SKenneth D. Merry case DISK_ZONE_RWP:
10589a6844d5SKenneth D. Merry subcmd = "RWP";
10599a6844d5SKenneth D. Merry break;
10609a6844d5SKenneth D. Merry case DISK_ZONE_REPORT_ZONES:
10619a6844d5SKenneth D. Merry subcmd = "REPORT ZONES";
10629a6844d5SKenneth D. Merry break;
10639a6844d5SKenneth D. Merry case DISK_ZONE_GET_PARAMS:
10649a6844d5SKenneth D. Merry subcmd = "GET PARAMS";
10659a6844d5SKenneth D. Merry break;
10669a6844d5SKenneth D. Merry default:
10679a6844d5SKenneth D. Merry subcmd = "UNKNOWN";
10689a6844d5SKenneth D. Merry break;
10699a6844d5SKenneth D. Merry }
1070ac03832eSConrad Meyer sbuf_printf(sb, "%s[%s,%s]", pname, cmd, subcmd);
10719a6844d5SKenneth D. Merry return;
10729a6844d5SKenneth D. Merry }
107372e33095SPawel Jakub Dawidek case BIO_READ:
107472e33095SPawel Jakub Dawidek cmd = "READ";
10757ce513a5SEdward Tomasz Napierala break;
107672e33095SPawel Jakub Dawidek case BIO_WRITE:
107772e33095SPawel Jakub Dawidek cmd = "WRITE";
10787ce513a5SEdward Tomasz Napierala break;
107972e33095SPawel Jakub Dawidek case BIO_DELETE:
108072e33095SPawel Jakub Dawidek cmd = "DELETE";
10817ce513a5SEdward Tomasz Napierala break;
108272e33095SPawel Jakub Dawidek default:
108372e33095SPawel Jakub Dawidek cmd = "UNKNOWN";
1084ac03832eSConrad Meyer sbuf_printf(sb, "%s[%s()]", pname, cmd);
108572e33095SPawel Jakub Dawidek return;
108672e33095SPawel Jakub Dawidek }
1087ac03832eSConrad Meyer sbuf_printf(sb, "%s[%s(offset=%jd, length=%jd)]", pname, cmd,
10887ce513a5SEdward Tomasz Napierala (intmax_t)bp->bio_offset, (intmax_t)bp->bio_length);
108972e33095SPawel Jakub Dawidek }
1090