The beatmaker runs on its own thread. It implements a Pub/Sub model via the use of mpsc for each subscriber.
I have thought about other design choices about beatmaker.
For example, a naive approach would be a callback mechanism, in which the user passes a callback function that gets called upon every beat. The problem is that in JACK framework, your main process code is also a callback function which is called on every process cycle. You don't get to do something at any time of your discretion.
With the Pub/Sub model, I can get a receiver channel of beatmaker's MIDI events, which I can drain on every process cycle.
I have recently implemented tempo scaling, which allows relative scaling of a single track relative to the global tempo. It is currently done by introducing BeatSorter, which maintains a search tree that sorts the upcoming notes from each track by their absolute position in the unit of global beat time. On every tick BeatMaker receives from Timeline, it pops notes from BeatSorter whose beat time is earlier than the current global beat time.