aboutsummaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--src/benchmarks/blowup.py62
-rw-r--r--src/benchmarks/blowup/Makefile25
-rw-r--r--src/benchmarks/blowup/blowup.c90
3 files changed, 177 insertions, 0 deletions
diff --git a/src/benchmarks/blowup.py b/src/benchmarks/blowup.py
new file mode 100644
index 0000000..ad033b3
--- /dev/null
+++ b/src/benchmarks/blowup.py
@@ -0,0 +1,62 @@
+# Copyright 2018-2019 Florian Fischer <florian.fl.fischer@fau.de>
+#
+# This file is part of allocbench.
+#
+# allocbench is free software: you can redistribute it and/or modify
+# it under the terms of the GNU General Public License as published by
+# the Free Software Foundation, either version 3 of the License, or
+# (at your option) any later version.
+#
+# allocbench is distributed in the hope that it will be useful,
+# but WITHOUT ANY WARRANTY; without even the implied warranty of
+# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+# GNU General Public License for more details.
+#
+# You should have received a copy of the GNU General Public License
+# along with allocbench. If not, see <http://www.gnu.org/licenses/>.
+
+"""Definition of the blowup micro benchmark"""
+
+from src.benchmark import Benchmark
+
+
+class BenchmarkBlowup(Benchmark):
+ """Blowup micro benchmark
+
+ Check if allocators reuse memory.
+ Thread behaviour:
+ 1. Do many allocations and frees (100MiB is the maximum of live allocations at all times)
+ 2. Start new thread
+ 3. Join new thread
+ A perfect reusing allocator would need 100MiB RSS and a non reusing allocator
+ would need 1GiB.
+ """
+ def __init__(self):
+ name = "blowup"
+
+ self.cmd = "blowup"
+
+ self.requirements = ["blowup"]
+ super().__init__(name)
+
+ def summary(self):
+ # hack ideal rss in data set
+ allocators = self.results["allocators"]
+ allocators["Ideal_RSS"] = {"color": "C" + str(len(allocators) + 1)}
+ self.results["stats"]["Ideal_RSS"] = {}
+ for perm in self.iterate_args(args=self.results["args"]):
+ self.results["stats"]["Ideal_RSS"][perm] = {"mean": {"VmHWM": 1024 * 100},
+ "std" : {"VmHWM": 0}}
+
+ self.barplot_single_arg("{VmHWM}",
+ ylabel='"VmHWM in KB"',
+ title='"realloc micro benchmark"',
+ filepostfix="vmhwm")
+
+ del allocators["Ideal_RSS"]
+ del self.results["stats"]["Ideal_RSS"]
+
+ self.export_stats_to_dataref("VmHWM")
+
+
+blowup = BenchmarkBlowup()
diff --git a/src/benchmarks/blowup/Makefile b/src/benchmarks/blowup/Makefile
new file mode 100644
index 0000000..afb5f2a
--- /dev/null
+++ b/src/benchmarks/blowup/Makefile
@@ -0,0 +1,25 @@
+OBJDIR ?= obj
+
+CC ?= gcc
+
+WARNFLAGS ?= -Wall -Wextra
+COMMONFLAGS ?= -fno-builtin -fPIC -DPIC -pthread
+OPTFLAGS ?= -O3 -DNDEBUG
+
+CFLAGS ?= $(OPTFLAGS) $(WARNFLAGS) $(COMMONFLAGS)
+
+LDFLAGS ?= -pthread -static-libgcc
+
+.PHONY = all clean
+
+all: $(OBJDIR)/blowup
+
+$(OBJDIR)/blowup: blowup.c | $(OBJDIR)
+ @echo compiling $@...
+ $(CC) $(LDFLAGS) $(CFLAGS) -o $@ $<
+
+$(OBJDIR):
+ mkdir -p $@
+
+clean:
+ rm -rf $(OBJDIR)
diff --git a/src/benchmarks/blowup/blowup.c b/src/benchmarks/blowup/blowup.c
new file mode 100644
index 0000000..defb58c
--- /dev/null
+++ b/src/benchmarks/blowup/blowup.c
@@ -0,0 +1,90 @@
+#include <pthread.h>
+#include <stdio.h>
+#include <stdlib.h>
+#include <string.h>
+
+#define NUM_THREADS 10
+#define ALLOCATIONS 1024 * 100
+#define SIZE 1024
+
+static void do_work()
+{
+ void** ptrs;
+
+ ptrs = malloc(sizeof(void*) * ALLOCATIONS);
+ if (ptrs == NULL) {
+ perror("malloc");
+ return;
+ }
+
+ for(int i = 0; i < ALLOCATIONS; i++)
+ {
+ ptrs[i] = malloc(SIZE);
+ memset(ptrs[i], 0, SIZE);
+ }
+
+ for(int i = 0; i < ALLOCATIONS; i++)
+ {
+ free(malloc(SIZE));
+ }
+
+ for(int i = 0; i < ALLOCATIONS; i++)
+ {
+ /* memset(ptrs[i], 0, SIZE); */
+ free(ptrs[i]);
+ }
+
+ for(int i = 0; i < ALLOCATIONS; i++)
+ {
+ free(malloc(SIZE));
+ }
+
+ return;
+}
+
+static void* thread_func(void* arg)
+{
+ unsigned int* thread_id = (unsigned int*)arg;
+ pthread_t next_thread;
+
+ printf("thread %d doing work\n", *thread_id);
+ do_work();
+
+ if (*thread_id == NUM_THREADS)
+ return NULL;
+
+ printf("thread %d spawning new thread work\n", *thread_id);
+ *thread_id = (*thread_id) + 1;
+ if (0 != pthread_create(&next_thread, NULL, thread_func, arg)) {
+ perror("pthread_create");
+ return NULL;
+ }
+
+ printf("thread %d joining thread %d work\n", *thread_id-1, *thread_id);
+ if (0 != pthread_join(next_thread, NULL)) {
+ perror("pthread_join");
+ return NULL;
+ }
+
+ return NULL;
+}
+
+int main()
+{
+ unsigned int thread_count = 0;
+ pthread_t first;
+
+ do_work();
+
+ if (0 != pthread_create(&first, NULL, thread_func, &thread_count)) {
+ perror("pthread_create");
+ return 1;
+ }
+
+ if (0 != pthread_join(first, NULL)) {
+ perror("pthread_join");
+ return 1;
+ }
+
+ return 0;
+}