1*4a5d661aSToomas Soome /* $NetBSD: lseek.c,v 1.4 1997/01/22 00:38:10 cgd Exp $ */ 2*4a5d661aSToomas Soome 3*4a5d661aSToomas Soome /*- 4*4a5d661aSToomas Soome * Copyright (c) 1993 5*4a5d661aSToomas Soome * The Regents of the University of California. All rights reserved. 6*4a5d661aSToomas Soome * 7*4a5d661aSToomas Soome * This code is derived from software contributed to Berkeley by 8*4a5d661aSToomas Soome * The Mach Operating System project at Carnegie-Mellon University. 9*4a5d661aSToomas Soome * 10*4a5d661aSToomas Soome * Redistribution and use in source and binary forms, with or without 11*4a5d661aSToomas Soome * modification, are permitted provided that the following conditions 12*4a5d661aSToomas Soome * are met: 13*4a5d661aSToomas Soome * 1. Redistributions of source code must retain the above copyright 14*4a5d661aSToomas Soome * notice, this list of conditions and the following disclaimer. 15*4a5d661aSToomas Soome * 2. Redistributions in binary form must reproduce the above copyright 16*4a5d661aSToomas Soome * notice, this list of conditions and the following disclaimer in the 17*4a5d661aSToomas Soome * documentation and/or other materials provided with the distribution. 18*4a5d661aSToomas Soome * 4. Neither the name of the University nor the names of its contributors 19*4a5d661aSToomas Soome * may be used to endorse or promote products derived from this software 20*4a5d661aSToomas Soome * without specific prior written permission. 21*4a5d661aSToomas Soome * 22*4a5d661aSToomas Soome * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND 23*4a5d661aSToomas Soome * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE 24*4a5d661aSToomas Soome * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE 25*4a5d661aSToomas Soome * ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE 26*4a5d661aSToomas Soome * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL 27*4a5d661aSToomas Soome * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS 28*4a5d661aSToomas Soome * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) 29*4a5d661aSToomas Soome * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT 30*4a5d661aSToomas Soome * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY 31*4a5d661aSToomas Soome * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF 32*4a5d661aSToomas Soome * SUCH DAMAGE. 33*4a5d661aSToomas Soome * 34*4a5d661aSToomas Soome * @(#)lseek.c 8.1 (Berkeley) 6/11/93 35*4a5d661aSToomas Soome * 36*4a5d661aSToomas Soome * 37*4a5d661aSToomas Soome * Copyright (c) 1989, 1990, 1991 Carnegie Mellon University 38*4a5d661aSToomas Soome * All Rights Reserved. 39*4a5d661aSToomas Soome * 40*4a5d661aSToomas Soome * Author: Alessandro Forin 41*4a5d661aSToomas Soome * 42*4a5d661aSToomas Soome * Permission to use, copy, modify and distribute this software and its 43*4a5d661aSToomas Soome * documentation is hereby granted, provided that both the copyright 44*4a5d661aSToomas Soome * notice and this permission notice appear in all copies of the 45*4a5d661aSToomas Soome * software, derivative works or modified versions, and any portions 46*4a5d661aSToomas Soome * thereof, and that both notices appear in supporting documentation. 47*4a5d661aSToomas Soome * 48*4a5d661aSToomas Soome * CARNEGIE MELLON ALLOWS FREE USE OF THIS SOFTWARE IN ITS "AS IS" 49*4a5d661aSToomas Soome * CONDITION. CARNEGIE MELLON DISCLAIMS ANY LIABILITY OF ANY KIND FOR 50*4a5d661aSToomas Soome * ANY DAMAGES WHATSOEVER RESULTING FROM THE USE OF THIS SOFTWARE. 51*4a5d661aSToomas Soome * 52*4a5d661aSToomas Soome * Carnegie Mellon requests users of this software to return to 53*4a5d661aSToomas Soome * 54*4a5d661aSToomas Soome * Software Distribution Coordinator or Software.Distribution@CS.CMU.EDU 55*4a5d661aSToomas Soome * School of Computer Science 56*4a5d661aSToomas Soome * Carnegie Mellon University 57*4a5d661aSToomas Soome * Pittsburgh PA 15213-3890 58*4a5d661aSToomas Soome * 59*4a5d661aSToomas Soome * any improvements or extensions that they make and grant Carnegie the 60*4a5d661aSToomas Soome * rights to redistribute these changes. 61*4a5d661aSToomas Soome */ 62*4a5d661aSToomas Soome 63*4a5d661aSToomas Soome #include <sys/cdefs.h> 64*4a5d661aSToomas Soome __FBSDID("$FreeBSD$"); 65*4a5d661aSToomas Soome 66*4a5d661aSToomas Soome #include "stand.h" 67*4a5d661aSToomas Soome 68*4a5d661aSToomas Soome off_t 69*4a5d661aSToomas Soome lseek(int fd, off_t offset, int where) 70*4a5d661aSToomas Soome { 71*4a5d661aSToomas Soome off_t bufpos, filepos, target; 72*4a5d661aSToomas Soome struct open_file *f = &files[fd]; 73*4a5d661aSToomas Soome 74*4a5d661aSToomas Soome if ((unsigned)fd >= SOPEN_MAX || f->f_flags == 0) { 75*4a5d661aSToomas Soome errno = EBADF; 76*4a5d661aSToomas Soome return (-1); 77*4a5d661aSToomas Soome } 78*4a5d661aSToomas Soome 79*4a5d661aSToomas Soome if (f->f_flags & F_RAW) { 80*4a5d661aSToomas Soome /* 81*4a5d661aSToomas Soome * On RAW devices, update internal offset. 82*4a5d661aSToomas Soome */ 83*4a5d661aSToomas Soome switch (where) { 84*4a5d661aSToomas Soome case SEEK_SET: 85*4a5d661aSToomas Soome f->f_offset = offset; 86*4a5d661aSToomas Soome break; 87*4a5d661aSToomas Soome case SEEK_CUR: 88*4a5d661aSToomas Soome f->f_offset += offset; 89*4a5d661aSToomas Soome break; 90*4a5d661aSToomas Soome case SEEK_END: 91*4a5d661aSToomas Soome default: 92*4a5d661aSToomas Soome errno = EOFFSET; 93*4a5d661aSToomas Soome return (-1); 94*4a5d661aSToomas Soome } 95*4a5d661aSToomas Soome return (f->f_offset); 96*4a5d661aSToomas Soome } 97*4a5d661aSToomas Soome 98*4a5d661aSToomas Soome /* 99*4a5d661aSToomas Soome * If there is some unconsumed data in the readahead buffer and it 100*4a5d661aSToomas Soome * contains the desired offset, simply adjust the buffer offset and 101*4a5d661aSToomas Soome * length. We don't bother with SEEK_END here, since the code to 102*4a5d661aSToomas Soome * handle it would fail in the same cases where the non-readahead 103*4a5d661aSToomas Soome * code fails (namely, for streams which cannot seek backward and whose 104*4a5d661aSToomas Soome * size isn't known in advance). 105*4a5d661aSToomas Soome */ 106*4a5d661aSToomas Soome if (f->f_ralen != 0 && where != SEEK_END) { 107*4a5d661aSToomas Soome if ((filepos = (f->f_ops->fo_seek)(f, (off_t)0, SEEK_CUR)) == -1) 108*4a5d661aSToomas Soome return (-1); 109*4a5d661aSToomas Soome bufpos = filepos - f->f_ralen; 110*4a5d661aSToomas Soome switch (where) { 111*4a5d661aSToomas Soome case SEEK_SET: 112*4a5d661aSToomas Soome target = offset; 113*4a5d661aSToomas Soome break; 114*4a5d661aSToomas Soome case SEEK_CUR: 115*4a5d661aSToomas Soome target = bufpos + offset; 116*4a5d661aSToomas Soome break; 117*4a5d661aSToomas Soome default: 118*4a5d661aSToomas Soome errno = EINVAL; 119*4a5d661aSToomas Soome return (-1); 120*4a5d661aSToomas Soome } 121*4a5d661aSToomas Soome if (bufpos <= target && target < filepos) { 122*4a5d661aSToomas Soome f->f_raoffset += target - bufpos; 123*4a5d661aSToomas Soome f->f_ralen -= target - bufpos; 124*4a5d661aSToomas Soome return (target); 125*4a5d661aSToomas Soome } 126*4a5d661aSToomas Soome } 127*4a5d661aSToomas Soome 128*4a5d661aSToomas Soome /* 129*4a5d661aSToomas Soome * If this is a relative seek, we need to correct the offset for 130*4a5d661aSToomas Soome * bytes that we have already read but the caller doesn't know 131*4a5d661aSToomas Soome * about. 132*4a5d661aSToomas Soome */ 133*4a5d661aSToomas Soome if (where == SEEK_CUR) 134*4a5d661aSToomas Soome offset -= f->f_ralen; 135*4a5d661aSToomas Soome 136*4a5d661aSToomas Soome /* 137*4a5d661aSToomas Soome * Invalidate the readahead buffer. 138*4a5d661aSToomas Soome */ 139*4a5d661aSToomas Soome f->f_ralen = 0; 140*4a5d661aSToomas Soome 141*4a5d661aSToomas Soome return (f->f_ops->fo_seek)(f, offset, where); 142*4a5d661aSToomas Soome } 143