xref: /openbsd-src/lib/libc/stdlib/reallocarray.c (revision 0d943ef019900239de431dfc98899a9f48f183de)
1*0d943ef0Sguenther /*	$OpenBSD: reallocarray.c,v 1.3 2015/09/13 08:31:47 guenther Exp $	*/
291c2a707Sderaadt /*
391c2a707Sderaadt  * Copyright (c) 2008 Otto Moerbeek <otto@drijf.net>
491c2a707Sderaadt  *
591c2a707Sderaadt  * Permission to use, copy, modify, and distribute this software for any
691c2a707Sderaadt  * purpose with or without fee is hereby granted, provided that the above
791c2a707Sderaadt  * copyright notice and this permission notice appear in all copies.
891c2a707Sderaadt  *
991c2a707Sderaadt  * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
1091c2a707Sderaadt  * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
1191c2a707Sderaadt  * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
1291c2a707Sderaadt  * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
1391c2a707Sderaadt  * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
1491c2a707Sderaadt  * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
1591c2a707Sderaadt  * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
1691c2a707Sderaadt  */
1791c2a707Sderaadt 
1891c2a707Sderaadt #include <sys/types.h>
1991c2a707Sderaadt #include <errno.h>
2091c2a707Sderaadt #include <stdint.h>
2191c2a707Sderaadt #include <stdlib.h>
2291c2a707Sderaadt 
2391c2a707Sderaadt /*
2491c2a707Sderaadt  * This is sqrt(SIZE_MAX+1), as s1*s2 <= SIZE_MAX
2591c2a707Sderaadt  * if both s1 < MUL_NO_OVERFLOW and s2 < MUL_NO_OVERFLOW
2691c2a707Sderaadt  */
27aaba0d34Sbcook #define MUL_NO_OVERFLOW	((size_t)1 << (sizeof(size_t) * 4))
2891c2a707Sderaadt 
2991c2a707Sderaadt void *
reallocarray(void * optr,size_t nmemb,size_t size)3091c2a707Sderaadt reallocarray(void *optr, size_t nmemb, size_t size)
3191c2a707Sderaadt {
3291c2a707Sderaadt 	if ((nmemb >= MUL_NO_OVERFLOW || size >= MUL_NO_OVERFLOW) &&
3391c2a707Sderaadt 	    nmemb > 0 && SIZE_MAX / nmemb < size) {
3491c2a707Sderaadt 		errno = ENOMEM;
3591c2a707Sderaadt 		return NULL;
3691c2a707Sderaadt 	}
3791c2a707Sderaadt 	return realloc(optr, size * nmemb);
3891c2a707Sderaadt }
39*0d943ef0Sguenther DEF_WEAK(reallocarray);
40