summaryrefslogtreecommitdiff
path: root/midi.c
blob: 3d21f794ff4057bf367513fb80f8cdb9c85faccb (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
#include <stdio.h>
#include <stdlib.h>
#include <alloca.h>
#include <unistd.h>
#include <alsa/asoundlib.h>

#include "midi.h"

static snd_seq_t *seq_handle;
static snd_seq_addr_t *ports;

void *midi(void *arg) {
	MidiArgs* args = (MidiArgs*)arg;

	while (1) {
		if (snd_seq_open(&seq_handle, "default", SND_SEQ_OPEN_DUPLEX, 0) < 0) {
			fprintf(stderr, "Error opening ALSA sequencer.\n");
			exit(1);
		}

		if (snd_seq_set_client_name(seq_handle, CLIENT_NAME) < 0) {
			fprintf(stderr, "Could not set up client name.\n");
			exit(1);
		}

		if (snd_seq_create_simple_port(seq_handle, CLIENT_NAME,
					SND_SEQ_PORT_CAP_WRITE |
					SND_SEQ_PORT_CAP_SUBS_WRITE,
					SND_SEQ_PORT_TYPE_MIDI_GENERIC |
					SND_SEQ_PORT_TYPE_APPLICATION) < 0) {
			fprintf(stderr, "Error creating sequencer port.\n");
			exit(1);
		}

		// Connecting ports.
		ports = realloc(ports, MAX_MIDI_PORTS * sizeof(snd_seq_addr_t));
		if (snd_seq_parse_address(seq_handle, &ports[0], args->port_name) < 0) {
			fprintf(stderr, "Invalid port %s.\n", args->port_name);
			exit(1);
		}

		// Listing assigned ports.
		fprintf(stdout, "Ports:\n");
		for (int j = 0; j < MAX_MIDI_PORTS; j++) {
			fprintf(stdout, " client: %d, port: %d\n", ports[j].client, ports[j].port);
		}

		// Connecting ports.
		for (int i = 0; i < MAX_MIDI_PORTS; ++i) {
			int err = snd_seq_connect_from(seq_handle, 0, ports[i].client, ports[i].port);
			if (err < 0) {
				fprintf(stderr, "Cannot connect from port %d:%d - %s", ports[i].client, ports[i].port, snd_strerror(err));
				exit(1);
			}
		}

		if (snd_seq_nonblock(seq_handle, 1) < 0) {
			fprintf(stderr, "Set nonblock mode failed.");
			exit(1);
		}


		// Reading MIDI device.
		struct pollfd *pfds;
		size_t npfds;

		npfds = snd_seq_poll_descriptors_count(seq_handle, POLLIN);
		pfds = alloca(sizeof(*pfds) * npfds);

		for (;;) {
			snd_seq_poll_descriptors(seq_handle, pfds, npfds, POLLIN);
			if (poll(pfds, npfds, -1) < 0) {
				break;
			}

			for (;;) {
				snd_seq_event_t *ev;

				if (snd_seq_event_input(seq_handle, &ev) < 0) {
					break;
				}

				if (ev) {
					switch (ev->type) {
						case SND_SEQ_EVENT_NOTEON:
							printf("%3d:%-3dNote on  %2d, note %d, velocity: %3d\n",
									ev->source.client, ev->source.port,
									ev->data.note.channel,
									ev->data.note.note,
									ev->data.note.velocity);
							break;

						case SND_SEQ_EVENT_NOTEOFF:
							printf("%3d:%-3dNote off\t%2d, note %d\n",
									ev->source.client, ev->source.port,
									ev->data.note.channel,
									ev->data.note.note);
							break;


					}
				}
			}

			fflush(stdout);
		}
	}
}