summaryrefslogtreecommitdiff
path: root/src/audio/tracker.cc
blob: 8f3da38b65a2c5c3cb13044f07dbe2aeaaf9f82a (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
#include "tracker.h"
#include "audio.h"
#include "audio/synth.h"
#include "util/asset_manager.h"
#include <cstring>
#include <vector>

static uint32_t g_last_trigger_idx = 0;

struct ManagedSpectrogram {
  int synth_id;
  float* data;
  bool active;
};

// Simple pool for dynamic spectrograms
static ManagedSpectrogram g_spec_pool[MAX_SPECTROGRAMS];
static int g_next_pool_slot = 0; // Round-robin allocation

void tracker_init() {
  g_last_trigger_idx = 0;
  g_next_pool_slot = 0;
  for (int i = 0; i < MAX_SPECTROGRAMS; ++i) {
    g_spec_pool[i].synth_id = -1;
    g_spec_pool[i].data = nullptr;
    g_spec_pool[i].active = false;
  }
}

static int get_free_pool_slot() {
  // Try to find an inactive slot first (unused slots)
  for (int i = 0; i < MAX_SPECTROGRAMS; ++i) {
    if (!g_spec_pool[i].active)
      return i;
  }

  // If all slots are active, reuse the oldest one (round-robin)
  // This automatically handles cleanup of old patterns
  const int slot = g_next_pool_slot;
  g_next_pool_slot = (g_next_pool_slot + 1) % MAX_SPECTROGRAMS;
  return slot;
}

void tracker_update(float time_sec) {
  while (g_last_trigger_idx < g_tracker_score.num_triggers) {
    const TrackerPatternTrigger& trigger =
        g_tracker_score.triggers[g_last_trigger_idx];

    if (trigger.time_sec > time_sec)
      break;

    const TrackerPattern& pattern = g_tracker_patterns[trigger.pattern_id];

    int dest_num_frames = 0;
    std::vector<float> pattern_data;

    float beat_to_sec = 60.0f / g_tracker_score.bpm;

    for (uint32_t i = 0; i < pattern.num_events; ++i) {
      const TrackerEvent& event = pattern.events[i];

      std::vector<float> note_data;
      int note_frames = 0;

      AssetId aid = g_tracker_sample_assets[event.sample_id];
      if (aid != AssetId::ASSET_LAST_ID) {
        size_t size;
        const uint8_t* data = GetAsset(aid, &size);
        if (data && size >= sizeof(SpecHeader)) {
          const SpecHeader* header = (const SpecHeader*)data;
          note_frames = header->num_frames;
          const float* src_spectral_data =
              (const float*)(data + sizeof(SpecHeader));
          note_data.assign(src_spectral_data,
                           src_spectral_data + (size_t)note_frames * DCT_SIZE);
        }
      } else {
        const NoteParams& params = g_tracker_samples[event.sample_id];
        note_data = generate_note_spectrogram(params, &note_frames);
      }

      if (note_frames > 0) {
        int frame_offset =
            (int)(event.beat * beat_to_sec * 32000.0f / DCT_SIZE);
        paste_spectrogram(pattern_data, &dest_num_frames, note_data,
                          note_frames, frame_offset);
      }
    }

    if (dest_num_frames > 0) {
      const int slot = get_free_pool_slot();

      // Clean up old data in this slot if reusing
      if (g_spec_pool[slot].synth_id != -1) {
        synth_unregister_spectrogram(g_spec_pool[slot].synth_id);
        g_spec_pool[slot].synth_id = -1;
      }
      if (g_spec_pool[slot].data != nullptr) {
        delete[] g_spec_pool[slot].data;
        g_spec_pool[slot].data = nullptr;
      }

      // Allocate and register new pattern
      g_spec_pool[slot].data = new float[pattern_data.size()];
      memcpy(g_spec_pool[slot].data, pattern_data.data(),
             pattern_data.size() * sizeof(float));

      Spectrogram spec;
      spec.spectral_data_a = g_spec_pool[slot].data;
      spec.spectral_data_b = g_spec_pool[slot].data;
      spec.num_frames = dest_num_frames;

      g_spec_pool[slot].synth_id = synth_register_spectrogram(&spec);
      g_spec_pool[slot].active = true;

      synth_trigger_voice(g_spec_pool[slot].synth_id, 1.0f, 0.0f);
    }

    g_last_trigger_idx++;
  }
}