140266059SGregory Neil Shapiro /* 2d9986b26SGregory Neil Shapiro * Copyright (c) 2000-2003 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> 16a7ec597cSGregory Neil Shapiro SM_RCSID("@(#)$Id: stdio.c,v 1.56.2.13 2003/09/04 01:18:08 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 66a7ec597cSGregory Neil Shapiro switch (SM_IO_MODE(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 } 90a7ec597cSGregory Neil Shapiro #ifdef O_BINARY 91a7ec597cSGregory Neil Shapiro if (SM_IS_BINARY(flags)) 92a7ec597cSGregory Neil Shapiro oflags |= O_BINARY; 93a7ec597cSGregory Neil Shapiro #endif /* O_BINARY */ 9440266059SGregory Neil Shapiro fp->f_file = open(path, oflags, 9540266059SGregory Neil Shapiro S_IRUSR|S_IWUSR|S_IRGRP|S_IWGRP|S_IROTH|S_IWOTH); 9640266059SGregory Neil Shapiro if (fp->f_file < 0) 9740266059SGregory Neil Shapiro return -1; /* errno set by open() */ 9840266059SGregory Neil Shapiro 9940266059SGregory Neil Shapiro if (oflags & O_APPEND) 10040266059SGregory Neil Shapiro (void) (*fp->f_seek)((void *)fp, (off_t)0, SEEK_END); 10140266059SGregory Neil Shapiro 10240266059SGregory Neil Shapiro return fp->f_file; 10340266059SGregory Neil Shapiro } 10440266059SGregory Neil Shapiro 10540266059SGregory Neil Shapiro /* 10640266059SGregory Neil Shapiro ** SM_STDREAD -- read from the file 10740266059SGregory Neil Shapiro ** 10840266059SGregory Neil Shapiro ** Parameters: 10940266059SGregory Neil Shapiro ** fp -- file pointer to read from 11040266059SGregory Neil Shapiro ** buf -- location to place read data 11140266059SGregory Neil Shapiro ** n -- number of bytes to read 11240266059SGregory Neil Shapiro ** 11340266059SGregory Neil Shapiro ** Returns: 11440266059SGregory Neil Shapiro ** Failure: -1 and sets errno 11540266059SGregory Neil Shapiro ** Success: number of bytes read 11640266059SGregory Neil Shapiro ** 11740266059SGregory Neil Shapiro ** Side Effects: 11840266059SGregory Neil Shapiro ** Updates internal offset into file. 11940266059SGregory Neil Shapiro */ 12040266059SGregory Neil Shapiro 12140266059SGregory Neil Shapiro ssize_t 12240266059SGregory Neil Shapiro sm_stdread(fp, buf, n) 12340266059SGregory Neil Shapiro SM_FILE_T *fp; 12440266059SGregory Neil Shapiro char *buf; 12540266059SGregory Neil Shapiro size_t n; 12640266059SGregory Neil Shapiro { 12740266059SGregory Neil Shapiro register int ret; 12840266059SGregory Neil Shapiro 12940266059SGregory Neil Shapiro ret = read(fp->f_file, buf, n); 13040266059SGregory Neil Shapiro 13140266059SGregory Neil Shapiro /* if the read succeeded, update the current offset */ 13240266059SGregory Neil Shapiro if (ret > 0) 13340266059SGregory Neil Shapiro fp->f_lseekoff += ret; 13440266059SGregory Neil Shapiro return ret; 13540266059SGregory Neil Shapiro } 13640266059SGregory Neil Shapiro 13740266059SGregory Neil Shapiro /* 13840266059SGregory Neil Shapiro ** SM_STDWRITE -- write to the file 13940266059SGregory Neil Shapiro ** 14040266059SGregory Neil Shapiro ** Parameters: 14140266059SGregory Neil Shapiro ** fp -- file pointer ro write to 14240266059SGregory Neil Shapiro ** buf -- location of data to be written 14340266059SGregory Neil Shapiro ** n - number of bytes to write 14440266059SGregory Neil Shapiro ** 14540266059SGregory Neil Shapiro ** Returns: 14640266059SGregory Neil Shapiro ** Failure: -1 and sets errno 14740266059SGregory Neil Shapiro ** Success: number of bytes written 14840266059SGregory Neil Shapiro */ 14940266059SGregory Neil Shapiro 15040266059SGregory Neil Shapiro ssize_t 15140266059SGregory Neil Shapiro sm_stdwrite(fp, buf, n) 15240266059SGregory Neil Shapiro SM_FILE_T *fp; 15340266059SGregory Neil Shapiro char const *buf; 15440266059SGregory Neil Shapiro size_t n; 15540266059SGregory Neil Shapiro { 15640266059SGregory Neil Shapiro return write(fp->f_file, buf, n); 15740266059SGregory Neil Shapiro } 15840266059SGregory Neil Shapiro 15940266059SGregory Neil Shapiro /* 16040266059SGregory Neil Shapiro ** SM_STDSEEK -- set the file offset position 16140266059SGregory Neil Shapiro ** 16240266059SGregory Neil Shapiro ** Parmeters: 16340266059SGregory Neil Shapiro ** fp -- file pointer to position 16440266059SGregory Neil Shapiro ** offset -- how far to position from "base" (set by 'whence') 16540266059SGregory Neil Shapiro ** whence -- indicates where the "base" of the 'offset' to start 16640266059SGregory Neil Shapiro ** 16740266059SGregory Neil Shapiro ** Results: 16840266059SGregory Neil Shapiro ** Failure: -1 and sets errno 16940266059SGregory Neil Shapiro ** Success: the current offset 17040266059SGregory Neil Shapiro ** 17140266059SGregory Neil Shapiro ** Side Effects: 17240266059SGregory Neil Shapiro ** Updates the internal value of the offset. 17340266059SGregory Neil Shapiro */ 17440266059SGregory Neil Shapiro 17540266059SGregory Neil Shapiro off_t 17640266059SGregory Neil Shapiro sm_stdseek(fp, offset, whence) 17740266059SGregory Neil Shapiro SM_FILE_T *fp; 17840266059SGregory Neil Shapiro off_t offset; 17940266059SGregory Neil Shapiro int whence; 18040266059SGregory Neil Shapiro { 18140266059SGregory Neil Shapiro register off_t ret; 18240266059SGregory Neil Shapiro 18340266059SGregory Neil Shapiro ret = lseek(fp->f_file, (off_t) offset, whence); 18440266059SGregory Neil Shapiro if (ret != (off_t) -1) 18540266059SGregory Neil Shapiro fp->f_lseekoff = ret; 18640266059SGregory Neil Shapiro return ret; 18740266059SGregory Neil Shapiro } 18840266059SGregory Neil Shapiro 18940266059SGregory Neil Shapiro /* 19040266059SGregory Neil Shapiro ** SM_STDCLOSE -- close the file 19140266059SGregory Neil Shapiro ** 19240266059SGregory Neil Shapiro ** Parameters: 19340266059SGregory Neil Shapiro ** fp -- the file pointer to close 19440266059SGregory Neil Shapiro ** 19540266059SGregory Neil Shapiro ** Returns: 19640266059SGregory Neil Shapiro ** Success: 0 (zero) 19740266059SGregory Neil Shapiro ** Failure: -1 and sets errno 19840266059SGregory Neil Shapiro */ 19940266059SGregory Neil Shapiro 20040266059SGregory Neil Shapiro int 20140266059SGregory Neil Shapiro sm_stdclose(fp) 20240266059SGregory Neil Shapiro SM_FILE_T *fp; 20340266059SGregory Neil Shapiro { 20440266059SGregory Neil Shapiro return close(fp->f_file); 20540266059SGregory Neil Shapiro } 20640266059SGregory Neil Shapiro 20740266059SGregory Neil Shapiro /* 20840266059SGregory Neil Shapiro ** SM_STDSETMODE -- set the access mode for the file 20940266059SGregory Neil Shapiro ** 21040266059SGregory Neil Shapiro ** Called by sm_stdsetinfo(). 21140266059SGregory Neil Shapiro ** 21240266059SGregory Neil Shapiro ** Parameters: 21340266059SGregory Neil Shapiro ** fp -- file pointer 21440266059SGregory Neil Shapiro ** mode -- new mode to set the file access to 21540266059SGregory Neil Shapiro ** 21640266059SGregory Neil Shapiro ** Results: 21740266059SGregory Neil Shapiro ** Success: 0 (zero); 21840266059SGregory Neil Shapiro ** Failure: -1 and sets errno 21940266059SGregory Neil Shapiro */ 22040266059SGregory Neil Shapiro 22140266059SGregory Neil Shapiro int 22240266059SGregory Neil Shapiro sm_stdsetmode(fp, mode) 22340266059SGregory Neil Shapiro SM_FILE_T *fp; 22440266059SGregory Neil Shapiro const int *mode; 22540266059SGregory Neil Shapiro { 22640266059SGregory Neil Shapiro int flags = 0; 22740266059SGregory Neil Shapiro 228a7ec597cSGregory Neil Shapiro switch (SM_IO_MODE(*mode)) 22940266059SGregory Neil Shapiro { 23040266059SGregory Neil Shapiro case SM_IO_RDWR: 23140266059SGregory Neil Shapiro flags |= SMRW; 23240266059SGregory Neil Shapiro break; 23340266059SGregory Neil Shapiro case SM_IO_RDONLY: 23440266059SGregory Neil Shapiro flags |= SMRD; 23540266059SGregory Neil Shapiro break; 23640266059SGregory Neil Shapiro case SM_IO_WRONLY: 23740266059SGregory Neil Shapiro flags |= SMWR; 23840266059SGregory Neil Shapiro break; 23940266059SGregory Neil Shapiro case SM_IO_APPEND: 24040266059SGregory Neil Shapiro default: 24140266059SGregory Neil Shapiro errno = EINVAL; 24240266059SGregory Neil Shapiro return -1; 24340266059SGregory Neil Shapiro } 24440266059SGregory Neil Shapiro fp->f_flags = fp->f_flags & ~SMMODEMASK; 24540266059SGregory Neil Shapiro fp->f_flags |= flags; 24640266059SGregory Neil Shapiro return 0; 24740266059SGregory Neil Shapiro } 24840266059SGregory Neil Shapiro 24940266059SGregory Neil Shapiro /* 25040266059SGregory Neil Shapiro ** SM_STDGETMODE -- for getinfo determine open mode 25140266059SGregory Neil Shapiro ** 25240266059SGregory Neil Shapiro ** Called by sm_stdgetinfo(). 25340266059SGregory Neil Shapiro ** 25440266059SGregory Neil Shapiro ** Parameters: 25540266059SGregory Neil Shapiro ** fp -- the file mode being determined 25640266059SGregory Neil Shapiro ** mode -- internal mode to map to external value 25740266059SGregory Neil Shapiro ** 25840266059SGregory Neil Shapiro ** Results: 25940266059SGregory Neil Shapiro ** Failure: -1 and sets errno 26040266059SGregory Neil Shapiro ** Success: external mode value 26140266059SGregory Neil Shapiro */ 26240266059SGregory Neil Shapiro 26340266059SGregory Neil Shapiro int 26440266059SGregory Neil Shapiro sm_stdgetmode(fp, mode) 26540266059SGregory Neil Shapiro SM_FILE_T *fp; 26640266059SGregory Neil Shapiro int *mode; 26740266059SGregory Neil Shapiro { 26840266059SGregory Neil Shapiro switch (fp->f_flags & SMMODEMASK) 26940266059SGregory Neil Shapiro { 27040266059SGregory Neil Shapiro case SMRW: 27140266059SGregory Neil Shapiro *mode = SM_IO_RDWR; 27240266059SGregory Neil Shapiro break; 27340266059SGregory Neil Shapiro case SMRD: 27440266059SGregory Neil Shapiro *mode = SM_IO_RDONLY; 27540266059SGregory Neil Shapiro break; 27640266059SGregory Neil Shapiro case SMWR: 27740266059SGregory Neil Shapiro *mode = SM_IO_WRONLY; 27840266059SGregory Neil Shapiro break; 27940266059SGregory Neil Shapiro default: 28040266059SGregory Neil Shapiro errno = EINVAL; 28140266059SGregory Neil Shapiro return -1; 28240266059SGregory Neil Shapiro } 28340266059SGregory Neil Shapiro return 0; 28440266059SGregory Neil Shapiro } 28540266059SGregory Neil Shapiro 28640266059SGregory Neil Shapiro /* 28740266059SGregory Neil Shapiro ** SM_STDSETINFO -- set/modify information for a file 28840266059SGregory Neil Shapiro ** 28940266059SGregory Neil Shapiro ** Parameters: 29040266059SGregory Neil Shapiro ** fp -- file to set info for 29140266059SGregory Neil Shapiro ** what -- type of info to set 29240266059SGregory Neil Shapiro ** valp -- location of data used for setting 29340266059SGregory Neil Shapiro ** 29440266059SGregory Neil Shapiro ** Returns: 29540266059SGregory Neil Shapiro ** Failure: -1 and sets errno 29640266059SGregory Neil Shapiro ** Success: >=0 29740266059SGregory Neil Shapiro */ 29840266059SGregory Neil Shapiro 29940266059SGregory Neil Shapiro int 30040266059SGregory Neil Shapiro sm_stdsetinfo(fp, what, valp) 30140266059SGregory Neil Shapiro SM_FILE_T *fp; 30240266059SGregory Neil Shapiro int what; 30340266059SGregory Neil Shapiro void *valp; 30440266059SGregory Neil Shapiro { 30540266059SGregory Neil Shapiro switch (what) 30640266059SGregory Neil Shapiro { 30740266059SGregory Neil Shapiro case SM_IO_WHAT_MODE: 30840266059SGregory Neil Shapiro return sm_stdsetmode(fp, (const int *)valp); 30940266059SGregory Neil Shapiro 31040266059SGregory Neil Shapiro default: 31140266059SGregory Neil Shapiro errno = EINVAL; 31240266059SGregory Neil Shapiro return -1; 31340266059SGregory Neil Shapiro } 31440266059SGregory Neil Shapiro } 31540266059SGregory Neil Shapiro 31640266059SGregory Neil Shapiro /* 31740266059SGregory Neil Shapiro ** SM_GETINFO -- get information about the open file 31840266059SGregory Neil Shapiro ** 31940266059SGregory Neil Shapiro ** Parameters: 32040266059SGregory Neil Shapiro ** fp -- file to get info for 32140266059SGregory Neil Shapiro ** what -- type of info to get 32240266059SGregory Neil Shapiro ** valp -- location to place found info 32340266059SGregory Neil Shapiro ** 32440266059SGregory Neil Shapiro ** Returns: 32540266059SGregory Neil Shapiro ** Success: may or may not place info in 'valp' depending 32640266059SGregory Neil Shapiro ** on 'what' value, and returns values >=0. Return 32740266059SGregory Neil Shapiro ** value may be the obtained info 32840266059SGregory Neil Shapiro ** Failure: -1 and sets errno 32940266059SGregory Neil Shapiro */ 33040266059SGregory Neil Shapiro 33140266059SGregory Neil Shapiro int 33240266059SGregory Neil Shapiro sm_stdgetinfo(fp, what, valp) 33340266059SGregory Neil Shapiro SM_FILE_T *fp; 33440266059SGregory Neil Shapiro int what; 33540266059SGregory Neil Shapiro void *valp; 33640266059SGregory Neil Shapiro { 33740266059SGregory Neil Shapiro switch (what) 33840266059SGregory Neil Shapiro { 33940266059SGregory Neil Shapiro case SM_IO_WHAT_MODE: 34040266059SGregory Neil Shapiro return sm_stdgetmode(fp, (int *)valp); 34140266059SGregory Neil Shapiro 34240266059SGregory Neil Shapiro case SM_IO_WHAT_FD: 34340266059SGregory Neil Shapiro return fp->f_file; 34440266059SGregory Neil Shapiro 345605302a5SGregory Neil Shapiro case SM_IO_WHAT_SIZE: 346605302a5SGregory Neil Shapiro { 347605302a5SGregory Neil Shapiro struct stat st; 348605302a5SGregory Neil Shapiro 349605302a5SGregory Neil Shapiro if (fstat(fp->f_file, &st) == 0) 350605302a5SGregory Neil Shapiro return st.st_size; 351605302a5SGregory Neil Shapiro else 352605302a5SGregory Neil Shapiro return -1; 353605302a5SGregory Neil Shapiro } 354605302a5SGregory Neil Shapiro 35540266059SGregory Neil Shapiro case SM_IO_IS_READABLE: 35640266059SGregory Neil Shapiro { 35740266059SGregory Neil Shapiro fd_set readfds; 35840266059SGregory Neil Shapiro struct timeval timeout; 35940266059SGregory Neil Shapiro 36013bd1963SGregory Neil Shapiro if (SM_FD_SETSIZE > 0 && fp->f_file >= SM_FD_SETSIZE) 36113bd1963SGregory Neil Shapiro { 36213bd1963SGregory Neil Shapiro errno = EINVAL; 36313bd1963SGregory Neil Shapiro return -1; 36413bd1963SGregory Neil Shapiro } 36540266059SGregory Neil Shapiro FD_ZERO(&readfds); 36640266059SGregory Neil Shapiro SM_FD_SET(fp->f_file, &readfds); 36740266059SGregory Neil Shapiro timeout.tv_sec = 0; 36840266059SGregory Neil Shapiro timeout.tv_usec = 0; 369605302a5SGregory Neil Shapiro if (select(fp->f_file + 1, FDSET_CAST &readfds, 370605302a5SGregory Neil Shapiro NULL, NULL, &timeout) > 0 && 37140266059SGregory Neil Shapiro SM_FD_ISSET(fp->f_file, &readfds)) 37240266059SGregory Neil Shapiro return 1; 37340266059SGregory Neil Shapiro return 0; 37440266059SGregory Neil Shapiro } 37540266059SGregory Neil Shapiro 37640266059SGregory Neil Shapiro default: 37740266059SGregory Neil Shapiro errno = EINVAL; 37840266059SGregory Neil Shapiro return -1; 37940266059SGregory Neil Shapiro } 38040266059SGregory Neil Shapiro } 38140266059SGregory Neil Shapiro 38240266059SGregory Neil Shapiro /* 383605302a5SGregory Neil Shapiro ** SM_STDFDOPEN -- open file by primitive 'fd' rather than pathname 38440266059SGregory Neil Shapiro ** 38540266059SGregory Neil Shapiro ** I/O function to handle fdopen() stdio equivalence. The rest of 38640266059SGregory Neil Shapiro ** the functions are the same as the sm_stdopen() above. 38740266059SGregory Neil Shapiro ** 38840266059SGregory Neil Shapiro ** Parameters: 38940266059SGregory Neil Shapiro ** fp -- the file pointer to be associated with the open 390605302a5SGregory Neil Shapiro ** name -- the primitive file descriptor for association 39140266059SGregory Neil Shapiro ** flags -- indicates type of access methods 39240266059SGregory Neil Shapiro ** rpool -- ignored 39340266059SGregory Neil Shapiro ** 39440266059SGregory Neil Shapiro ** Results: 395605302a5SGregory Neil Shapiro ** Success: primitive file descriptor value 39640266059SGregory Neil Shapiro ** Failure: -1 and sets errno 39740266059SGregory Neil Shapiro */ 39840266059SGregory Neil Shapiro 39940266059SGregory Neil Shapiro /* ARGSUSED3 */ 40040266059SGregory Neil Shapiro int 40140266059SGregory Neil Shapiro sm_stdfdopen(fp, info, flags, rpool) 40240266059SGregory Neil Shapiro SM_FILE_T *fp; 40340266059SGregory Neil Shapiro const void *info; 40440266059SGregory Neil Shapiro int flags; 40540266059SGregory Neil Shapiro const void *rpool; 40640266059SGregory Neil Shapiro { 40740266059SGregory Neil Shapiro int oflags, tmp, fdflags, fd = *((int *) info); 40840266059SGregory Neil Shapiro 409a7ec597cSGregory Neil Shapiro switch (SM_IO_MODE(flags)) 41040266059SGregory Neil Shapiro { 41140266059SGregory Neil Shapiro case SM_IO_RDWR: 41240266059SGregory Neil Shapiro oflags = O_RDWR | O_CREAT; 41340266059SGregory Neil Shapiro break; 41440266059SGregory Neil Shapiro case SM_IO_RDONLY: 41540266059SGregory Neil Shapiro oflags = O_RDONLY; 41640266059SGregory Neil Shapiro break; 41740266059SGregory Neil Shapiro case SM_IO_WRONLY: 41840266059SGregory Neil Shapiro oflags = O_WRONLY | O_CREAT | O_TRUNC; 41940266059SGregory Neil Shapiro break; 42040266059SGregory Neil Shapiro case SM_IO_APPEND: 42140266059SGregory Neil Shapiro oflags = O_APPEND | O_WRONLY | O_CREAT; 42240266059SGregory Neil Shapiro break; 42340266059SGregory Neil Shapiro case SM_IO_APPENDRW: 42440266059SGregory Neil Shapiro oflags = O_APPEND | O_RDWR | O_CREAT; 42540266059SGregory Neil Shapiro break; 42640266059SGregory Neil Shapiro default: 42740266059SGregory Neil Shapiro errno = EINVAL; 42840266059SGregory Neil Shapiro return -1; 42940266059SGregory Neil Shapiro } 430a7ec597cSGregory Neil Shapiro #ifdef O_BINARY 431a7ec597cSGregory Neil Shapiro if (SM_IS_BINARY(flags)) 432a7ec597cSGregory Neil Shapiro oflags |= O_BINARY; 433a7ec597cSGregory Neil Shapiro #endif /* O_BINARY */ 43440266059SGregory Neil Shapiro 43540266059SGregory Neil Shapiro /* Make sure the mode the user wants is a subset of the actual mode. */ 43640266059SGregory Neil Shapiro if ((fdflags = fcntl(fd, F_GETFL, 0)) < 0) 43740266059SGregory Neil Shapiro return -1; 43840266059SGregory Neil Shapiro tmp = fdflags & O_ACCMODE; 43940266059SGregory Neil Shapiro if (tmp != O_RDWR && (tmp != (oflags & O_ACCMODE))) 44040266059SGregory Neil Shapiro { 44140266059SGregory Neil Shapiro errno = EINVAL; 44240266059SGregory Neil Shapiro return -1; 44340266059SGregory Neil Shapiro } 44440266059SGregory Neil Shapiro fp->f_file = fd; 44540266059SGregory Neil Shapiro if (oflags & O_APPEND) 44640266059SGregory Neil Shapiro (void) (*fp->f_seek)(fp, (off_t)0, SEEK_END); 44740266059SGregory Neil Shapiro return fp->f_file; 44840266059SGregory Neil Shapiro } 44940266059SGregory Neil Shapiro 45040266059SGregory Neil Shapiro /* 45140266059SGregory Neil Shapiro ** SM_IO_FOPEN -- open a file 45240266059SGregory Neil Shapiro ** 45340266059SGregory Neil Shapiro ** Same interface and semantics as the open() system call, 45440266059SGregory Neil Shapiro ** except that it returns SM_FILE_T* instead of a file descriptor. 45540266059SGregory Neil Shapiro ** 45640266059SGregory Neil Shapiro ** Parameters: 45740266059SGregory Neil Shapiro ** pathname -- path of file to open 45840266059SGregory Neil Shapiro ** flags -- flags controlling the open 45940266059SGregory Neil Shapiro ** ... -- option "mode" for opening the file 46040266059SGregory Neil Shapiro ** 46140266059SGregory Neil Shapiro ** Returns: 46240266059SGregory Neil Shapiro ** Raises an exception on heap exhaustion. 46340266059SGregory Neil Shapiro ** Returns NULL and sets errno if open() fails. 46440266059SGregory Neil Shapiro ** Returns an SM_FILE_T pointer on success. 46540266059SGregory Neil Shapiro */ 46640266059SGregory Neil Shapiro 46740266059SGregory Neil Shapiro SM_FILE_T * 46840266059SGregory Neil Shapiro #if SM_VA_STD 46940266059SGregory Neil Shapiro sm_io_fopen(char *pathname, int flags, ...) 47040266059SGregory Neil Shapiro #else /* SM_VA_STD */ 47140266059SGregory Neil Shapiro sm_io_fopen(pathname, flags, va_alist) 47240266059SGregory Neil Shapiro char *pathname; 47340266059SGregory Neil Shapiro int flags; 47440266059SGregory Neil Shapiro va_dcl 47540266059SGregory Neil Shapiro #endif /* SM_VA_STD */ 47640266059SGregory Neil Shapiro { 47740266059SGregory Neil Shapiro MODE_T mode; 47840266059SGregory Neil Shapiro SM_FILE_T *fp; 47940266059SGregory Neil Shapiro int ioflags; 48040266059SGregory Neil Shapiro 48140266059SGregory Neil Shapiro if (flags & O_CREAT) 48240266059SGregory Neil Shapiro { 48340266059SGregory Neil Shapiro SM_VA_LOCAL_DECL 48440266059SGregory Neil Shapiro 48540266059SGregory Neil Shapiro SM_VA_START(ap, flags); 48640266059SGregory Neil Shapiro mode = (MODE_T) SM_VA_ARG(ap, int); 48740266059SGregory Neil Shapiro SM_VA_END(ap); 48840266059SGregory Neil Shapiro } 48940266059SGregory Neil Shapiro else 49040266059SGregory Neil Shapiro mode = 0; 49140266059SGregory Neil Shapiro 49240266059SGregory Neil Shapiro switch (flags & O_ACCMODE) 49340266059SGregory Neil Shapiro { 49440266059SGregory Neil Shapiro case O_RDONLY: 49540266059SGregory Neil Shapiro ioflags = SMRD; 49640266059SGregory Neil Shapiro break; 49740266059SGregory Neil Shapiro case O_WRONLY: 49840266059SGregory Neil Shapiro ioflags = SMWR; 49940266059SGregory Neil Shapiro break; 50040266059SGregory Neil Shapiro case O_RDWR: 50140266059SGregory Neil Shapiro ioflags = SMRW; 50240266059SGregory Neil Shapiro break; 50340266059SGregory Neil Shapiro default: 50440266059SGregory Neil Shapiro sm_abort("sm_io_fopen: bad flags 0%o", flags); 50540266059SGregory Neil Shapiro } 50640266059SGregory Neil Shapiro 50740266059SGregory Neil Shapiro fp = sm_fp(SmFtStdio, ioflags, NULL); 50840266059SGregory Neil Shapiro fp->f_file = open(pathname, flags, mode); 50940266059SGregory Neil Shapiro if (fp->f_file == -1) 51040266059SGregory Neil Shapiro { 51140266059SGregory Neil Shapiro fp->f_flags = 0; 51240266059SGregory Neil Shapiro fp->sm_magic = NULL; 51340266059SGregory Neil Shapiro return NULL; 51440266059SGregory Neil Shapiro } 51540266059SGregory Neil Shapiro return fp; 51640266059SGregory Neil Shapiro } 517