158f0484fSRodney W. Grimes /*- 2*8a16b7a1SPedro F. Giffuni * SPDX-License-Identifier: BSD-3-Clause 3*8a16b7a1SPedro F. Giffuni * 458f0484fSRodney W. Grimes * Copyright (c) 1990, 1993 558f0484fSRodney W. Grimes * The Regents of the University of California. All rights reserved. 658f0484fSRodney W. Grimes * 73c87aa1dSDavid Chisnall * Copyright (c) 2011 The FreeBSD Foundation 83c87aa1dSDavid Chisnall * All rights reserved. 93c87aa1dSDavid Chisnall * Portions of this software were developed by David Chisnall 103c87aa1dSDavid Chisnall * under sponsorship from the FreeBSD Foundation. 113c87aa1dSDavid Chisnall * 1258f0484fSRodney W. Grimes * This code is derived from software contributed to Berkeley by 1358f0484fSRodney W. Grimes * Chris Torek. 1458f0484fSRodney W. Grimes * 1558f0484fSRodney W. Grimes * Redistribution and use in source and binary forms, with or without 1658f0484fSRodney W. Grimes * modification, are permitted provided that the following conditions 1758f0484fSRodney W. Grimes * are met: 1858f0484fSRodney W. Grimes * 1. Redistributions of source code must retain the above copyright 1958f0484fSRodney W. Grimes * notice, this list of conditions and the following disclaimer. 2058f0484fSRodney W. Grimes * 2. Redistributions in binary form must reproduce the above copyright 2158f0484fSRodney W. Grimes * notice, this list of conditions and the following disclaimer in the 2258f0484fSRodney W. Grimes * documentation and/or other materials provided with the distribution. 231d8053c5SEd Maste * 3. Neither the name of the University nor the names of its contributors 2458f0484fSRodney W. Grimes * may be used to endorse or promote products derived from this software 2558f0484fSRodney W. Grimes * without specific prior written permission. 2658f0484fSRodney W. Grimes * 2758f0484fSRodney W. Grimes * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND 2858f0484fSRodney W. Grimes * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE 2958f0484fSRodney W. Grimes * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE 3058f0484fSRodney W. Grimes * ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE 3158f0484fSRodney W. Grimes * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL 3258f0484fSRodney W. Grimes * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS 3358f0484fSRodney W. Grimes * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) 3458f0484fSRodney W. Grimes * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT 3558f0484fSRodney W. Grimes * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY 3658f0484fSRodney W. Grimes * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF 3758f0484fSRodney W. Grimes * SUCH DAMAGE. 3858f0484fSRodney W. Grimes */ 3958f0484fSRodney W. Grimes 4058f0484fSRodney W. Grimes #if defined(LIBC_SCCS) && !defined(lint) 4158f0484fSRodney W. Grimes static char sccsid[] = "@(#)snprintf.c 8.1 (Berkeley) 6/4/93"; 4258f0484fSRodney W. Grimes #endif /* LIBC_SCCS and not lint */ 43333fc21eSDavid E. O'Brien #include <sys/cdefs.h> 44333fc21eSDavid E. O'Brien __FBSDID("$FreeBSD$"); 4558f0484fSRodney W. Grimes 46666d00d3SDavid Schultz #include <errno.h> 478c6d2f42SAndrey A. Chernov #include <limits.h> 4858f0484fSRodney W. Grimes #include <stdio.h> 4958f0484fSRodney W. Grimes #include <stdarg.h> 503c87aa1dSDavid Chisnall #include "xlocale_private.h" 5158f0484fSRodney W. Grimes 5229ac6bd2SDaniel Eischen #include "local.h" 5329ac6bd2SDaniel Eischen 54f70177e7SJulian Elischer int 5571a00a44SRobert Drehmel snprintf(char * __restrict str, size_t n, char const * __restrict fmt, ...) 5658f0484fSRodney W. Grimes { 578c6d2f42SAndrey A. Chernov size_t on; 5858f0484fSRodney W. Grimes int ret; 5958f0484fSRodney W. Grimes va_list ap; 601b0181dfSJohn Baldwin FILE f = FAKE_FILE; 6158f0484fSRodney W. Grimes 626e690ad4SAndrey A. Chernov on = n; 638c6d2f42SAndrey A. Chernov if (n != 0) 646e690ad4SAndrey A. Chernov n--; 65666d00d3SDavid Schultz if (n > INT_MAX) { 66666d00d3SDavid Schultz errno = EOVERFLOW; 67666d00d3SDavid Schultz *str = '\0'; 68666d00d3SDavid Schultz return (EOF); 69666d00d3SDavid Schultz } 7058f0484fSRodney W. Grimes va_start(ap, fmt); 7158f0484fSRodney W. Grimes f._flags = __SWR | __SSTR; 7258f0484fSRodney W. Grimes f._bf._base = f._p = (unsigned char *)str; 734ecaf220SAndrey A. Chernov f._bf._size = f._w = n; 743c87aa1dSDavid Chisnall ret = __vfprintf(&f, __get_locale(), fmt, ap); 753c87aa1dSDavid Chisnall if (on > 0) 763c87aa1dSDavid Chisnall *f._p = '\0'; 773c87aa1dSDavid Chisnall va_end(ap); 783c87aa1dSDavid Chisnall return (ret); 793c87aa1dSDavid Chisnall } 803c87aa1dSDavid Chisnall int 813c87aa1dSDavid Chisnall snprintf_l(char * __restrict str, size_t n, locale_t locale, 823c87aa1dSDavid Chisnall char const * __restrict fmt, ...) 833c87aa1dSDavid Chisnall { 843c87aa1dSDavid Chisnall size_t on; 853c87aa1dSDavid Chisnall int ret; 863c87aa1dSDavid Chisnall va_list ap; 873c87aa1dSDavid Chisnall FILE f = FAKE_FILE; 883c87aa1dSDavid Chisnall FIX_LOCALE(locale); 893c87aa1dSDavid Chisnall 903c87aa1dSDavid Chisnall on = n; 913c87aa1dSDavid Chisnall if (n != 0) 923c87aa1dSDavid Chisnall n--; 93666d00d3SDavid Schultz if (n > INT_MAX) { 94666d00d3SDavid Schultz errno = EOVERFLOW; 95666d00d3SDavid Schultz *str = '\0'; 96666d00d3SDavid Schultz return (EOF); 97666d00d3SDavid Schultz } 983c87aa1dSDavid Chisnall va_start(ap, fmt); 993c87aa1dSDavid Chisnall f._flags = __SWR | __SSTR; 1003c87aa1dSDavid Chisnall f._bf._base = f._p = (unsigned char *)str; 1013c87aa1dSDavid Chisnall f._bf._size = f._w = n; 1023c87aa1dSDavid Chisnall ret = __vfprintf(&f, locale, fmt, ap); 1036e690ad4SAndrey A. Chernov if (on > 0) 1046e690ad4SAndrey A. Chernov *f._p = '\0'; 10558f0484fSRodney W. Grimes va_end(ap); 1066e690ad4SAndrey A. Chernov return (ret); 10758f0484fSRodney W. Grimes } 108