/* $NetBSD: lseek.c,v 1.4 1997/01/22 00:38:10 cgd Exp $ */12/*-3* Copyright (c) 19934* The Regents of the University of California. All rights reserved.5*6* This code is derived from software contributed to Berkeley by7* The Mach Operating System project at Carnegie-Mellon University.8*9* Redistribution and use in source and binary forms, with or without10* modification, are permitted provided that the following conditions11* are met:12* 1. Redistributions of source code must retain the above copyright13* notice, this list of conditions and the following disclaimer.14* 2. Redistributions in binary form must reproduce the above copyright15* notice, this list of conditions and the following disclaimer in the16* documentation and/or other materials provided with the distribution.17* 3. Neither the name of the University nor the names of its contributors18* may be used to endorse or promote products derived from this software19* without specific prior written permission.20*21* THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND22* ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE23* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE24* ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE25* FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL26* DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS27* OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)28* HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT29* LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY30* OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF31* SUCH DAMAGE.32*33*34* Copyright (c) 1989, 1990, 1991 Carnegie Mellon University35* All Rights Reserved.36*37* Author: Alessandro Forin38*39* Permission to use, copy, modify and distribute this software and its40* documentation is hereby granted, provided that both the copyright41* notice and this permission notice appear in all copies of the42* software, derivative works or modified versions, and any portions43* thereof, and that both notices appear in supporting documentation.44*45* CARNEGIE MELLON ALLOWS FREE USE OF THIS SOFTWARE IN ITS "AS IS"46* CONDITION. CARNEGIE MELLON DISCLAIMS ANY LIABILITY OF ANY KIND FOR47* ANY DAMAGES WHATSOEVER RESULTING FROM THE USE OF THIS SOFTWARE.48*49* Carnegie Mellon requests users of this software to return to50*51* Software Distribution Coordinator or [email protected]52* School of Computer Science53* Carnegie Mellon University54* Pittsburgh PA 15213-389055*56* any improvements or extensions that they make and grant Carnegie the57* rights to redistribute these changes.58*/5960#include "stand.h"6162off_t63lseek(int fd, off_t offset, int where)64{65off_t bufpos, filepos, target;66struct open_file *f;6768f = fd2open_file(fd);69if (f == NULL || f->f_flags == 0) {70errno = EBADF;71return (-1);72}7374if (f->f_flags & F_RAW) {75/*76* On RAW devices, update internal offset.77*/78switch (where) {79case SEEK_SET:80f->f_offset = offset;81break;82case SEEK_CUR:83f->f_offset += offset;84break;85default:86errno = EOFFSET;87return (-1);88}89return (f->f_offset);90}9192/*93* If there is some unconsumed data in the readahead buffer and it94* contains the desired offset, simply adjust the buffer offset and95* length. We don't bother with SEEK_END here, since the code to96* handle it would fail in the same cases where the non-readahead97* code fails (namely, for streams which cannot seek backward and whose98* size isn't known in advance).99*/100if (f->f_ralen != 0 && where != SEEK_END) {101filepos = (f->f_ops->fo_seek)(f, 0, SEEK_CUR);102if (filepos == -1)103return (-1);104bufpos = filepos - f->f_ralen;105switch (where) {106case SEEK_SET:107target = offset;108break;109case SEEK_CUR:110target = bufpos + offset;111break;112default:113errno = EINVAL;114return (-1);115}116if (bufpos <= target && target < filepos) {117f->f_raoffset += target - bufpos;118f->f_ralen -= target - bufpos;119return (target);120}121}122123/*124* If this is a relative seek, we need to correct the offset for125* bytes that we have already read but the caller doesn't know126* about.127*/128if (where == SEEK_CUR)129offset -= f->f_ralen;130131/*132* Invalidate the readahead buffer.133*/134f->f_ralen = 0;135136return (f->f_ops->fo_seek)(f, offset, where);137}138139140