about summary refs log tree commit diff
path: root/src/malloc/calloc.c
diff options
context:
space:
mode:
authorRich Felker <dalias@aerifal.cx>2020-06-10 20:44:51 -0400
committerRich Felker <dalias@aerifal.cx>2020-06-10 20:51:17 -0400
commit25cef5c591fbee755a53f0d7920f4f554f343a53 (patch)
tree13b38ac1a7522dcc2f988407187a98bc0612a888 /src/malloc/calloc.c
parent501a92660ca2ed6469a0a01474574ce9930356cd (diff)
downloadmusl-25cef5c591fbee755a53f0d7920f4f554f343a53.tar.gz
musl-25cef5c591fbee755a53f0d7920f4f554f343a53.tar.xz
musl-25cef5c591fbee755a53f0d7920f4f554f343a53.zip
reintroduce calloc elison of memset for direct-mmapped allocations
a new weak predicate function replacable by the malloc implementation,
__malloc_allzerop, is introduced. by default it's always false; the
default version will be used when static linking if the bump allocator
was used (in which case performance doesn't matter) or if malloc was
replaced by the application. only if the real internal malloc is
linked (always the case with dynamic linking) does the real version
get used.

if malloc was replaced dynamically, as indicated by __malloc_replaced,
the predicate function is ignored and conditional-memset is always
performed.
Diffstat (limited to 'src/malloc/calloc.c')
-rw-r--r--src/malloc/calloc.c10
1 files changed, 9 insertions, 1 deletions
diff --git a/src/malloc/calloc.c b/src/malloc/calloc.c
index 322193ca..bf6bddca 100644
--- a/src/malloc/calloc.c
+++ b/src/malloc/calloc.c
@@ -2,6 +2,7 @@
 #include <stdint.h>
 #include <string.h>
 #include <errno.h>
+#include "dynlink.h"
 
 static size_t mal0_clear(char *p, size_t n)
 {
@@ -23,6 +24,12 @@ static size_t mal0_clear(char *p, size_t n)
 	}
 }
 
+static int allzerop(void *p)
+{
+	return 0;
+}
+weak_alias(allzerop, __malloc_allzerop);
+
 void *calloc(size_t m, size_t n)
 {
 	if (n && m > (size_t)-1/n) {
@@ -31,7 +38,8 @@ void *calloc(size_t m, size_t n)
 	}
 	n *= m;
 	void *p = malloc(n);
-	if (!p) return p;
+	if (!p || (!__malloc_replaced && __malloc_allzerop(p)))
+		return p;
 	n = mal0_clear(p, n);
 	return memset(p, 0, n);
 }