 a3bda44a41
			
		
	
	
		a3bda44a41
		
	
	
	
	
		
			
			. ignore ESPIPE error from lseek() in fflush() on read streams (because fifo's aren't detected when stdin, stdout or stderr)
		
			
				
	
	
		
			130 lines
		
	
	
		
			3.0 KiB
		
	
	
	
		
			C
		
	
	
		
			Executable File
		
	
	
	
	
			
		
		
	
	
			130 lines
		
	
	
		
			3.0 KiB
		
	
	
	
		
			C
		
	
	
		
			Executable File
		
	
	
	
	
| /*
 | |
|  * fopen.c - open a stream
 | |
|  */
 | |
| /* $Header$ */
 | |
| 
 | |
| #if	defined(_POSIX_SOURCE)
 | |
| #include	<sys/types.h>
 | |
| #endif
 | |
| #include	<stdio.h>
 | |
| #include	<stdlib.h>
 | |
| #include	"loc_incl.h"
 | |
| #include	<sys/stat.h>
 | |
| 
 | |
| #define	PMODE		0666
 | |
| 
 | |
| /* The next 3 defines are true in all UNIX systems known to me.
 | |
|  */
 | |
| #define	O_RDONLY	0
 | |
| #define	O_WRONLY	1
 | |
| #define	O_RDWR		2
 | |
| 
 | |
| /* Since the O_CREAT flag is not available on all systems, we can't get it
 | |
|  * from the standard library. Furthermore, even if we know that <fcntl.h>
 | |
|  * contains such a flag, it's not sure whether it can be used, since we
 | |
|  * might be cross-compiling for another system, which may use an entirely
 | |
|  * different value for O_CREAT (or not support such a mode). The safest
 | |
|  * thing is to just use the Version 7 semantics for open, and use creat()
 | |
|  * whenever necessary.
 | |
|  *
 | |
|  * Another problem is O_APPEND, for which the same holds. When "a"
 | |
|  * open-mode is used, an lseek() to the end is done before every write()
 | |
|  * system-call.
 | |
|  *
 | |
|  * The O_CREAT, O_TRUNC and O_APPEND given here, are only for convenience.
 | |
|  * They are not passed to open(), so the values don't have to match a value
 | |
|  * from the real world. It is enough when they are unique.
 | |
|  */
 | |
| #define	O_CREAT		0x010
 | |
| #define	O_TRUNC		0x020
 | |
| #define	O_APPEND	0x040
 | |
| 
 | |
| int _open(const char *path, int flags);
 | |
| int _creat(const char *path, _mnx_Mode_t mode);
 | |
| int _close(int d);
 | |
| 
 | |
| FILE *
 | |
| fopen(const char *name, const char *mode)
 | |
| {
 | |
| 	register int i;
 | |
| 	int rwmode = 0, rwflags = 0;
 | |
| 	FILE *stream;
 | |
| 	struct stat st;
 | |
| 	int fd, flags = 0;
 | |
| 
 | |
| 	for (i = 0; __iotab[i] != 0 ; i++) 
 | |
| 		if ( i >= FOPEN_MAX-1 )
 | |
| 			return (FILE *)NULL;
 | |
| 
 | |
| 	switch(*mode++) {
 | |
| 	case 'r':
 | |
| 		flags |= _IOREAD | _IOREADING;	
 | |
| 		rwmode = O_RDONLY;
 | |
| 		break;
 | |
| 	case 'w':
 | |
| 		flags |= _IOWRITE | _IOWRITING;
 | |
| 		rwmode = O_WRONLY;
 | |
| 		rwflags = O_CREAT | O_TRUNC;
 | |
| 		break;
 | |
| 	case 'a': 
 | |
| 		flags |= _IOWRITE | _IOWRITING | _IOAPPEND;
 | |
| 		rwmode = O_WRONLY;
 | |
| 		rwflags |= O_APPEND | O_CREAT;
 | |
| 		break;         
 | |
| 	default:
 | |
| 		return (FILE *)NULL;
 | |
| 	}
 | |
| 
 | |
| 	while (*mode) {
 | |
| 		switch(*mode++) {
 | |
| 		case 'b':
 | |
| 			continue;
 | |
| 		case '+':
 | |
| 			rwmode = O_RDWR;
 | |
| 			flags |= _IOREAD | _IOWRITE;
 | |
| 			continue;
 | |
| 		/* The sequence may be followed by additional characters */
 | |
| 		default:
 | |
| 			break;
 | |
| 		}
 | |
| 		break;
 | |
| 	}
 | |
| 
 | |
| 	/* Perform a creat() when the file should be truncated or when
 | |
| 	 * the file is opened for writing and the open() failed.
 | |
| 	 */
 | |
| 	if ((rwflags & O_TRUNC)
 | |
| 	    || (((fd = _open(name, rwmode)) < 0)
 | |
| 		    && (rwflags & O_CREAT))) {
 | |
| 		if (((fd = _creat(name, PMODE)) > 0) && flags  | _IOREAD) {
 | |
| 			(void) _close(fd);
 | |
| 			fd = _open(name, rwmode);
 | |
| 		}
 | |
| 			
 | |
| 	}
 | |
| 
 | |
| 	if (fd < 0) return (FILE *)NULL;
 | |
| 
 | |
| 	if ( fstat( fd, &st ) < 0 ) {
 | |
| 		_close(fd);
 | |
| 		return (FILE *)NULL;
 | |
| 	}
 | |
| 	
 | |
| 	if ( S_ISFIFO(st.st_mode) ) flags |= _IOFIFO;
 | |
| 	
 | |
| 	if (( stream = (FILE *) malloc(sizeof(FILE))) == NULL ) {
 | |
| 		_close(fd);
 | |
| 		return (FILE *)NULL;
 | |
| 	}
 | |
| 
 | |
| 	if ((flags & (_IOREAD | _IOWRITE))  == (_IOREAD | _IOWRITE))
 | |
| 		flags &= ~(_IOREADING | _IOWRITING);
 | |
| 
 | |
| 	stream->_count = 0;
 | |
| 	stream->_fd = fd;
 | |
| 	stream->_flags = flags;
 | |
| 	stream->_buf = NULL;
 | |
| 	__iotab[i] = stream;
 | |
| 	return stream;
 | |
| }
 |