11.6Schristos/* $NetBSD: vdprintf.c,v 1.6 2020/08/28 22:02:24 christos Exp $ */ 21.1Schristos 31.1Schristos/*- 41.1Schristos * Copyright (c) 1990, 1993 51.1Schristos * The Regents of the University of California. All rights reserved. 61.1Schristos * 71.1Schristos * This code is derived from software contributed to Berkeley by 81.1Schristos * Chris Torek. 91.1Schristos * 101.1Schristos * Redistribution and use in source and binary forms, with or without 111.1Schristos * modification, are permitted provided that the following conditions 121.1Schristos * are met: 131.1Schristos * 1. Redistributions of source code must retain the above copyright 141.1Schristos * notice, this list of conditions and the following disclaimer. 151.1Schristos * 2. Redistributions in binary form must reproduce the above copyright 161.1Schristos * notice, this list of conditions and the following disclaimer in the 171.1Schristos * documentation and/or other materials provided with the distribution. 181.1Schristos * 3. Neither the name of the University nor the names of its contributors 191.1Schristos * may be used to endorse or promote products derived from this software 201.1Schristos * without specific prior written permission. 211.1Schristos * 221.1Schristos * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND 231.1Schristos * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE 241.1Schristos * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE 251.1Schristos * ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE 261.1Schristos * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL 271.1Schristos * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS 281.1Schristos * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) 291.1Schristos * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT 301.1Schristos * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY 311.1Schristos * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF 321.1Schristos * SUCH DAMAGE. 331.1Schristos */ 341.1Schristos 351.1Schristos#include <sys/cdefs.h> 361.1Schristos#if defined(LIBC_SCCS) && !defined(lint) 371.6Schristos__RCSID("$NetBSD: vdprintf.c,v 1.6 2020/08/28 22:02:24 christos Exp $"); 381.1Schristos#endif /* LIBC_SCCS and not lint */ 391.1Schristos 401.1Schristos#include "namespace.h" 411.1Schristos#include <sys/types.h> 421.1Schristos 431.1Schristos#include <assert.h> 441.1Schristos#include <errno.h> 451.1Schristos#include <fcntl.h> 461.3Sjoerg#include <locale.h> 471.1Schristos#include <unistd.h> 481.1Schristos#include <stdio.h> 491.1Schristos#include <limits.h> 501.1Schristos 511.1Schristos#include "reentrant.h" 521.3Sjoerg#include "setlocale_local.h" 531.1Schristos#include "local.h" 541.1Schristos 551.1Schristos#ifdef __weak_alias 561.1Schristos__weak_alias(vdprintf,_vdprintf) 571.3Sjoerg__weak_alias(vdprintf_l,_vdprintf_l) 581.1Schristos#endif 591.1Schristos 601.1Schristosint 611.3Sjoergvdprintf_l(int fd, locale_t loc, const char * __restrict fmt, va_list ap) 621.1Schristos{ 631.1Schristos FILE f; 641.1Schristos struct __sfileext fext; 651.1Schristos unsigned char buf[BUFSIZ]; 661.1Schristos int ret, fdflags, tmp; 671.1Schristos 681.1Schristos _DIAGASSERT(fd != -1); 691.1Schristos 701.1Schristos /* 711.1Schristos * File descriptors are a full int, but _file is only a short. 721.1Schristos * If we get a valid file descriptor that is greater or equal to 731.1Schristos * USHRT_MAX, then the fd will get sign-extended into an 741.1Schristos * invalid file descriptor. Handle this case by failing the 751.1Schristos * open. (We treat the short as unsigned, and special-case -1). 761.1Schristos */ 771.1Schristos if (fd >= USHRT_MAX) { 781.1Schristos errno = EMFILE; 791.1Schristos return EOF; 801.1Schristos } 811.1Schristos 821.1Schristos if ((fdflags = fcntl(fd, F_GETFL, 0)) == -1) 831.1Schristos return EOF; 841.1Schristos 851.1Schristos tmp = fdflags & O_ACCMODE; 861.1Schristos if (tmp != O_RDWR && tmp != O_WRONLY) { 871.1Schristos errno = EINVAL; 881.1Schristos return EOF; 891.1Schristos } 901.1Schristos 911.1Schristos _FILEEXT_SETUP(&f, &fext); 921.1Schristos __sfpinit(&f); 931.1Schristos f._p = buf; 941.1Schristos f._w = sizeof(buf); 951.1Schristos f._flags = __SWR; 961.1Schristos f._file = fd; 971.1Schristos f._bf._base = buf; 981.1Schristos f._bf._size = sizeof(buf); 991.1Schristos f._cookie = &f; 1001.1Schristos f._read = NULL; 1011.1Schristos f._write = __swrite; 1021.1Schristos f._seek = NULL; 1031.1Schristos f._close = NULL; 1041.1Schristos 1051.3Sjoerg if ((ret = vfprintf_l(&f, loc, fmt, ap)) < 0) 1061.1Schristos return ret; 1071.1Schristos 1081.1Schristos return fflush(&f) ? EOF : ret; 1091.1Schristos} 1101.3Sjoerg 1111.3Sjoergint 1121.3Sjoergvdprintf(int fd, const char * __restrict fmt, va_list ap) 1131.3Sjoerg{ 1141.4Sjoerg return vdprintf_l(fd, _current_locale(), fmt, ap); 1151.3Sjoerg} 116