]> git.saurik.com Git - apple/xnu.git/blob - bsd/ufs/ufs/ufs_readwrite.c
1fcbcfcd3b02070a36a61eb04f582f498eefec3c
[apple/xnu.git] / bsd / ufs / ufs / ufs_readwrite.c
1 /*
2 * Copyright (c) 2000 Apple Computer, Inc. All rights reserved.
3 *
4 * @APPLE_LICENSE_HEADER_START@
5 *
6 * Copyright (c) 1999-2003 Apple Computer, Inc. All Rights Reserved.
7 *
8 * This file contains Original Code and/or Modifications of Original Code
9 * as defined in and that are subject to the Apple Public Source License
10 * Version 2.0 (the 'License'). You may not use this file except in
11 * compliance with the License. Please obtain a copy of the License at
12 * http://www.opensource.apple.com/apsl/ and read it before using this
13 * file.
14 *
15 * The Original Code and all software distributed under the License are
16 * distributed on an 'AS IS' basis, WITHOUT WARRANTY OF ANY KIND, EITHER
17 * EXPRESS OR IMPLIED, AND APPLE HEREBY DISCLAIMS ALL SUCH WARRANTIES,
18 * INCLUDING WITHOUT LIMITATION, ANY WARRANTIES OF MERCHANTABILITY,
19 * FITNESS FOR A PARTICULAR PURPOSE, QUIET ENJOYMENT OR NON-INFRINGEMENT.
20 * Please see the License for the specific language governing rights and
21 * limitations under the License.
22 *
23 * @APPLE_LICENSE_HEADER_END@
24 */
25 /* Copyright (c) 1995 NeXT Computer, Inc. All Rights Reserved */
26 /*-
27 * Copyright (c) 1993
28 * The Regents of the University of California. All rights reserved.
29 *
30 * Redistribution and use in source and binary forms, with or without
31 * modification, are permitted provided that the following conditions
32 * are met:
33 * 1. Redistributions of source code must retain the above copyright
34 * notice, this list of conditions and the following disclaimer.
35 * 2. Redistributions in binary form must reproduce the above copyright
36 * notice, this list of conditions and the following disclaimer in the
37 * documentation and/or other materials provided with the distribution.
38 * 3. All advertising materials mentioning features or use of this software
39 * must display the following acknowledgement:
40 * This product includes software developed by the University of
41 * California, Berkeley and its contributors.
42 * 4. Neither the name of the University nor the names of its contributors
43 * may be used to endorse or promote products derived from this software
44 * without specific prior written permission.
45 *
46 * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
47 * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
48 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
49 * ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
50 * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
51 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
52 * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
53 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
54 * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
55 * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
56 * SUCH DAMAGE.
57 *
58 * @(#)ufs_readwrite.c 8.11 (Berkeley) 5/8/95
59 */
60
61 #define BLKSIZE(a, b, c) blksize(a, b, c)
62 #define FS struct fs
63 #define I_FS i_fs
64 #define PGRD ffs_pgrd
65 #define PGRD_S "ffs_pgrd"
66 #define PGWR ffs_pgwr
67 #define PGWR_S "ffs_pgwr"
68
69 /*
70 * Vnode op for reading.
71 */
72 /* ARGSUSED */
73 ffs_read(ap)
74 struct vop_read_args /* {
75 struct vnode *a_vp;
76 struct uio *a_uio;
77 int a_ioflag;
78 struct ucred *a_cred;
79 } */ *ap;
80 {
81 register struct vnode *vp;
82 register struct inode *ip;
83 register struct uio *uio;
84 register FS *fs;
85 struct buf *bp = (struct buf *)0;
86 ufs_daddr_t lbn, nextlbn;
87 off_t bytesinfile;
88 long size, xfersize, blkoffset;
89 int devBlockSize=0;
90 int error;
91 u_short mode;
92 #if REV_ENDIAN_FS
93 int rev_endian=0;
94 #endif /* REV_ENDIAN_FS */
95
96 vp = ap->a_vp;
97 ip = VTOI(vp);
98 mode = ip->i_mode;
99 uio = ap->a_uio;
100
101 #if REV_ENDIAN_FS
102 rev_endian=(vp->v_mount->mnt_flag & MNT_REVEND);
103 #endif /* REV_ENDIAN_FS */
104
105 #if DIAGNOSTIC
106 if (uio->uio_rw != UIO_READ)
107 panic("ffs_read: invalid uio_rw = %x", uio->uio_rw);
108
109 if (vp->v_type == VLNK) {
110 if ((int)ip->i_size < vp->v_mount->mnt_maxsymlinklen)
111 panic("ffs_read: short symlink = %d", ip->i_size);
112 } else if (vp->v_type != VREG && vp->v_type != VDIR)
113 panic("ffs_read: invalid v_type = %x", vp->v_type);
114 #endif
115 fs = ip->I_FS;
116 if (uio->uio_offset < 0)
117 return (EINVAL);
118 if (uio->uio_offset > fs->fs_maxfilesize)
119 return (EFBIG);
120
121 VOP_DEVBLOCKSIZE(ip->i_devvp, &devBlockSize);
122
123 if (UBCISVALID(vp)) {
124 error = cluster_read(vp, uio, (off_t)ip->i_size,
125 devBlockSize, 0);
126 } else {
127 for (error = 0, bp = NULL; uio->uio_resid > 0;
128 bp = NULL) {
129 if ((bytesinfile = ip->i_size - uio->uio_offset) <= 0)
130 break;
131 lbn = lblkno(fs, uio->uio_offset);
132 nextlbn = lbn + 1;
133 size = BLKSIZE(fs, ip, lbn);
134 blkoffset = blkoff(fs, uio->uio_offset);
135 xfersize = fs->fs_bsize - blkoffset;
136 if (uio->uio_resid < xfersize)
137 xfersize = uio->uio_resid;
138 if (bytesinfile < xfersize)
139 xfersize = bytesinfile;
140
141 if (lblktosize(fs, nextlbn) >= ip->i_size)
142 error = bread(vp, lbn, size, NOCRED, &bp);
143 else if (lbn - 1 == vp->v_lastr && !(vp->v_flag & VRAOFF)) {
144 int nextsize = BLKSIZE(fs, ip, nextlbn);
145 error = breadn(vp, lbn,
146 size, &nextlbn, &nextsize, 1, NOCRED, &bp);
147 } else
148 error = bread(vp, lbn, size, NOCRED, &bp);
149 if (error)
150 break;
151 vp->v_lastr = lbn;
152
153 /*
154 * We should only get non-zero b_resid when an I/O error
155 * has occurred, which should cause us to break above.
156 * However, if the short read did not cause an error,
157 * then we want to ensure that we do not uiomove bad
158 * or uninitialized data.
159 */
160 size -= bp->b_resid;
161 if (size < xfersize) {
162 if (size == 0)
163 break;
164 xfersize = size;
165 }
166 #if REV_ENDIAN_FS
167 if (rev_endian && S_ISDIR(mode)) {
168 byte_swap_dir_block_in((char *)bp->b_data + blkoffset, xfersize);
169 }
170 #endif /* REV_ENDIAN_FS */
171 if (error =
172 uiomove((char *)bp->b_data + blkoffset, (int)xfersize, uio)) {
173 #if REV_ENDIAN_FS
174 if (rev_endian && S_ISDIR(mode)) {
175 byte_swap_dir_block_in((char *)bp->b_data + blkoffset, xfersize);
176 }
177 #endif /* REV_ENDIAN_FS */
178 break;
179 }
180
181 #if REV_ENDIAN_FS
182 if (rev_endian && S_ISDIR(mode)) {
183 byte_swap_dir_out((char *)bp->b_data + blkoffset, xfersize);
184 }
185 #endif /* REV_ENDIAN_FS */
186 if (S_ISREG(mode) && (xfersize + blkoffset == fs->fs_bsize ||
187 uio->uio_offset == ip->i_size))
188 bp->b_flags |= B_AGE;
189 brelse(bp);
190 }
191 }
192 if (bp != NULL)
193 brelse(bp);
194 ip->i_flag |= IN_ACCESS;
195 return (error);
196 }
197
198 /*
199 * Vnode op for writing.
200 */
201 ffs_write(ap)
202 struct vop_write_args /* {
203 struct vnode *a_vp;
204 struct uio *a_uio;
205 int a_ioflag;
206 struct ucred *a_cred;
207 } */ *ap;
208 {
209 register struct vnode *vp;
210 register struct uio *uio;
211 register struct inode *ip;
212 register FS *fs;
213 struct buf *bp;
214 struct proc *p;
215 ufs_daddr_t lbn;
216 off_t osize;
217 int blkoffset, flags, ioflag, resid, rsd, size, xfersize;
218 int devBlockSize=0;
219 int save_error=0, save_size=0;
220 int blkalloc = 0;
221 int error = 0;
222
223 #if REV_ENDIAN_FS
224 int rev_endian=0;
225 #endif /* REV_ENDIAN_FS */
226
227 ioflag = ap->a_ioflag;
228 uio = ap->a_uio;
229 vp = ap->a_vp;
230 ip = VTOI(vp);
231 #if REV_ENDIAN_FS
232 rev_endian=(vp->v_mount->mnt_flag & MNT_REVEND);
233 #endif /* REV_ENDIAN_FS */
234
235 #if DIAGNOSTIC
236 if (uio->uio_rw != UIO_WRITE)
237 panic("ffs_write: uio_rw = %x\n", uio->uio_rw);
238 #endif
239
240 switch (vp->v_type) {
241 case VREG:
242 if (ioflag & IO_APPEND)
243 uio->uio_offset = ip->i_size;
244 if ((ip->i_flags & APPEND) && uio->uio_offset != ip->i_size)
245 return (EPERM);
246 /* FALLTHROUGH */
247 case VLNK:
248 break;
249 case VDIR:
250 if ((ioflag & IO_SYNC) == 0)
251 panic("ffs_write: nonsync dir write");
252 break;
253 default:
254 panic("ffs_write: invalid v_type=%x", vp->v_type);
255 }
256
257 fs = ip->I_FS;
258 if (uio->uio_offset < 0 ||
259 (u_int64_t)uio->uio_offset + uio->uio_resid > fs->fs_maxfilesize)
260 return (EFBIG);
261 if (uio->uio_resid == 0)
262 return (0);
263
264 VOP_DEVBLOCKSIZE(ip->i_devvp, &devBlockSize);
265
266 /*
267 * Maybe this should be above the vnode op call, but so long as
268 * file servers have no limits, I don't think it matters.
269 */
270 p = uio->uio_procp;
271 if (vp->v_type == VREG && p &&
272 uio->uio_offset + uio->uio_resid >
273 p->p_rlimit[RLIMIT_FSIZE].rlim_cur) {
274 psignal(p, SIGXFSZ);
275 return (EFBIG);
276 }
277
278 resid = uio->uio_resid;
279 osize = ip->i_size;
280 flags = ioflag & IO_SYNC ? B_SYNC : 0;
281
282 if (UBCISVALID(vp)) {
283 off_t filesize;
284 off_t endofwrite;
285 off_t local_offset;
286 off_t head_offset;
287 int local_flags;
288 int first_block;
289 int fboff;
290 int fblk;
291 int loopcount;
292 int file_extended = 0;
293
294 endofwrite = uio->uio_offset + uio->uio_resid;
295
296 if (endofwrite > ip->i_size) {
297 filesize = endofwrite;
298 file_extended = 1;
299 } else
300 filesize = ip->i_size;
301
302 head_offset = ip->i_size;
303
304 /* Go ahead and allocate the block that are going to be written */
305 rsd = uio->uio_resid;
306 local_offset = uio->uio_offset;
307 local_flags = ioflag & IO_SYNC ? B_SYNC : 0;
308 local_flags |= B_NOBUFF;
309
310 first_block = 1;
311 fboff = 0;
312 fblk = 0;
313 loopcount = 0;
314
315 for (error = 0; rsd > 0;) {
316 blkalloc = 0;
317 lbn = lblkno(fs, local_offset);
318 blkoffset = blkoff(fs, local_offset);
319 xfersize = fs->fs_bsize - blkoffset;
320 if (first_block)
321 fboff = blkoffset;
322 if (rsd < xfersize)
323 xfersize = rsd;
324 if (fs->fs_bsize > xfersize)
325 local_flags |= B_CLRBUF;
326 else
327 local_flags &= ~B_CLRBUF;
328
329 /* Allocate block without reading into a buf */
330 error = ffs_balloc(ip,
331 lbn, blkoffset + xfersize, ap->a_cred,
332 &bp, local_flags, &blkalloc);
333 if (error)
334 break;
335 if (first_block) {
336 fblk = blkalloc;
337 first_block = 0;
338 }
339 loopcount++;
340
341 rsd -= xfersize;
342 local_offset += (off_t)xfersize;
343 if (local_offset > ip->i_size)
344 ip->i_size = local_offset;
345 }
346
347 if(error) {
348 save_error = error;
349 save_size = rsd;
350 uio->uio_resid -= rsd;
351 if (file_extended)
352 filesize -= rsd;
353 }
354
355 flags = ioflag & IO_SYNC ? IO_SYNC : 0;
356 /* flags |= IO_NOZEROVALID; */
357
358 if((error == 0) && fblk && fboff) {
359 if( fblk > fs->fs_bsize)
360 panic("ffs_balloc : allocated more than bsize(head)");
361 /* We need to zero out the head */
362 head_offset = uio->uio_offset - (off_t)fboff ;
363 flags |= IO_HEADZEROFILL;
364 /* flags &= ~IO_NOZEROVALID; */
365 }
366
367 if((error == 0) && blkalloc && ((blkalloc - xfersize) > 0)) {
368 /* We need to zero out the tail */
369 if( blkalloc > fs->fs_bsize)
370 panic("ffs_balloc : allocated more than bsize(tail)");
371 local_offset += (blkalloc - xfersize);
372 if (loopcount == 1) {
373 /* blkalloc is same as fblk; so no need to check again*/
374 local_offset -= fboff;
375 }
376 flags |= IO_TAILZEROFILL;
377 /* Freshly allocated block; bzero even if
378 * find a page
379 */
380 /* flags &= ~IO_NOZEROVALID; */
381 }
382 /*
383 * if the write starts beyond the current EOF then
384 * we we'll zero fill from the current EOF to where the write begins
385 */
386
387 error = cluster_write(vp, uio, osize, filesize, head_offset, local_offset, devBlockSize, flags);
388
389 if (uio->uio_offset > osize) {
390 if (error && ((ioflag & IO_UNIT)==0))
391 (void)VOP_TRUNCATE(vp, uio->uio_offset,
392 ioflag & IO_SYNC, ap->a_cred, uio->uio_procp);
393 ip->i_size = uio->uio_offset;
394 ubc_setsize(vp, (off_t)ip->i_size);
395 }
396 if(save_error) {
397 uio->uio_resid += save_size;
398 if(!error)
399 error = save_error;
400 }
401 ip->i_flag |= IN_CHANGE | IN_UPDATE;
402 } else {
403 flags = ioflag & IO_SYNC ? B_SYNC : 0;
404
405 for (error = 0; uio->uio_resid > 0;) {
406 lbn = lblkno(fs, uio->uio_offset);
407 blkoffset = blkoff(fs, uio->uio_offset);
408 xfersize = fs->fs_bsize - blkoffset;
409 if (uio->uio_resid < xfersize)
410 xfersize = uio->uio_resid;
411
412 if (fs->fs_bsize > xfersize)
413 flags |= B_CLRBUF;
414 else
415 flags &= ~B_CLRBUF;
416
417 error = ffs_balloc(ip,
418 lbn, blkoffset + xfersize, ap->a_cred, &bp, flags, 0);
419 if (error)
420 break;
421 if (uio->uio_offset + xfersize > ip->i_size) {
422 ip->i_size = uio->uio_offset + xfersize;
423
424 if (UBCISVALID(vp))
425 ubc_setsize(vp, (u_long)ip->i_size); /* XXX check errors */
426 }
427
428 size = BLKSIZE(fs, ip, lbn) - bp->b_resid;
429 if (size < xfersize)
430 xfersize = size;
431
432 error =
433 uiomove((char *)bp->b_data + blkoffset, (int)xfersize, uio);
434 #if REV_ENDIAN_FS
435 if (rev_endian && S_ISDIR(ip->i_mode)) {
436 byte_swap_dir_out((char *)bp->b_data + blkoffset, xfersize);
437 }
438 #endif /* REV_ENDIAN_FS */
439 if (ioflag & IO_SYNC)
440 (void)bwrite(bp);
441 else if (xfersize + blkoffset == fs->fs_bsize) {
442 bp->b_flags |= B_AGE;
443 bawrite(bp);
444 }
445 else
446 bdwrite(bp);
447 if (error || xfersize == 0)
448 break;
449 ip->i_flag |= IN_CHANGE | IN_UPDATE;
450 }
451 }
452 /*
453 * If we successfully wrote any data, and we are not the superuser
454 * we clear the setuid and setgid bits as a precaution against
455 * tampering.
456 */
457 if (resid > uio->uio_resid && ap->a_cred && ap->a_cred->cr_uid != 0)
458 ip->i_mode &= ~(ISUID | ISGID);
459 if (error) {
460 if (ioflag & IO_UNIT) {
461 (void)VOP_TRUNCATE(vp, osize,
462 ioflag & IO_SYNC, ap->a_cred, uio->uio_procp);
463 uio->uio_offset -= resid - uio->uio_resid;
464 uio->uio_resid = resid;
465 }
466 } else if (resid > uio->uio_resid && (ioflag & IO_SYNC))
467 error = VOP_UPDATE(vp, &time, &time, 1);
468 return (error);
469 }
470
471 /*
472 * Vnode op for page read.
473 */
474 /* ARGSUSED */
475 PGRD(ap)
476 struct vop_pgrd_args /* {
477 struct vnode *a_vp;
478 struct uio *a_uio;
479 struct ucred *a_cred;
480 } */ *ap;
481 {
482
483 #warning ufs_readwrite PGRD need to implement
484 return (EOPNOTSUPP);
485
486 }
487
488 /*
489 * Vnode op for page read.
490 */
491 /* ARGSUSED */
492 PGWR(ap)
493 struct vop_pgwr_args /* {
494 struct vnode *a_vp;
495 struct uio *a_uio;
496 struct ucred *a_cred;
497 memory_object_t a_pager;
498 vm_offset_t a_offset;
499 } */ *ap;
500 {
501
502 #warning ufs_readwrite PGWR need to implement
503 return (EOPNOTSUPP);
504
505 }
506
507 /*
508 * Vnode op for pagein.
509 * Similar to ffs_read()
510 */
511 /* ARGSUSED */
512 ffs_pagein(ap)
513 struct vop_pagein_args /* {
514 struct vnode *a_vp,
515 upl_t a_pl,
516 vm_offset_t a_pl_offset,
517 off_t a_f_offset,
518 size_t a_size,
519 struct ucred *a_cred,
520 int a_flags
521 } */ *ap;
522 {
523 register struct vnode *vp = ap->a_vp;
524 upl_t pl = ap->a_pl;
525 size_t size= ap->a_size;
526 off_t f_offset = ap->a_f_offset;
527 vm_offset_t pl_offset = ap->a_pl_offset;
528 int flags = ap->a_flags;
529 register struct inode *ip;
530 int devBlockSize=0;
531 int error;
532
533 ip = VTOI(vp);
534
535 /* check pageins for reg file only and ubc info is present*/
536 if (UBCINVALID(vp))
537 panic("ffs_pagein: Not a VREG: vp=%x", vp);
538 if (UBCINFOMISSING(vp))
539 panic("ffs_pagein: No mapping: vp=%x", vp);
540
541 #if DIAGNOSTIC
542 if (vp->v_type == VLNK) {
543 if ((int)ip->i_size < vp->v_mount->mnt_maxsymlinklen)
544 panic("%s: short symlink", "ffs_pagein");
545 } else if (vp->v_type != VREG && vp->v_type != VDIR)
546 panic("%s: type %d", "ffs_pagein", vp->v_type);
547 #endif
548
549 VOP_DEVBLOCKSIZE(ip->i_devvp, &devBlockSize);
550
551 error = cluster_pagein(vp, pl, pl_offset, f_offset, size,
552 (off_t)ip->i_size, devBlockSize, flags);
553 /* ip->i_flag |= IN_ACCESS; */
554 return (error);
555 }
556
557 /*
558 * Vnode op for pageout.
559 * Similar to ffs_write()
560 * make sure the buf is not in hash queue when you return
561 */
562 ffs_pageout(ap)
563 struct vop_pageout_args /* {
564 struct vnode *a_vp,
565 upl_t a_pl,
566 vm_offset_t a_pl_offset,
567 off_t a_f_offset,
568 size_t a_size,
569 struct ucred *a_cred,
570 int a_flags
571 } */ *ap;
572 {
573 register struct vnode *vp = ap->a_vp;
574 upl_t pl = ap->a_pl;
575 size_t size= ap->a_size;
576 off_t f_offset = ap->a_f_offset;
577 vm_offset_t pl_offset = ap->a_pl_offset;
578 int flags = ap->a_flags;
579 register struct inode *ip;
580 register FS *fs;
581 int error ;
582 int devBlockSize=0;
583 size_t xfer_size = 0;
584 int local_flags=0;
585 off_t local_offset;
586 int resid, blkoffset;
587 size_t xsize, lsize;
588 daddr_t lbn;
589 int save_error =0, save_size=0;
590 vm_offset_t lupl_offset;
591 int nocommit = flags & UPL_NOCOMMIT;
592 struct buf *bp;
593
594 ip = VTOI(vp);
595
596 /* check pageouts for reg file only and ubc info is present*/
597 if (UBCINVALID(vp))
598 panic("ffs_pageout: Not a VREG: vp=%x", vp);
599 if (UBCINFOMISSING(vp))
600 panic("ffs_pageout: No mapping: vp=%x", vp);
601
602 if (vp->v_mount->mnt_flag & MNT_RDONLY) {
603 if (!nocommit)
604 ubc_upl_abort_range(pl, pl_offset, size,
605 UPL_ABORT_FREE_ON_EMPTY);
606 return (EROFS);
607 }
608 fs = ip->I_FS;
609
610 if (f_offset < 0 || f_offset >= ip->i_size) {
611 if (!nocommit)
612 ubc_upl_abort_range(pl, pl_offset, size,
613 UPL_ABORT_FREE_ON_EMPTY);
614 return (EINVAL);
615 }
616
617 /*
618 * once we enable multi-page pageouts we will
619 * need to make sure we abort any pages in the upl
620 * that we don't issue an I/O for
621 */
622 if (f_offset + size > ip->i_size)
623 xfer_size = ip->i_size - f_offset;
624 else
625 xfer_size = size;
626
627 VOP_DEVBLOCKSIZE(ip->i_devvp, &devBlockSize);
628
629 if (xfer_size & (PAGE_SIZE - 1)) {
630 /* if not a multiple of page size
631 * then round up to be a multiple
632 * the physical disk block size
633 */
634 xfer_size = (xfer_size + (devBlockSize - 1)) & ~(devBlockSize - 1);
635 }
636
637 /*
638 * once the block allocation is moved to ufs_cmap
639 * we can remove all the size and offset checks above
640 * cluster_pageout does all of this now
641 * we need to continue to do it here so as not to
642 * allocate blocks that aren't going to be used because
643 * of a bogus parameter being passed in
644 */
645 local_flags = 0;
646 resid = xfer_size;
647 local_offset = f_offset;
648 for (error = 0; resid > 0;) {
649 lbn = lblkno(fs, local_offset);
650 blkoffset = blkoff(fs, local_offset);
651 xsize = fs->fs_bsize - blkoffset;
652 if (resid < xsize)
653 xsize = resid;
654 /* Allocate block without reading into a buf */
655 error = ffs_blkalloc(ip,
656 lbn, blkoffset + xsize, ap->a_cred,
657 local_flags);
658 if (error)
659 break;
660 resid -= xsize;
661 local_offset += (off_t)xsize;
662 }
663
664 if (error) {
665 save_size = resid;
666 save_error = error;
667 xfer_size -= save_size;
668 }
669
670
671 error = cluster_pageout(vp, pl, pl_offset, f_offset, round_page(xfer_size), ip->i_size, devBlockSize, flags);
672
673 if(save_error) {
674 lupl_offset = size - save_size;
675 resid = round_page(save_size);
676 if (!nocommit)
677 ubc_upl_abort_range(pl, lupl_offset, resid,
678 UPL_ABORT_FREE_ON_EMPTY);
679 if(!error)
680 error= save_error;
681 }
682 return (error);
683 }