138451Smsmith/*	$NetBSD: read.c,v 1.8 1997/01/22 00:38:12 cgd Exp $	*/
238451Smsmith
338451Smsmith/*-
438451Smsmith * Copyright (c) 1993
538451Smsmith *	The Regents of the University of California.  All rights reserved.
638451Smsmith *
738451Smsmith * This code is derived from software contributed to Berkeley by
838451Smsmith * The Mach Operating System project at Carnegie-Mellon University.
938451Smsmith *
1038451Smsmith * Redistribution and use in source and binary forms, with or without
1138451Smsmith * modification, are permitted provided that the following conditions
1238451Smsmith * are met:
1338451Smsmith * 1. Redistributions of source code must retain the above copyright
1438451Smsmith *    notice, this list of conditions and the following disclaimer.
1538451Smsmith * 2. Redistributions in binary form must reproduce the above copyright
1638451Smsmith *    notice, this list of conditions and the following disclaimer in the
1738451Smsmith *    documentation and/or other materials provided with the distribution.
1838451Smsmith * 4. Neither the name of the University nor the names of its contributors
1938451Smsmith *    may be used to endorse or promote products derived from this software
2038451Smsmith *    without specific prior written permission.
2138451Smsmith *
2238451Smsmith * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
2338451Smsmith * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
2438451Smsmith * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
2538451Smsmith * ARE DISCLAIMED.  IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
2638451Smsmith * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
2738451Smsmith * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
2838451Smsmith * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
2938451Smsmith * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
3038451Smsmith * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
3138451Smsmith * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
3238451Smsmith * SUCH DAMAGE.
3338451Smsmith *
3438451Smsmith *	@(#)read.c	8.1 (Berkeley) 6/11/93
3538451Smsmith *
3638451Smsmith *
3738451Smsmith * Copyright (c) 1989, 1990, 1991 Carnegie Mellon University
3838451Smsmith * All Rights Reserved.
3938451Smsmith *
4038451Smsmith * Author: Alessandro Forin
4138451Smsmith *
4238451Smsmith * Permission to use, copy, modify and distribute this software and its
4338451Smsmith * documentation is hereby granted, provided that both the copyright
4438451Smsmith * notice and this permission notice appear in all copies of the
4538451Smsmith * software, derivative works or modified versions, and any portions
4638451Smsmith * thereof, and that both notices appear in supporting documentation.
4738451Smsmith *
4838451Smsmith * CARNEGIE MELLON ALLOWS FREE USE OF THIS SOFTWARE IN ITS "AS IS"
4938451Smsmith * CONDITION.  CARNEGIE MELLON DISCLAIMS ANY LIABILITY OF ANY KIND FOR
5038451Smsmith * ANY DAMAGES WHATSOEVER RESULTING FROM THE USE OF THIS SOFTWARE.
5138451Smsmith *
5238451Smsmith * Carnegie Mellon requests users of this software to return to
5338451Smsmith *
5438451Smsmith *  Software Distribution Coordinator  or  Software.Distribution@CS.CMU.EDU
5538451Smsmith *  School of Computer Science
5638451Smsmith *  Carnegie Mellon University
5738451Smsmith *  Pittsburgh PA 15213-3890
5838451Smsmith *
5938451Smsmith * any improvements or extensions that they make and grant Carnegie the
6038451Smsmith * rights to redistribute these changes.
6138451Smsmith */
6238451Smsmith
6384221Sdillon#include <sys/cdefs.h>
6484221Sdillon__FBSDID("$FreeBSD$");
6584221Sdillon
6638451Smsmith#include <sys/param.h>
6738451Smsmith#include "stand.h"
6838451Smsmith
6938451Smsmithssize_t
7065470Smsmithread(int fd, void *dest, size_t bcount)
7138451Smsmith{
7265470Smsmith    struct open_file	*f = &files[fd];
7365470Smsmith    size_t		resid;
7438451Smsmith
7565470Smsmith    if ((unsigned)fd >= SOPEN_MAX || !(f->f_flags & F_READ)) {
7665470Smsmith	errno = EBADF;
7765470Smsmith	return (-1);
7865470Smsmith    }
7965470Smsmith    if (f->f_flags & F_RAW) {
8065470Smsmith	twiddle();
8165470Smsmith	errno = (f->f_dev->dv_strategy)(f->f_devdata, F_READ,
8265470Smsmith					btodb(f->f_offset), bcount, dest, &resid);
8365470Smsmith	if (errno)
8465470Smsmith	    return (-1);
8565470Smsmith	f->f_offset += resid;
8665470Smsmith	return (resid);
8765470Smsmith    }
8865470Smsmith
8965470Smsmith    /*
9065470Smsmith     * Optimise reads from regular files using a readahead buffer.
9165470Smsmith     * If the request can't be satisfied from the current buffer contents,
9265470Smsmith     * check to see if it should be bypassed, or refill the buffer and complete
9365470Smsmith     * the request.
9465470Smsmith     */
9565470Smsmith    resid = bcount;
9665470Smsmith    for (;;) {
9765470Smsmith	size_t	ccount, cresid;
9865470Smsmith	/* how much can we supply? */
9965470Smsmith	ccount = imin(f->f_ralen, resid);
10065470Smsmith	if (ccount > 0) {
10165470Smsmith	    bcopy(f->f_rabuf + f->f_raoffset, dest, ccount);
10265470Smsmith	    f->f_raoffset += ccount;
10365470Smsmith	    f->f_ralen -= ccount;
10465470Smsmith	    resid -= ccount;
10565470Smsmith	    if (resid == 0)
10665470Smsmith		return(bcount);
107136093Sstefanf	    dest = (char *)dest + ccount;
10838451Smsmith	}
10965470Smsmith
11065470Smsmith	/* will filling the readahead buffer again not help? */
11165470Smsmith	if (resid >= SOPEN_RASIZE) {
11265470Smsmith	    /* bypass the rest of the request and leave the buffer empty */
11365470Smsmith	    if ((errno = (f->f_ops->fo_read)(f, dest, resid, &cresid)))
114124811Sjhb		    return (-1);
11565470Smsmith	    return(bcount - cresid);
11638451Smsmith	}
11765470Smsmith
11865470Smsmith	/* fetch more data */
11965470Smsmith	if ((errno = (f->f_ops->fo_read)(f, f->f_rabuf, SOPEN_RASIZE, &cresid)))
120124811Sjhb	    return (-1);
12165470Smsmith	f->f_raoffset = 0;
12265470Smsmith	f->f_ralen = SOPEN_RASIZE - cresid;
12365470Smsmith	/* no more data, return what we had */
12465470Smsmith	if (f->f_ralen == 0)
12565470Smsmith	    return(bcount - resid);
12665470Smsmith    }
12738451Smsmith}
128