fdopen.c revision 1.14
11.14Sagc/*	$NetBSD: fdopen.c,v 1.14 2003/08/07 16:43:22 agc 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.14Sagc * 3. Neither the name of the University nor the names of its contributors
191.1Scgd *    may be used to endorse or promote products derived from this software
201.1Scgd *    without specific prior written permission.
211.1Scgd *
221.1Scgd * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
231.1Scgd * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
241.1Scgd * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
251.1Scgd * ARE DISCLAIMED.  IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
261.1Scgd * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
271.1Scgd * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
281.1Scgd * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
291.1Scgd * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
301.1Scgd * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
311.1Scgd * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
321.1Scgd * SUCH DAMAGE.
331.1Scgd */
341.1Scgd
351.6Schristos#include <sys/cdefs.h>
361.1Scgd#if defined(LIBC_SCCS) && !defined(lint)
371.5Sjtc#if 0
381.5Sjtcstatic char sccsid[] = "@(#)fdopen.c	8.1 (Berkeley) 6/4/93";
391.6Schristos#else
401.14Sagc__RCSID("$NetBSD: fdopen.c,v 1.14 2003/08/07 16:43:22 agc Exp $");
411.5Sjtc#endif
421.1Scgd#endif /* LIBC_SCCS and not lint */
431.1Scgd
441.8Skleink#include "namespace.h"
451.1Scgd#include <sys/types.h>
461.9Slukem
471.9Slukem#include <assert.h>
481.9Slukem#include <errno.h>
491.1Scgd#include <fcntl.h>
501.1Scgd#include <unistd.h>
511.1Scgd#include <stdio.h>
521.9Slukem
531.13Sthorpej#include "reentrant.h"
541.1Scgd#include "local.h"
551.8Skleink
561.8Skleink#ifdef __weak_alias
571.11Smycroft__weak_alias(fdopen,_fdopen)
581.8Skleink#endif
591.1Scgd
601.1ScgdFILE *
611.1Scgdfdopen(fd, mode)
621.1Scgd	int fd;
631.1Scgd	const char *mode;
641.1Scgd{
651.7Sperry	FILE *fp;
661.1Scgd	int flags, oflags, fdflags, tmp;
671.9Slukem
681.9Slukem	_DIAGASSERT(fd != -1);
691.1Scgd
701.1Scgd	if ((flags = __sflags(mode, &oflags)) == 0)
711.1Scgd		return (NULL);
721.1Scgd
731.1Scgd	/* Make sure the mode the user wants is a subset of the actual mode. */
741.1Scgd	if ((fdflags = fcntl(fd, F_GETFL, 0)) < 0)
751.1Scgd		return (NULL);
761.1Scgd	tmp = fdflags & O_ACCMODE;
771.1Scgd	if (tmp != O_RDWR && (tmp != (oflags & O_ACCMODE))) {
781.1Scgd		errno = EINVAL;
791.1Scgd		return (NULL);
801.12Schristos	}
811.12Schristos
821.12Schristos	if (oflags & O_NONBLOCK) {
831.12Schristos		struct stat st;
841.12Schristos		if (fstat(fd, &st) == -1) {
851.12Schristos			return (NULL);
861.12Schristos		}
871.12Schristos		if (!S_ISREG(st.st_mode)) {
881.12Schristos			errno = EFTYPE;
891.12Schristos			return (NULL);
901.12Schristos		}
911.1Scgd	}
921.1Scgd
931.1Scgd	if ((fp = __sfp()) == NULL)
941.1Scgd		return (NULL);
951.1Scgd	fp->_flags = flags;
961.1Scgd	/*
971.1Scgd	 * If opened for appending, but underlying descriptor does not have
981.1Scgd	 * O_APPEND bit set, assert __SAPP so that __swrite() will lseek to
991.1Scgd	 * end before each write.
1001.1Scgd	 */
1011.1Scgd	if ((oflags & O_APPEND) && !(fdflags & O_APPEND))
1021.1Scgd		fp->_flags |= __SAPP;
1031.1Scgd	fp->_file = fd;
1041.1Scgd	fp->_cookie = fp;
1051.1Scgd	fp->_read = __sread;
1061.1Scgd	fp->_write = __swrite;
1071.1Scgd	fp->_seek = __sseek;
1081.1Scgd	fp->_close = __sclose;
1091.1Scgd	return (fp);
1101.1Scgd}
111