fwrite.c revision b2e1abda0503874a5bb097e0b55ad4f8ce211a8d
11dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project/*- 21dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project * Copyright (c) 1990, 1993 31dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project * The Regents of the University of California. All rights reserved. 41dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project * 51dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project * This code is derived from software contributed to Berkeley by 61dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project * Chris Torek. 71dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project * 81dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project * Redistribution and use in source and binary forms, with or without 91dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project * modification, are permitted provided that the following conditions 101dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project * are met: 111dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project * 1. Redistributions of source code must retain the above copyright 121dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project * notice, this list of conditions and the following disclaimer. 131dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project * 2. Redistributions in binary form must reproduce the above copyright 141dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project * notice, this list of conditions and the following disclaimer in the 151dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project * documentation and/or other materials provided with the distribution. 16b2e1abda0503874a5bb097e0b55ad4f8ce211a8dElliott Hughes * 3. Neither the name of the University nor the names of its contributors 171dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project * may be used to endorse or promote products derived from this software 181dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project * without specific prior written permission. 191dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project * 201dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND 211dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE 221dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE 231dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project * ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE 241dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL 251dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS 261dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) 271dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT 281dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY 291dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF 301dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project * SUCH DAMAGE. 311dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project */ 321dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project 336b05c8e28017518fae04a3a601d0d245916561d2Elliott Hughes#if defined(LIBC_SCCS) && !defined(lint) 346b05c8e28017518fae04a3a601d0d245916561d2Elliott Hughesstatic char sccsid[] = "@(#)fwrite.c 8.1 (Berkeley) 6/4/93"; 356b05c8e28017518fae04a3a601d0d245916561d2Elliott Hughes#endif /* LIBC_SCCS and not lint */ 366b05c8e28017518fae04a3a601d0d245916561d2Elliott Hughes#include <sys/cdefs.h> 376b05c8e28017518fae04a3a601d0d245916561d2Elliott Hughes__FBSDID("$FreeBSD$"); 386b05c8e28017518fae04a3a601d0d245916561d2Elliott Hughes 396b05c8e28017518fae04a3a601d0d245916561d2Elliott Hughes#include "namespace.h" 406b05c8e28017518fae04a3a601d0d245916561d2Elliott Hughes#include <errno.h> 416b05c8e28017518fae04a3a601d0d245916561d2Elliott Hughes#include <stdint.h> 421dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project#include <stdio.h> 436b05c8e28017518fae04a3a601d0d245916561d2Elliott Hughes#include "un-namespace.h" 441dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project#include "local.h" 451dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project#include "fvwrite.h" 466b05c8e28017518fae04a3a601d0d245916561d2Elliott Hughes#include "libc_private.h" 471dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project 481dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project/* 491dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project * Write `count' objects (each size `size') from memory to the given file. 501dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project * Return the number of whole objects written. 511dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project */ 521dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Projectsize_t 53b2e1abda0503874a5bb097e0b55ad4f8ce211a8dElliott Hughesfwrite(const void * __restrict buf, size_t size, size_t count, FILE * __restrict fp) 541dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project{ 551dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project size_t n; 561dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project struct __suio uio; 571dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project struct __siov iov; 586b05c8e28017518fae04a3a601d0d245916561d2Elliott Hughes 596b05c8e28017518fae04a3a601d0d245916561d2Elliott Hughes /* 606b05c8e28017518fae04a3a601d0d245916561d2Elliott Hughes * ANSI and SUSv2 require a return value of 0 if size or count are 0. 616b05c8e28017518fae04a3a601d0d245916561d2Elliott Hughes */ 626b05c8e28017518fae04a3a601d0d245916561d2Elliott Hughes if ((count == 0) || (size == 0)) 636b05c8e28017518fae04a3a601d0d245916561d2Elliott Hughes return (0); 646b05c8e28017518fae04a3a601d0d245916561d2Elliott Hughes 656b05c8e28017518fae04a3a601d0d245916561d2Elliott Hughes /* 666b05c8e28017518fae04a3a601d0d245916561d2Elliott Hughes * Check for integer overflow. As an optimization, first check that 676b05c8e28017518fae04a3a601d0d245916561d2Elliott Hughes * at least one of {count, size} is at least 2^16, since if both 686b05c8e28017518fae04a3a601d0d245916561d2Elliott Hughes * values are less than that, their product can't possible overflow 696b05c8e28017518fae04a3a601d0d245916561d2Elliott Hughes * (size_t is always at least 32 bits on FreeBSD). 706b05c8e28017518fae04a3a601d0d245916561d2Elliott Hughes */ 716b05c8e28017518fae04a3a601d0d245916561d2Elliott Hughes if (((count | size) > 0xFFFF) && 726b05c8e28017518fae04a3a601d0d245916561d2Elliott Hughes (count > SIZE_MAX / size)) { 736b05c8e28017518fae04a3a601d0d245916561d2Elliott Hughes errno = EINVAL; 746b05c8e28017518fae04a3a601d0d245916561d2Elliott Hughes fp->_flags |= __SERR; 756b05c8e28017518fae04a3a601d0d245916561d2Elliott Hughes return (0); 766b05c8e28017518fae04a3a601d0d245916561d2Elliott Hughes } 776b05c8e28017518fae04a3a601d0d245916561d2Elliott Hughes 786b05c8e28017518fae04a3a601d0d245916561d2Elliott Hughes n = count * size; 791dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project 801dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project iov.iov_base = (void *)buf; 816b05c8e28017518fae04a3a601d0d245916561d2Elliott Hughes uio.uio_resid = iov.iov_len = n; 821dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project uio.uio_iov = &iov; 831dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project uio.uio_iovcnt = 1; 841dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project 856b05c8e28017518fae04a3a601d0d245916561d2Elliott Hughes FLOCKFILE(fp); 866b05c8e28017518fae04a3a601d0d245916561d2Elliott Hughes ORIENT(fp, -1); 871dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project /* 881dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project * The usual case is success (__sfvwrite returns 0); 891dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project * skip the divide if this happens, since divides are 901dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project * generally slow and since this occurs whenever size==0. 911dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project */ 926b05c8e28017518fae04a3a601d0d245916561d2Elliott Hughes if (__sfvwrite(fp, &uio) != 0) 936b05c8e28017518fae04a3a601d0d245916561d2Elliott Hughes count = (n - uio.uio_resid) / size; 94f582340a6a48588aa50da17e1620e8f91b146941Kenny Root FUNLOCKFILE(fp); 956b05c8e28017518fae04a3a601d0d245916561d2Elliott Hughes return (count); 961dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project} 97