fdopen.c revision 1.12
11.12Schristos/*	$NetBSD: fdopen.c,v 1.12 2000/11/29 15:36:42 christos Exp $	*/
21.5Sjtc
31.1Scgd/*-
41.5Sjtc * Copyright (c) 1990, 1993
51.5Sjtc *	The Regents of the University of California.  All rights reserved.
61.1Scgd *
71.1Scgd * This code is derived from software contributed to Berkeley by
81.1Scgd * Chris Torek.
91.1Scgd *
101.1Scgd * Redistribution and use in source and binary forms, with or without
111.1Scgd * modification, are permitted provided that the following conditions
121.1Scgd * are met:
131.1Scgd * 1. Redistributions of source code must retain the above copyright
141.1Scgd *    notice, this list of conditions and the following disclaimer.
151.1Scgd * 2. Redistributions in binary form must reproduce the above copyright
161.1Scgd *    notice, this list of conditions and the following disclaimer in the
171.1Scgd *    documentation and/or other materials provided with the distribution.
181.1Scgd * 3. All advertising materials mentioning features or use of this software
191.1Scgd *    must display the following acknowledgement:
201.1Scgd *	This product includes software developed by the University of
211.1Scgd *	California, Berkeley and its contributors.
221.1Scgd * 4. Neither the name of the University nor the names of its contributors
231.1Scgd *    may be used to endorse or promote products derived from this software
241.1Scgd *    without specific prior written permission.
251.1Scgd *
261.1Scgd * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
271.1Scgd * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
281.1Scgd * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
291.1Scgd * ARE DISCLAIMED.  IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
301.1Scgd * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
311.1Scgd * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
321.1Scgd * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
331.1Scgd * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
341.1Scgd * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
351.1Scgd * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
361.1Scgd * SUCH DAMAGE.
371.1Scgd */
381.1Scgd
391.6Schristos#include <sys/cdefs.h>
401.1Scgd#if defined(LIBC_SCCS) && !defined(lint)
411.5Sjtc#if 0
421.5Sjtcstatic char sccsid[] = "@(#)fdopen.c	8.1 (Berkeley) 6/4/93";
431.6Schristos#else
441.12Schristos__RCSID("$NetBSD: fdopen.c,v 1.12 2000/11/29 15:36:42 christos Exp $");
451.5Sjtc#endif
461.1Scgd#endif /* LIBC_SCCS and not lint */
471.1Scgd
481.8Skleink#include "namespace.h"
491.1Scgd#include <sys/types.h>
501.9Slukem
511.9Slukem#include <assert.h>
521.9Slukem#include <errno.h>
531.1Scgd#include <fcntl.h>
541.1Scgd#include <unistd.h>
551.1Scgd#include <stdio.h>
561.9Slukem
571.1Scgd#include "local.h"
581.8Skleink
591.8Skleink#ifdef __weak_alias
601.11Smycroft__weak_alias(fdopen,_fdopen)
611.8Skleink#endif
621.1Scgd
631.1ScgdFILE *
641.1Scgdfdopen(fd, mode)
651.1Scgd	int fd;
661.1Scgd	const char *mode;
671.1Scgd{
681.7Sperry	FILE *fp;
691.1Scgd	int flags, oflags, fdflags, tmp;
701.9Slukem
711.9Slukem	_DIAGASSERT(fd != -1);
721.1Scgd
731.1Scgd	if ((flags = __sflags(mode, &oflags)) == 0)
741.1Scgd		return (NULL);
751.1Scgd
761.1Scgd	/* Make sure the mode the user wants is a subset of the actual mode. */
771.1Scgd	if ((fdflags = fcntl(fd, F_GETFL, 0)) < 0)
781.1Scgd		return (NULL);
791.1Scgd	tmp = fdflags & O_ACCMODE;
801.1Scgd	if (tmp != O_RDWR && (tmp != (oflags & O_ACCMODE))) {
811.1Scgd		errno = EINVAL;
821.1Scgd		return (NULL);
831.12Schristos	}
841.12Schristos
851.12Schristos	if (oflags & O_NONBLOCK) {
861.12Schristos		struct stat st;
871.12Schristos		if (fstat(fd, &st) == -1) {
881.12Schristos			return (NULL);
891.12Schristos		}
901.12Schristos		if (!S_ISREG(st.st_mode)) {
911.12Schristos			errno = EFTYPE;
921.12Schristos			return (NULL);
931.12Schristos		}
941.1Scgd	}
951.1Scgd
961.1Scgd	if ((fp = __sfp()) == NULL)
971.1Scgd		return (NULL);
981.1Scgd	fp->_flags = flags;
991.1Scgd	/*
1001.1Scgd	 * If opened for appending, but underlying descriptor does not have
1011.1Scgd	 * O_APPEND bit set, assert __SAPP so that __swrite() will lseek to
1021.1Scgd	 * end before each write.
1031.1Scgd	 */
1041.1Scgd	if ((oflags & O_APPEND) && !(fdflags & O_APPEND))
1051.1Scgd		fp->_flags |= __SAPP;
1061.1Scgd	fp->_file = fd;
1071.1Scgd	fp->_cookie = fp;
1081.1Scgd	fp->_read = __sread;
1091.1Scgd	fp->_write = __swrite;
1101.1Scgd	fp->_seek = __sseek;
1111.1Scgd	fp->_close = __sclose;
1121.1Scgd	return (fp);
1131.1Scgd}
114