From c24044230b89967fa46f077e04226a429737cac9 Mon Sep 17 00:00:00 2001
From: "Meredith L. Patterson" <mlp@thesmartpolitenerd.com>
Date: Thu, 8 Nov 2012 01:36:19 -0500
Subject: [PATCH] Benchmark is actually broken right now, for mysterious
 double-free related reasons. I will dig into this tomorrow.

---
 src/benchmark.c | 22 ++++++++++++++++++++++
 src/hammer.h    | 18 ++++++++++++++++++
 2 files changed, 40 insertions(+)

diff --git a/src/benchmark.c b/src/benchmark.c
index 5d193caa..bf7f2071 100644
--- a/src/benchmark.c
+++ b/src/benchmark.c
@@ -20,23 +20,39 @@
 
 */
 
+#define false 0
+#define true 1
+
+#include <stdlib.h>
 
 HBenchmarkResults *h_benchmark(const HParser* parser, HParserTestcase* testcases) {
   // For now, just output the results to stderr
   HParserTestcase* tc = testcases;
   HParserBackend backend = PB_MIN;
+  HBenchmarkResults *ret = (HBenchmarkResults*)malloc(sizeof(HBenchmarkResults*));
+  ret->len = PB_MAX-PB_MIN;
+  ret->results = (HBackendResults*)malloc(ret->len * sizeof(HBackendResults*));
 
   for (backend = PB_MIN; backend < PB_MAX; backend++) {
     fprintf(stderr, "Compiling for backend %d ... ", backend);
+    ret->results[backend].backend = backend;
     // Step 1: Compile grammar for given parser...
     if (h_compile(parser, PB_MIN, NULL) == -1) {
       // backend inappropriate for grammar...
       fprintf(stderr, "failed\n");
+      ret->results[backend].compile_success = false;
+      ret->results[backend].n_testcases = 0;
+      ret->results[backend].failed_testcases = 0;
+      ret->results[backend].cases = NULL;
       continue;
     }
+    ret->results[backend].compile_success = true;
     int tc_failed = 0;
     // Step 1: verify all test cases.
+    ret->results[backend].n_testcases = 0;
+    ret->results[backend].failed_testcases = 0;
     for (tc = testcases; tc->input != NULL; tc++) {
+      ret->results[backend].n_testcases++;
       HParseResult *res = h_parse(parser, tc->input, tc->length);
       char* res_unamb;
       if (res != NULL) {
@@ -51,6 +67,7 @@ HBenchmarkResults *h_benchmark(const HParser* parser, HParserTestcase* testcases
 	// report. (eg, if users are trying to fix a grammar for a
 	// faster backend)
 	tc_failed++;
+	ret->results[backend].failed_testcases++;
       }
       h_parse_result_free(res);
     }
@@ -61,6 +78,9 @@ HBenchmarkResults *h_benchmark(const HParser* parser, HParserTestcase* testcases
       continue;
     }
 
+    ret->results[backend].cases = (HCaseResult*)malloc(ret->results[backend].n_testcases * sizeof(HCaseResult*));
+    size_t cur_case = 0;
+
     for (tc = testcases; tc->input != NULL; tc++) {
       // The goal is to run each testcase for at least 50ms each
       // TODO: replace this with a posix timer-based benchmark. (cf. timerfd_create, timer_create, setitimer)
@@ -78,7 +98,9 @@ HBenchmarkResults *h_benchmark(const HParser* parser, HParserTestcase* testcases
 	// time_diff is in ns
 	time_diff = (ts_end.tv_sec - ts_start.tv_sec) * 1000000000 + (ts_end.tv_nsec - ts_start.tv_nsec);
       } while (time_diff < 100000000);
+      ret->results[backend].cases[cur_case].parse_time = (time_diff / count);
       fprintf(stderr, "Case %d: %lld ns/parse\n", (int)(tc - testcases),  time_diff / count);
+      cur_case++;
     }
   }
   return NULL;
diff --git a/src/hammer.h b/src/hammer.h
index bf034922..256644cd 100644
--- a/src/hammer.h
+++ b/src/hammer.h
@@ -125,7 +125,25 @@ typedef struct HParserTestcase_ {
   char* output_unambiguous;
 } HParserTestcase;
 
+typedef struct HCaseResult_ {
+  bool success;
+  union {
+    const char* actual_results; // on failure, filled in with the results of h_write_result_unamb
+    size_t parse_time; // on success, filled in with time for a single parse, in nsec
+  };
+} HCaseResult;
+
+typedef struct HBackendResults_ {
+  HParserBackend backend;
+  bool compile_success;
+  size_t n_testcases;
+  size_t failed_testcases; // actually a count...
+  HCaseResult *cases;
+} HBackendResults;
+
 typedef struct HBenchmarkResults_ {
+  size_t len;
+  HBackendResults *results;
 } HBenchmarkResults;
 // }}}
 
-- 
GitLab