summaryrefslogtreecommitdiffhomepage
path: root/main.cpp
blob: 7d423f6fc8ea95a9e6f68c3377df1e95adcc90f1 (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
#include "Click.h"
#include "MIDI.h"
#include "PCM.h"
#include "Timer.h"
#include "UI.h"
#include "config.h"
#include "log.h"

#include <chrono>
#include <cmath>
#include <cstdint>
#include <exception>
#include <iostream>
#include <limits>
#include <memory>
#include <stdexcept>

#include <stdio.h>
#include <sys/types.h>
#include <sys/time.h>
#include <time.h>

#include <boost/signals2.hpp>

using namespace std::chrono_literals;
using namespace std::string_literals;

double diff_timespec(const struct timespec *time1, const struct timespec *time0) {
  return (time1->tv_sec - time0->tv_sec)
      + (time1->tv_nsec - time0->tv_nsec) / 1000000000.0;
}

int main(void)
{
  try {
    //debug_cout.activate();
    log_cout.activate();
    log_cout.log_lines(log_lines);

    Config config;

    std::shared_ptr<ClockClick> clock_click = std::make_shared<ClockClick>();
    std::shared_ptr<NoteClick> note_click = std::make_shared<NoteClick>(config);
    std::shared_ptr<InternalClick> internal_click = std::make_shared<InternalClick>(config);

    // Active Mode
    std::shared_ptr<Click> click = note_click;

    MIDI midi;
    PCM pcm;
    UI ui;

    pcm.write();

    Timer timer_500ms(500ms, true);
    timer_500ms.start();

    // Main signals
    boost::signals2::signal<void()> signal_count_loops;

    //
    // Signal-Slot Connections:
    //
    midi.signal_note.connect([&](int channel, int note){note_click->receive_note(channel, note);});
    note_click->signal_click.connect([&](){pcm.click();});
    midi.signal_active_sensing.connect([&](){ui.slot_active_sensing();});
    timer_500ms.elapsed.connect([&](){ui.draw();});
    signal_count_loops.connect([&](){ui.count_main_loops();});

    midi.flush();

    while (true) {
      debug_cout << "Main loop entered." << std::endl;
      signal_count_loops();

      fd_set read_set;
      FD_ZERO(&read_set);
      FD_SET(midi.fd(), &read_set);

#if 0
      // PCM fd almost always writeable: for single frames at high speed
      fd_set write_set;
      FD_ZERO(&write_set);
      FD_SET(pcm.fd(), &write_set);
#endif

      struct timeval timeout;
      timeout.tv_sec = 0;
      timeout.tv_usec = 20000;

      int result = select(FD_SETSIZE, &read_set, nullptr/*&write_set*/, nullptr, &timeout);
      if (result < 0) {
        throw std::runtime_error("select() failed");
      } else if (result == 0) {
        debug_cout << "select() timeout" << std::endl;
      }

      while (midi.event_ready())
      {
        //std::cout << "read..." << std::endl;
        auto event = midi.read();
        //std::cout << "process..." << std::endl;
        midi.process(event);
      }

      if (pcm.write_available()) {
        //std::cout << "DEBUG: WRITE" << std::endl;
        pcm.write();
      }

      // handle timers, TODO: make updates more efficient at scale
      timer_500ms.update();
    }
  } catch (const std::exception& ex) {
    std::cerr << "Error: " << ex.what() << std::endl;
  }
 
  return 0;
}