/* emalloc.c */
#ifndef EREALLOC_CALLSITE /* ntp_malloc.h defines */
extern void * ereallocz (void *, size_t, size_t, int);
-#define erealloczsite(p, n, o, z, f, l) ereallocz(p, n, o, (z))
-#define emalloc(n) ereallocz(NULL, n, 0, FALSE)
+extern void * oreallocarray (void *optr, size_t nmemb, size_t size);
+#define erealloczsite(p, n, o, z, f, l) ereallocz((p), (n), (o), (z))
+#define emalloc(n) ereallocz(NULL, (n), 0, FALSE)
#define emalloc_zero(c) ereallocz(NULL, (c), 0, TRUE)
-#define erealloc(p, c) ereallocz(p, (c), 0, FALSE)
-#define erealloc_zero(p, n, o) ereallocz(p, n, (o), TRUE)
-extern char * estrdup_impl (const char *);
+#define erealloc(p, c) ereallocz((p), (c), 0, FALSE)
+#define erealloc_zero(p, n, o) ereallocz((p), (n), (o), TRUE)
+#define ereallocarray(p, n, s) oreallocarray((p), (n), (s))
+#define eallocarray(n, s) oreallocarray(NULL, (n), (s))
+extern char * estrdup_impl(const char *);
#define estrdup(s) estrdup_impl(s)
#else
extern void * ereallocz (void *, size_t, size_t, int,
const char *, int);
+extern void * oreallocarray (void *optr, size_t nmemb, size_t size,
+ const char *, int);
#define erealloczsite ereallocz
#define emalloc(c) ereallocz(NULL, (c), 0, FALSE, \
__FILE__, __LINE__)
#define emalloc_zero(c) ereallocz(NULL, (c), 0, TRUE, \
__FILE__, __LINE__)
-#define erealloc(p, c) ereallocz(p, (c), 0, FALSE, \
+#define erealloc(p, c) ereallocz((p), (c), 0, FALSE, \
+ __FILE__, __LINE__)
+#define erealloc_zero(p, n, o) ereallocz((p), (n), (o), TRUE, \
+ __FILE__, __LINE__)
+#define ereallocarray(p, n, s) oreallocarray((p), (n), (s), \
__FILE__, __LINE__)
-#define erealloc_zero(p, n, o) ereallocz(p, n, (o), TRUE, \
+#define eallocarray(n, s) oreallocarray(NULL, (n), (s), \
__FILE__, __LINE__)
-extern char * estrdup_impl (const char *, const char *, int);
+extern char * estrdup_impl(const char *, const char *, int);
#define estrdup(s) estrdup_impl((s), __FILE__, __LINE__)
#endif
return mem;
}
+/* oreallocarray.c is licensed under the following:
+ * Copyright (c) 2008 Otto Moerbeek <otto@drijf.net>
+ *
+ * Permission to use, copy, modify, and distribute this software for any
+ * purpose with or without fee is hereby granted, provided that the above
+ * copyright notice and this permission notice appear in all copies.
+ *
+ * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
+ * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
+ * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
+ * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
+ * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
+ * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
+ * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
+ */
+
+#include <stdint.h>
+
+/*
+ * This is sqrt(SIZE_MAX+1), as s1*s2 <= SIZE_MAX
+ * if both s1 < MUL_NO_OVERFLOW and s2 < MUL_NO_OVERFLOW
+ */
+#define MUL_NO_OVERFLOW ((size_t)1 << (sizeof(size_t) * 4))
+
+void *
+oreallocarray(
+ void *optr,
+ size_t nmemb,
+ size_t size
+#ifdef EREALLOC_CALLSITE /* ntp_malloc.h */
+ ,
+ const char * file,
+ int line
+#endif
+ )
+{
+ if ((nmemb >= MUL_NO_OVERFLOW || size >= MUL_NO_OVERFLOW) &&
+ nmemb > 0 && SIZE_MAX / nmemb < size) {
+#ifndef EREALLOC_CALLSITE
+ msyslog(LOG_ERR, "fatal allocation size overflow");
+#else
+ msyslog(LOG_ERR,
+ "fatal allocation size overflow %s line %d",
+ file, line);
+#endif
+ exit(1);
+ }
+#ifndef EREALLOC_CALLSITE
+ return ereallocz(optr, (size * nmemb), 0, FALSE);
+#else
+ return ereallocz(optr, (size * nmemb), 0, FALSE, file, line);
+#endif
+}
char *
estrdup_impl(