140266059SGregory Neil Shapiro /* 2605302a5SGregory Neil Shapiro * Copyright (c) 2000-2002 Sendmail, Inc. and its suppliers. 340266059SGregory Neil Shapiro * All rights reserved. 440266059SGregory Neil Shapiro * Copyright (c) 1990, 1993 540266059SGregory Neil Shapiro * The Regents of the University of California. All rights reserved. 640266059SGregory Neil Shapiro * 740266059SGregory Neil Shapiro * This code is derived from software contributed to Berkeley by 840266059SGregory Neil Shapiro * Chris Torek. 940266059SGregory Neil Shapiro * 1040266059SGregory Neil Shapiro * By using this file, you agree to the terms and conditions set 1140266059SGregory Neil Shapiro * forth in the LICENSE file which can be found at the top level of 1240266059SGregory Neil Shapiro * the sendmail distribution. 1340266059SGregory Neil Shapiro */ 1440266059SGregory Neil Shapiro 1540266059SGregory Neil Shapiro #include <sm/gen.h> 1613bd1963SGregory Neil Shapiro SM_RCSID("@(#)$Id: stdio.c,v 1.56.2.3 2002/10/22 23:07:19 ca Exp $") 1740266059SGregory Neil Shapiro #include <unistd.h> 1840266059SGregory Neil Shapiro #include <errno.h> 1940266059SGregory Neil Shapiro #include <fcntl.h> 2040266059SGregory Neil Shapiro #include <string.h> /* FreeBSD: FD_ZERO needs <string.h> */ 2140266059SGregory Neil Shapiro #include <sys/stat.h> 2240266059SGregory Neil Shapiro #include <sys/time.h> 2340266059SGregory Neil Shapiro #include <sm/heap.h> 2440266059SGregory Neil Shapiro #include <sm/assert.h> 2540266059SGregory Neil Shapiro #include <sm/varargs.h> 2640266059SGregory Neil Shapiro #include <sm/io.h> 2740266059SGregory Neil Shapiro #include <sm/setjmp.h> 2840266059SGregory Neil Shapiro #include <sm/conf.h> 2913bd1963SGregory Neil Shapiro #include <sm/fdset.h> 3040266059SGregory Neil Shapiro #include "local.h" 3140266059SGregory Neil Shapiro 3240266059SGregory Neil Shapiro /* 3340266059SGregory Neil Shapiro ** Overall: 3440266059SGregory Neil Shapiro ** Small standard I/O/seek/close functions. 3540266059SGregory Neil Shapiro ** These maintain the `known seek offset' for seek optimization. 3640266059SGregory Neil Shapiro */ 3740266059SGregory Neil Shapiro 3840266059SGregory Neil Shapiro /* 3940266059SGregory Neil Shapiro ** SM_STDOPEN -- open a file with stdio behavior 4040266059SGregory Neil Shapiro ** 4140266059SGregory Neil Shapiro ** Not associated with the system's stdio in libc. 4240266059SGregory Neil Shapiro ** 4340266059SGregory Neil Shapiro ** Parameters: 4440266059SGregory Neil Shapiro ** fp -- file pointer to be associated with the open 4540266059SGregory Neil Shapiro ** info -- pathname of the file to be opened 4640266059SGregory Neil Shapiro ** flags -- indicates type of access methods 4740266059SGregory Neil Shapiro ** rpool -- ignored 4840266059SGregory Neil Shapiro ** 4940266059SGregory Neil Shapiro ** Returns: 5040266059SGregory Neil Shapiro ** Failure: -1 and set errno 5140266059SGregory Neil Shapiro ** Success: 0 or greater (fd of file from open(2)). 5240266059SGregory Neil Shapiro ** 5340266059SGregory Neil Shapiro */ 5440266059SGregory Neil Shapiro 5540266059SGregory Neil Shapiro /* ARGSUSED3 */ 5640266059SGregory Neil Shapiro int 5740266059SGregory Neil Shapiro sm_stdopen(fp, info, flags, rpool) 5840266059SGregory Neil Shapiro SM_FILE_T *fp; 5940266059SGregory Neil Shapiro const void *info; 6040266059SGregory Neil Shapiro int flags; 6140266059SGregory Neil Shapiro const void *rpool; 6240266059SGregory Neil Shapiro { 6340266059SGregory Neil Shapiro char *path = (char *) info; 6440266059SGregory Neil Shapiro int oflags; 6540266059SGregory Neil Shapiro 6640266059SGregory Neil Shapiro switch (flags) 6740266059SGregory Neil Shapiro { 6840266059SGregory Neil Shapiro case SM_IO_RDWR: 6940266059SGregory Neil Shapiro oflags = O_RDWR; 7040266059SGregory Neil Shapiro break; 7140266059SGregory Neil Shapiro case SM_IO_RDWRTR: 7240266059SGregory Neil Shapiro oflags = O_RDWR | O_CREAT | O_TRUNC; 7340266059SGregory Neil Shapiro break; 7440266059SGregory Neil Shapiro case SM_IO_RDONLY: 7540266059SGregory Neil Shapiro oflags = O_RDONLY; 7640266059SGregory Neil Shapiro break; 7740266059SGregory Neil Shapiro case SM_IO_WRONLY: 7840266059SGregory Neil Shapiro oflags = O_WRONLY | O_CREAT | O_TRUNC; 7940266059SGregory Neil Shapiro break; 8040266059SGregory Neil Shapiro case SM_IO_APPEND: 8140266059SGregory Neil Shapiro oflags = O_APPEND | O_WRONLY | O_CREAT; 8240266059SGregory Neil Shapiro break; 8340266059SGregory Neil Shapiro case SM_IO_APPENDRW: 8440266059SGregory Neil Shapiro oflags = O_APPEND | O_RDWR | O_CREAT; 8540266059SGregory Neil Shapiro break; 8640266059SGregory Neil Shapiro default: 8740266059SGregory Neil Shapiro errno = EINVAL; 8840266059SGregory Neil Shapiro return -1; 8940266059SGregory Neil Shapiro } 9040266059SGregory Neil Shapiro fp->f_file = open(path, oflags, 9140266059SGregory Neil Shapiro S_IRUSR|S_IWUSR|S_IRGRP|S_IWGRP|S_IROTH|S_IWOTH); 9240266059SGregory Neil Shapiro if (fp->f_file < 0) 9340266059SGregory Neil Shapiro return -1; /* errno set by open() */ 9440266059SGregory Neil Shapiro 9540266059SGregory Neil Shapiro if (oflags & O_APPEND) 9640266059SGregory Neil Shapiro (void) (*fp->f_seek)((void *)fp, (off_t)0, SEEK_END); 9740266059SGregory Neil Shapiro 9840266059SGregory Neil Shapiro return fp->f_file; 9940266059SGregory Neil Shapiro } 10040266059SGregory Neil Shapiro 10140266059SGregory Neil Shapiro /* 10240266059SGregory Neil Shapiro ** SM_STDREAD -- read from the file 10340266059SGregory Neil Shapiro ** 10440266059SGregory Neil Shapiro ** Parameters: 10540266059SGregory Neil Shapiro ** fp -- file pointer to read from 10640266059SGregory Neil Shapiro ** buf -- location to place read data 10740266059SGregory Neil Shapiro ** n -- number of bytes to read 10840266059SGregory Neil Shapiro ** 10940266059SGregory Neil Shapiro ** Returns: 11040266059SGregory Neil Shapiro ** Failure: -1 and sets errno 11140266059SGregory Neil Shapiro ** Success: number of bytes read 11240266059SGregory Neil Shapiro ** 11340266059SGregory Neil Shapiro ** Side Effects: 11440266059SGregory Neil Shapiro ** Updates internal offset into file. 11540266059SGregory Neil Shapiro */ 11640266059SGregory Neil Shapiro 11740266059SGregory Neil Shapiro ssize_t 11840266059SGregory Neil Shapiro sm_stdread(fp, buf, n) 11940266059SGregory Neil Shapiro SM_FILE_T *fp; 12040266059SGregory Neil Shapiro char *buf; 12140266059SGregory Neil Shapiro size_t n; 12240266059SGregory Neil Shapiro { 12340266059SGregory Neil Shapiro register int ret; 12440266059SGregory Neil Shapiro 12540266059SGregory Neil Shapiro ret = read(fp->f_file, buf, n); 12640266059SGregory Neil Shapiro 12740266059SGregory Neil Shapiro /* if the read succeeded, update the current offset */ 12840266059SGregory Neil Shapiro if (ret > 0) 12940266059SGregory Neil Shapiro fp->f_lseekoff += ret; 13040266059SGregory Neil Shapiro return ret; 13140266059SGregory Neil Shapiro } 13240266059SGregory Neil Shapiro 13340266059SGregory Neil Shapiro /* 13440266059SGregory Neil Shapiro ** SM_STDWRITE -- write to the file 13540266059SGregory Neil Shapiro ** 13640266059SGregory Neil Shapiro ** Parameters: 13740266059SGregory Neil Shapiro ** fp -- file pointer ro write to 13840266059SGregory Neil Shapiro ** buf -- location of data to be written 13940266059SGregory Neil Shapiro ** n - number of bytes to write 14040266059SGregory Neil Shapiro ** 14140266059SGregory Neil Shapiro ** Returns: 14240266059SGregory Neil Shapiro ** Failure: -1 and sets errno 14340266059SGregory Neil Shapiro ** Success: number of bytes written 14440266059SGregory Neil Shapiro */ 14540266059SGregory Neil Shapiro 14640266059SGregory Neil Shapiro ssize_t 14740266059SGregory Neil Shapiro sm_stdwrite(fp, buf, n) 14840266059SGregory Neil Shapiro SM_FILE_T *fp; 14940266059SGregory Neil Shapiro char const *buf; 15040266059SGregory Neil Shapiro size_t n; 15140266059SGregory Neil Shapiro { 15240266059SGregory Neil Shapiro return write(fp->f_file, buf, n); 15340266059SGregory Neil Shapiro } 15440266059SGregory Neil Shapiro 15540266059SGregory Neil Shapiro /* 15640266059SGregory Neil Shapiro ** SM_STDSEEK -- set the file offset position 15740266059SGregory Neil Shapiro ** 15840266059SGregory Neil Shapiro ** Parmeters: 15940266059SGregory Neil Shapiro ** fp -- file pointer to position 16040266059SGregory Neil Shapiro ** offset -- how far to position from "base" (set by 'whence') 16140266059SGregory Neil Shapiro ** whence -- indicates where the "base" of the 'offset' to start 16240266059SGregory Neil Shapiro ** 16340266059SGregory Neil Shapiro ** Results: 16440266059SGregory Neil Shapiro ** Failure: -1 and sets errno 16540266059SGregory Neil Shapiro ** Success: the current offset 16640266059SGregory Neil Shapiro ** 16740266059SGregory Neil Shapiro ** Side Effects: 16840266059SGregory Neil Shapiro ** Updates the internal value of the offset. 16940266059SGregory Neil Shapiro */ 17040266059SGregory Neil Shapiro 17140266059SGregory Neil Shapiro off_t 17240266059SGregory Neil Shapiro sm_stdseek(fp, offset, whence) 17340266059SGregory Neil Shapiro SM_FILE_T *fp; 17440266059SGregory Neil Shapiro off_t offset; 17540266059SGregory Neil Shapiro int whence; 17640266059SGregory Neil Shapiro { 17740266059SGregory Neil Shapiro register off_t ret; 17840266059SGregory Neil Shapiro 17940266059SGregory Neil Shapiro ret = lseek(fp->f_file, (off_t) offset, whence); 18040266059SGregory Neil Shapiro if (ret != (off_t) -1) 18140266059SGregory Neil Shapiro fp->f_lseekoff = ret; 18240266059SGregory Neil Shapiro return ret; 18340266059SGregory Neil Shapiro } 18440266059SGregory Neil Shapiro 18540266059SGregory Neil Shapiro /* 18640266059SGregory Neil Shapiro ** SM_STDCLOSE -- close the file 18740266059SGregory Neil Shapiro ** 18840266059SGregory Neil Shapiro ** Parameters: 18940266059SGregory Neil Shapiro ** fp -- the file pointer to close 19040266059SGregory Neil Shapiro ** 19140266059SGregory Neil Shapiro ** Returns: 19240266059SGregory Neil Shapiro ** Success: 0 (zero) 19340266059SGregory Neil Shapiro ** Failure: -1 and sets errno 19440266059SGregory Neil Shapiro */ 19540266059SGregory Neil Shapiro 19640266059SGregory Neil Shapiro int 19740266059SGregory Neil Shapiro sm_stdclose(fp) 19840266059SGregory Neil Shapiro SM_FILE_T *fp; 19940266059SGregory Neil Shapiro { 20040266059SGregory Neil Shapiro return close(fp->f_file); 20140266059SGregory Neil Shapiro } 20240266059SGregory Neil Shapiro 20340266059SGregory Neil Shapiro /* 20440266059SGregory Neil Shapiro ** SM_STDSETMODE -- set the access mode for the file 20540266059SGregory Neil Shapiro ** 20640266059SGregory Neil Shapiro ** Called by sm_stdsetinfo(). 20740266059SGregory Neil Shapiro ** 20840266059SGregory Neil Shapiro ** Parameters: 20940266059SGregory Neil Shapiro ** fp -- file pointer 21040266059SGregory Neil Shapiro ** mode -- new mode to set the file access to 21140266059SGregory Neil Shapiro ** 21240266059SGregory Neil Shapiro ** Results: 21340266059SGregory Neil Shapiro ** Success: 0 (zero); 21440266059SGregory Neil Shapiro ** Failure: -1 and sets errno 21540266059SGregory Neil Shapiro */ 21640266059SGregory Neil Shapiro 21740266059SGregory Neil Shapiro int 21840266059SGregory Neil Shapiro sm_stdsetmode(fp, mode) 21940266059SGregory Neil Shapiro SM_FILE_T *fp; 22040266059SGregory Neil Shapiro const int *mode; 22140266059SGregory Neil Shapiro { 22240266059SGregory Neil Shapiro int flags = 0; 22340266059SGregory Neil Shapiro 22440266059SGregory Neil Shapiro switch (*mode) 22540266059SGregory Neil Shapiro { 22640266059SGregory Neil Shapiro case SM_IO_RDWR: 22740266059SGregory Neil Shapiro flags |= SMRW; 22840266059SGregory Neil Shapiro break; 22940266059SGregory Neil Shapiro case SM_IO_RDONLY: 23040266059SGregory Neil Shapiro flags |= SMRD; 23140266059SGregory Neil Shapiro break; 23240266059SGregory Neil Shapiro case SM_IO_WRONLY: 23340266059SGregory Neil Shapiro flags |= SMWR; 23440266059SGregory Neil Shapiro break; 23540266059SGregory Neil Shapiro case SM_IO_APPEND: 23640266059SGregory Neil Shapiro default: 23740266059SGregory Neil Shapiro errno = EINVAL; 23840266059SGregory Neil Shapiro return -1; 23940266059SGregory Neil Shapiro } 24040266059SGregory Neil Shapiro fp->f_flags = fp->f_flags & ~SMMODEMASK; 24140266059SGregory Neil Shapiro fp->f_flags |= flags; 24240266059SGregory Neil Shapiro return 0; 24340266059SGregory Neil Shapiro } 24440266059SGregory Neil Shapiro 24540266059SGregory Neil Shapiro /* 24640266059SGregory Neil Shapiro ** SM_STDGETMODE -- for getinfo determine open mode 24740266059SGregory Neil Shapiro ** 24840266059SGregory Neil Shapiro ** Called by sm_stdgetinfo(). 24940266059SGregory Neil Shapiro ** 25040266059SGregory Neil Shapiro ** Parameters: 25140266059SGregory Neil Shapiro ** fp -- the file mode being determined 25240266059SGregory Neil Shapiro ** mode -- internal mode to map to external value 25340266059SGregory Neil Shapiro ** 25440266059SGregory Neil Shapiro ** Results: 25540266059SGregory Neil Shapiro ** Failure: -1 and sets errno 25640266059SGregory Neil Shapiro ** Success: external mode value 25740266059SGregory Neil Shapiro */ 25840266059SGregory Neil Shapiro 25940266059SGregory Neil Shapiro int 26040266059SGregory Neil Shapiro sm_stdgetmode(fp, mode) 26140266059SGregory Neil Shapiro SM_FILE_T *fp; 26240266059SGregory Neil Shapiro int *mode; 26340266059SGregory Neil Shapiro { 26440266059SGregory Neil Shapiro switch (fp->f_flags & SMMODEMASK) 26540266059SGregory Neil Shapiro { 26640266059SGregory Neil Shapiro case SMRW: 26740266059SGregory Neil Shapiro *mode = SM_IO_RDWR; 26840266059SGregory Neil Shapiro break; 26940266059SGregory Neil Shapiro case SMRD: 27040266059SGregory Neil Shapiro *mode = SM_IO_RDONLY; 27140266059SGregory Neil Shapiro break; 27240266059SGregory Neil Shapiro case SMWR: 27340266059SGregory Neil Shapiro *mode = SM_IO_WRONLY; 27440266059SGregory Neil Shapiro break; 27540266059SGregory Neil Shapiro default: 27640266059SGregory Neil Shapiro errno = EINVAL; 27740266059SGregory Neil Shapiro return -1; 27840266059SGregory Neil Shapiro } 27940266059SGregory Neil Shapiro return 0; 28040266059SGregory Neil Shapiro } 28140266059SGregory Neil Shapiro 28240266059SGregory Neil Shapiro /* 28340266059SGregory Neil Shapiro ** SM_STDSETINFO -- set/modify information for a file 28440266059SGregory Neil Shapiro ** 28540266059SGregory Neil Shapiro ** Parameters: 28640266059SGregory Neil Shapiro ** fp -- file to set info for 28740266059SGregory Neil Shapiro ** what -- type of info to set 28840266059SGregory Neil Shapiro ** valp -- location of data used for setting 28940266059SGregory Neil Shapiro ** 29040266059SGregory Neil Shapiro ** Returns: 29140266059SGregory Neil Shapiro ** Failure: -1 and sets errno 29240266059SGregory Neil Shapiro ** Success: >=0 29340266059SGregory Neil Shapiro */ 29440266059SGregory Neil Shapiro 29540266059SGregory Neil Shapiro int 29640266059SGregory Neil Shapiro sm_stdsetinfo(fp, what, valp) 29740266059SGregory Neil Shapiro SM_FILE_T *fp; 29840266059SGregory Neil Shapiro int what; 29940266059SGregory Neil Shapiro void *valp; 30040266059SGregory Neil Shapiro { 30140266059SGregory Neil Shapiro switch (what) 30240266059SGregory Neil Shapiro { 30340266059SGregory Neil Shapiro case SM_IO_WHAT_MODE: 30440266059SGregory Neil Shapiro return sm_stdsetmode(fp, (const int *)valp); 30540266059SGregory Neil Shapiro 30640266059SGregory Neil Shapiro default: 30740266059SGregory Neil Shapiro errno = EINVAL; 30840266059SGregory Neil Shapiro return -1; 30940266059SGregory Neil Shapiro } 31040266059SGregory Neil Shapiro } 31140266059SGregory Neil Shapiro 31240266059SGregory Neil Shapiro /* 31340266059SGregory Neil Shapiro ** SM_GETINFO -- get information about the open file 31440266059SGregory Neil Shapiro ** 31540266059SGregory Neil Shapiro ** Parameters: 31640266059SGregory Neil Shapiro ** fp -- file to get info for 31740266059SGregory Neil Shapiro ** what -- type of info to get 31840266059SGregory Neil Shapiro ** valp -- location to place found info 31940266059SGregory Neil Shapiro ** 32040266059SGregory Neil Shapiro ** Returns: 32140266059SGregory Neil Shapiro ** Success: may or may not place info in 'valp' depending 32240266059SGregory Neil Shapiro ** on 'what' value, and returns values >=0. Return 32340266059SGregory Neil Shapiro ** value may be the obtained info 32440266059SGregory Neil Shapiro ** Failure: -1 and sets errno 32540266059SGregory Neil Shapiro */ 32640266059SGregory Neil Shapiro 32740266059SGregory Neil Shapiro int 32840266059SGregory Neil Shapiro sm_stdgetinfo(fp, what, valp) 32940266059SGregory Neil Shapiro SM_FILE_T *fp; 33040266059SGregory Neil Shapiro int what; 33140266059SGregory Neil Shapiro void *valp; 33240266059SGregory Neil Shapiro { 33340266059SGregory Neil Shapiro switch (what) 33440266059SGregory Neil Shapiro { 33540266059SGregory Neil Shapiro case SM_IO_WHAT_MODE: 33640266059SGregory Neil Shapiro return sm_stdgetmode(fp, (int *)valp); 33740266059SGregory Neil Shapiro 33840266059SGregory Neil Shapiro case SM_IO_WHAT_FD: 33940266059SGregory Neil Shapiro return fp->f_file; 34040266059SGregory Neil Shapiro 341605302a5SGregory Neil Shapiro case SM_IO_WHAT_SIZE: 342605302a5SGregory Neil Shapiro { 343605302a5SGregory Neil Shapiro struct stat st; 344605302a5SGregory Neil Shapiro 345605302a5SGregory Neil Shapiro if (fstat(fp->f_file, &st) == 0) 346605302a5SGregory Neil Shapiro return st.st_size; 347605302a5SGregory Neil Shapiro else 348605302a5SGregory Neil Shapiro return -1; 349605302a5SGregory Neil Shapiro } 350605302a5SGregory Neil Shapiro 35140266059SGregory Neil Shapiro case SM_IO_IS_READABLE: 35240266059SGregory Neil Shapiro { 35340266059SGregory Neil Shapiro fd_set readfds; 35440266059SGregory Neil Shapiro struct timeval timeout; 35540266059SGregory Neil Shapiro 35613bd1963SGregory Neil Shapiro if (SM_FD_SETSIZE > 0 && fp->f_file >= SM_FD_SETSIZE) 35713bd1963SGregory Neil Shapiro { 35813bd1963SGregory Neil Shapiro errno = EINVAL; 35913bd1963SGregory Neil Shapiro return -1; 36013bd1963SGregory Neil Shapiro } 36140266059SGregory Neil Shapiro FD_ZERO(&readfds); 36240266059SGregory Neil Shapiro SM_FD_SET(fp->f_file, &readfds); 36340266059SGregory Neil Shapiro timeout.tv_sec = 0; 36440266059SGregory Neil Shapiro timeout.tv_usec = 0; 365605302a5SGregory Neil Shapiro if (select(fp->f_file + 1, FDSET_CAST &readfds, 366605302a5SGregory Neil Shapiro NULL, NULL, &timeout) > 0 && 36740266059SGregory Neil Shapiro SM_FD_ISSET(fp->f_file, &readfds)) 36840266059SGregory Neil Shapiro return 1; 36940266059SGregory Neil Shapiro return 0; 37040266059SGregory Neil Shapiro } 37140266059SGregory Neil Shapiro 37240266059SGregory Neil Shapiro default: 37340266059SGregory Neil Shapiro errno = EINVAL; 37440266059SGregory Neil Shapiro return -1; 37540266059SGregory Neil Shapiro } 37640266059SGregory Neil Shapiro } 37740266059SGregory Neil Shapiro 37840266059SGregory Neil Shapiro /* 379605302a5SGregory Neil Shapiro ** SM_STDFDOPEN -- open file by primitive 'fd' rather than pathname 38040266059SGregory Neil Shapiro ** 38140266059SGregory Neil Shapiro ** I/O function to handle fdopen() stdio equivalence. The rest of 38240266059SGregory Neil Shapiro ** the functions are the same as the sm_stdopen() above. 38340266059SGregory Neil Shapiro ** 38440266059SGregory Neil Shapiro ** Parameters: 38540266059SGregory Neil Shapiro ** fp -- the file pointer to be associated with the open 386605302a5SGregory Neil Shapiro ** name -- the primitive file descriptor for association 38740266059SGregory Neil Shapiro ** flags -- indicates type of access methods 38840266059SGregory Neil Shapiro ** rpool -- ignored 38940266059SGregory Neil Shapiro ** 39040266059SGregory Neil Shapiro ** Results: 391605302a5SGregory Neil Shapiro ** Success: primitive file descriptor value 39240266059SGregory Neil Shapiro ** Failure: -1 and sets errno 39340266059SGregory Neil Shapiro */ 39440266059SGregory Neil Shapiro 39540266059SGregory Neil Shapiro /* ARGSUSED3 */ 39640266059SGregory Neil Shapiro int 39740266059SGregory Neil Shapiro sm_stdfdopen(fp, info, flags, rpool) 39840266059SGregory Neil Shapiro SM_FILE_T *fp; 39940266059SGregory Neil Shapiro const void *info; 40040266059SGregory Neil Shapiro int flags; 40140266059SGregory Neil Shapiro const void *rpool; 40240266059SGregory Neil Shapiro { 40340266059SGregory Neil Shapiro int oflags, tmp, fdflags, fd = *((int *) info); 40440266059SGregory Neil Shapiro 40540266059SGregory Neil Shapiro switch (flags) 40640266059SGregory Neil Shapiro { 40740266059SGregory Neil Shapiro case SM_IO_RDWR: 40840266059SGregory Neil Shapiro oflags = O_RDWR | O_CREAT; 40940266059SGregory Neil Shapiro break; 41040266059SGregory Neil Shapiro case SM_IO_RDONLY: 41140266059SGregory Neil Shapiro oflags = O_RDONLY; 41240266059SGregory Neil Shapiro break; 41340266059SGregory Neil Shapiro case SM_IO_WRONLY: 41440266059SGregory Neil Shapiro oflags = O_WRONLY | O_CREAT | O_TRUNC; 41540266059SGregory Neil Shapiro break; 41640266059SGregory Neil Shapiro case SM_IO_APPEND: 41740266059SGregory Neil Shapiro oflags = O_APPEND | O_WRONLY | O_CREAT; 41840266059SGregory Neil Shapiro break; 41940266059SGregory Neil Shapiro case SM_IO_APPENDRW: 42040266059SGregory Neil Shapiro oflags = O_APPEND | O_RDWR | O_CREAT; 42140266059SGregory Neil Shapiro break; 42240266059SGregory Neil Shapiro default: 42340266059SGregory Neil Shapiro errno = EINVAL; 42440266059SGregory Neil Shapiro return -1; 42540266059SGregory Neil Shapiro } 42640266059SGregory Neil Shapiro 42740266059SGregory Neil Shapiro /* Make sure the mode the user wants is a subset of the actual mode. */ 42840266059SGregory Neil Shapiro if ((fdflags = fcntl(fd, F_GETFL, 0)) < 0) 42940266059SGregory Neil Shapiro return -1; 43040266059SGregory Neil Shapiro tmp = fdflags & O_ACCMODE; 43140266059SGregory Neil Shapiro if (tmp != O_RDWR && (tmp != (oflags & O_ACCMODE))) 43240266059SGregory Neil Shapiro { 43340266059SGregory Neil Shapiro errno = EINVAL; 43440266059SGregory Neil Shapiro return -1; 43540266059SGregory Neil Shapiro } 43640266059SGregory Neil Shapiro fp->f_file = fd; 43740266059SGregory Neil Shapiro if (oflags & O_APPEND) 43840266059SGregory Neil Shapiro (void) (*fp->f_seek)(fp, (off_t)0, SEEK_END); 43940266059SGregory Neil Shapiro return fp->f_file; 44040266059SGregory Neil Shapiro } 44140266059SGregory Neil Shapiro 44240266059SGregory Neil Shapiro /* 44340266059SGregory Neil Shapiro ** SM_IO_FOPEN -- open a file 44440266059SGregory Neil Shapiro ** 44540266059SGregory Neil Shapiro ** Same interface and semantics as the open() system call, 44640266059SGregory Neil Shapiro ** except that it returns SM_FILE_T* instead of a file descriptor. 44740266059SGregory Neil Shapiro ** 44840266059SGregory Neil Shapiro ** Parameters: 44940266059SGregory Neil Shapiro ** pathname -- path of file to open 45040266059SGregory Neil Shapiro ** flags -- flags controlling the open 45140266059SGregory Neil Shapiro ** ... -- option "mode" for opening the file 45240266059SGregory Neil Shapiro ** 45340266059SGregory Neil Shapiro ** Returns: 45440266059SGregory Neil Shapiro ** Raises an exception on heap exhaustion. 45540266059SGregory Neil Shapiro ** Returns NULL and sets errno if open() fails. 45640266059SGregory Neil Shapiro ** Returns an SM_FILE_T pointer on success. 45740266059SGregory Neil Shapiro */ 45840266059SGregory Neil Shapiro 45940266059SGregory Neil Shapiro SM_FILE_T * 46040266059SGregory Neil Shapiro #if SM_VA_STD 46140266059SGregory Neil Shapiro sm_io_fopen(char *pathname, int flags, ...) 46240266059SGregory Neil Shapiro #else /* SM_VA_STD */ 46340266059SGregory Neil Shapiro sm_io_fopen(pathname, flags, va_alist) 46440266059SGregory Neil Shapiro char *pathname; 46540266059SGregory Neil Shapiro int flags; 46640266059SGregory Neil Shapiro va_dcl 46740266059SGregory Neil Shapiro #endif /* SM_VA_STD */ 46840266059SGregory Neil Shapiro { 46940266059SGregory Neil Shapiro MODE_T mode; 47040266059SGregory Neil Shapiro SM_FILE_T *fp; 47140266059SGregory Neil Shapiro int ioflags; 47240266059SGregory Neil Shapiro 47340266059SGregory Neil Shapiro if (flags & O_CREAT) 47440266059SGregory Neil Shapiro { 47540266059SGregory Neil Shapiro SM_VA_LOCAL_DECL 47640266059SGregory Neil Shapiro 47740266059SGregory Neil Shapiro SM_VA_START(ap, flags); 47840266059SGregory Neil Shapiro mode = (MODE_T) SM_VA_ARG(ap, int); 47940266059SGregory Neil Shapiro SM_VA_END(ap); 48040266059SGregory Neil Shapiro } 48140266059SGregory Neil Shapiro else 48240266059SGregory Neil Shapiro mode = 0; 48340266059SGregory Neil Shapiro 48440266059SGregory Neil Shapiro switch (flags & O_ACCMODE) 48540266059SGregory Neil Shapiro { 48640266059SGregory Neil Shapiro case O_RDONLY: 48740266059SGregory Neil Shapiro ioflags = SMRD; 48840266059SGregory Neil Shapiro break; 48940266059SGregory Neil Shapiro case O_WRONLY: 49040266059SGregory Neil Shapiro ioflags = SMWR; 49140266059SGregory Neil Shapiro break; 49240266059SGregory Neil Shapiro case O_RDWR: 49340266059SGregory Neil Shapiro ioflags = SMRW; 49440266059SGregory Neil Shapiro break; 49540266059SGregory Neil Shapiro default: 49640266059SGregory Neil Shapiro sm_abort("sm_io_fopen: bad flags 0%o", flags); 49740266059SGregory Neil Shapiro } 49840266059SGregory Neil Shapiro 49940266059SGregory Neil Shapiro fp = sm_fp(SmFtStdio, ioflags, NULL); 50040266059SGregory Neil Shapiro fp->f_file = open(pathname, flags, mode); 50140266059SGregory Neil Shapiro if (fp->f_file == -1) 50240266059SGregory Neil Shapiro { 50340266059SGregory Neil Shapiro fp->f_flags = 0; 50440266059SGregory Neil Shapiro fp->sm_magic = NULL; 50540266059SGregory Neil Shapiro return NULL; 50640266059SGregory Neil Shapiro } 50740266059SGregory Neil Shapiro return fp; 50840266059SGregory Neil Shapiro } 509