[otbn] Move trace collation to the otbn_tracer core file

This code shouldn't be in otbn_top_sim.cc: it's going to be needed for
the UVM-based test machinery too. This patch defines a new class
called OtbnTraceSource. It's a singleton, whose instance listens to a
DPI function and broadcasts calls to all listeners.

Signed-off-by: Rupert Swarbrick <rswarbrick@lowrisc.org>
diff --git a/hw/ip/otbn/dv/tracer/cpp/otbn_trace_source.cc b/hw/ip/otbn/dv/tracer/cpp/otbn_trace_source.cc
new file mode 100644
index 0000000..76fba15
--- /dev/null
+++ b/hw/ip/otbn/dv/tracer/cpp/otbn_trace_source.cc
@@ -0,0 +1,41 @@
+// Copyright lowRISC contributors.
+// Licensed under the Apache License, Version 2.0, see LICENSE for details.
+// SPDX-License-Identifier: Apache-2.0
+
+#include "otbn_trace_source.h"
+
+#include <algorithm>
+#include <cassert>
+#include <memory>
+
+static std::unique_ptr<OtbnTraceSource> trace_source;
+
+OtbnTraceSource &OtbnTraceSource::get() {
+  if (!trace_source) {
+    trace_source.reset(new OtbnTraceSource());
+  }
+  return *trace_source;
+}
+
+void OtbnTraceSource::AddListener(OtbnTraceListener *listener) {
+  listeners_.push_back(listener);
+}
+
+void OtbnTraceSource::RemoveListener(const OtbnTraceListener *listener) {
+  auto it = std::find(listeners_.begin(), listeners_.end(), listener);
+  assert(it != listeners_.end());
+  listeners_.erase(it);
+}
+
+void OtbnTraceSource::Broadcast(const std::string &trace,
+                                unsigned cycle_count) {
+  for (OtbnTraceListener *listener : listeners_) {
+    listener->AcceptTraceString(trace, cycle_count);
+  }
+}
+
+extern "C" void accept_otbn_trace_string(const char *trace,
+                                         unsigned int cycle_count) {
+  assert(trace != nullptr);
+  OtbnTraceSource::get().Broadcast(trace, cycle_count);
+}
diff --git a/hw/ip/otbn/dv/tracer/cpp/otbn_trace_source.h b/hw/ip/otbn/dv/tracer/cpp/otbn_trace_source.h
new file mode 100644
index 0000000..534c1cb
--- /dev/null
+++ b/hw/ip/otbn/dv/tracer/cpp/otbn_trace_source.h
@@ -0,0 +1,36 @@
+// Copyright lowRISC contributors.
+// Licensed under the Apache License, Version 2.0, see LICENSE for details.
+// SPDX-License-Identifier: Apache-2.0
+
+#pragma once
+
+#include <vector>
+
+#include "otbn_trace_listener.h"
+
+// A source for simulation trace data.
+//
+// This is a singleton class, which will be constructed on the first call to
+// get() or the first trace data that comes back from the simulation.
+//
+// The object is in charge of taking trace data from the simulation (which is
+// sent by calling the accept_otbn_trace_string DPI function) and passing it
+// out to registered listeners.
+
+class OtbnTraceSource {
+ public:
+  // Get the (singleton) OtbnTraceSource object
+  static OtbnTraceSource &get();
+
+  // Add a listener to the source
+  void AddListener(OtbnTraceListener *listener);
+
+  // Remove a listener from the source
+  void RemoveListener(const OtbnTraceListener *listener);
+
+  // Send a trace string to all listeners
+  void Broadcast(const std::string &trace, unsigned cycle_count);
+
+ private:
+  std::vector<OtbnTraceListener *> listeners_;
+};
diff --git a/hw/ip/otbn/dv/tracer/otbn_tracer.core b/hw/ip/otbn/dv/tracer/otbn_tracer.core
index f6f420f..89f7ebe 100644
--- a/hw/ip/otbn/dv/tracer/otbn_tracer.core
+++ b/hw/ip/otbn/dv/tracer/otbn_tracer.core
@@ -11,6 +11,8 @@
       - lowrisc:ip:otbn_pkg
     files:
       - cpp/otbn_trace_listener.h: { is_include_file: true, file_type: cppSource }
+      - cpp/otbn_trace_source.h: { is_include_file: true, file_type: cppSource }
+      - cpp/otbn_trace_source.cc: { file_type: cppSource }
       - cpp/log_trace_listener.h: { is_include_file: true, file_type: cppSource }
       - cpp/log_trace_listener.cc: { file_type: cppSource }
       - rtl/otbn_tracer.sv: { file_type: systemVerilogSource }
diff --git a/hw/ip/otbn/dv/verilator/otbn_top_sim.cc b/hw/ip/otbn/dv/verilator/otbn_top_sim.cc
index 58ffabe..286253d 100644
--- a/hw/ip/otbn/dv/verilator/otbn_top_sim.cc
+++ b/hw/ip/otbn/dv/verilator/otbn_top_sim.cc
@@ -2,56 +2,26 @@
 // Licensed under the Apache License, Version 2.0, see LICENSE for details.
 // SPDX-License-Identifier: Apache-2.0
 
-#include <algorithm>
-#include <cassert>
 #include <fstream>
 #include <getopt.h>
 #include <iomanip>
 #include <iostream>
 #include <memory>
-#include <sstream>
 #include <string>
 #include <svdpi.h>
-#include <vector>
 
 #include "log_trace_listener.h"
 #include "otbn_memutil.h"
-#include "otbn_trace_listener.h"
+#include "otbn_trace_source.h"
 #include "verilated_toplevel.h"
 #include "verilator_memutil.h"
 #include "verilator_sim_ctrl.h"
 
-std::vector<OtbnTraceListener *> otbn_trace_listeners;
-
-void AddOtbnTraceListener(OtbnTraceListener *l) {
-  otbn_trace_listeners.push_back(l);
-}
-
-void RemoveOtbnTraceListener(OtbnTraceListener *l) {
-  auto l_iter =
-      std::find(otbn_trace_listeners.begin(), otbn_trace_listeners.end(), l);
-
-  if (l_iter != otbn_trace_listeners.end()) {
-    otbn_trace_listeners.erase(l_iter);
-  }
-}
-
 extern "C" {
 extern unsigned int otbn_base_call_stack_get_size();
 extern unsigned int otbn_base_call_stack_get_element(int index);
 extern unsigned int otbn_base_reg_get(int index);
 extern unsigned int otbn_bignum_reg_get(int index, int quarter);
-
-extern void accept_otbn_trace_string(const char *trace,
-                                     unsigned int cycle_count) {
-  assert(trace != nullptr);
-
-  std::string trace_str(trace);
-
-  for (auto l : otbn_trace_listeners) {
-    l->AcceptTraceString(trace, cycle_count);
-  }
-}
 }
 
 /**
@@ -66,7 +36,7 @@
   bool SetupTraceLog(const std::string &log_filename) {
     try {
       log_trace_listener_ = std::make_unique<LogTraceListener>(log_filename);
-      AddOtbnTraceListener(log_trace_listener_.get());
+      OtbnTraceSource::get().AddListener(log_trace_listener_.get());
       return true;
     } catch (const std::runtime_error &err) {
       std::cerr << "ERROR: Failed to set up trace log: " << err.what()
@@ -113,7 +83,10 @@
     return true;
   }
 
-  ~OtbnTraceUtil() { RemoveOtbnTraceListener(log_trace_listener_.get()); }
+  ~OtbnTraceUtil() {
+    if (log_trace_listener_)
+      OtbnTraceSource::get().RemoveListener(log_trace_listener_.get());
+  }
 };
 
 int main(int argc, char **argv) {