1*66932323SSascha Wildner /* $OpenBSD: reallocarray.c,v 1.3 2015/09/13 08:31:47 guenther Exp $ */
2*66932323SSascha Wildner /*
3*66932323SSascha Wildner * Copyright (c) 2008 Otto Moerbeek <otto@drijf.net>
4*66932323SSascha Wildner *
5*66932323SSascha Wildner * Permission to use, copy, modify, and distribute this software for any
6*66932323SSascha Wildner * purpose with or without fee is hereby granted, provided that the above
7*66932323SSascha Wildner * copyright notice and this permission notice appear in all copies.
8*66932323SSascha Wildner *
9*66932323SSascha Wildner * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
10*66932323SSascha Wildner * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
11*66932323SSascha Wildner * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
12*66932323SSascha Wildner * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
13*66932323SSascha Wildner * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
14*66932323SSascha Wildner * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
15*66932323SSascha Wildner * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
16*66932323SSascha Wildner */
17*66932323SSascha Wildner
18*66932323SSascha Wildner #include <sys/types.h>
19*66932323SSascha Wildner #include <errno.h>
20*66932323SSascha Wildner #include <stdint.h>
21*66932323SSascha Wildner #include <stdlib.h>
22*66932323SSascha Wildner
23*66932323SSascha Wildner /*
24*66932323SSascha Wildner * This is sqrt(SIZE_MAX+1), as s1*s2 <= SIZE_MAX
25*66932323SSascha Wildner * if both s1 < MUL_NO_OVERFLOW and s2 < MUL_NO_OVERFLOW
26*66932323SSascha Wildner */
27*66932323SSascha Wildner #define MUL_NO_OVERFLOW ((size_t)1 << (sizeof(size_t) * 4))
28*66932323SSascha Wildner
29*66932323SSascha Wildner void *
reallocarray(void * ptr,size_t number,size_t size)30*66932323SSascha Wildner reallocarray(void *ptr, size_t number, size_t size)
31*66932323SSascha Wildner {
32*66932323SSascha Wildner if ((number >= MUL_NO_OVERFLOW || size >= MUL_NO_OVERFLOW) &&
33*66932323SSascha Wildner number > 0 && SIZE_MAX / number < size) {
34*66932323SSascha Wildner errno = ENOMEM;
35*66932323SSascha Wildner return NULL;
36*66932323SSascha Wildner }
37*66932323SSascha Wildner return realloc(ptr, size * number);
38*66932323SSascha Wildner }
39