4698cb9131c795352289004d5be0fb1ac8afeba8
[ardour.git] / libs / plugins / reasonablesynth.lv2 / lv2.c
1 /* reasonable simple synth
2  *
3  * Copyright (C) 2013 Robin Gareus <robin@gareus.org>
4  *
5  * This program is free software; you can redistribute it and/or modify
6  * it under the terms of the GNU General Public License as published by
7  * the Free Software Foundation; either version 2, or (at your option)
8  * any later version.
9  *
10  * This program is distributed in the hope that it will be useful,
11  * but WITHOUT ANY WARRANTY; without even the implied warranty of
12  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
13  * GNU General Public License for more details.
14  *
15  * You should have received a copy of the GNU General Public License
16  * along with this program; if not, write to the Free Software Foundation,
17  * Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
18  */
19
20 #define _GNU_SOURCE
21
22 #include <stdio.h>
23 #include <stdlib.h>
24 #include <stdint.h>
25 #include <time.h>
26
27 /* LV2 */
28 #include "lv2/lv2plug.in/ns/lv2core/lv2.h"
29 #include "lv2/lv2plug.in/ns/ext/atom/atom.h"
30 #include "lv2/lv2plug.in/ns/ext/urid/urid.h"
31 #include "lv2/lv2plug.in/ns/ext/midi/midi.h"
32
33 #define RSY_URI "https://community.ardour.org/node/7596"
34
35 /* the synth interface */
36 static void *   synth_alloc      (void);
37 static void     synth_init       (void *, double rate);
38 static void     synth_free       (void *);
39 static void     synth_parse_midi (void *, const uint8_t *data, const size_t size);
40 static uint32_t synth_sound      (void *, uint32_t written, uint32_t nframes, float **out);
41
42 #include "rsynth.c"
43
44 typedef enum {
45   RSY_MIDIIN = 0,
46   RSY_OUTL,
47   RSY_OUTR
48 } PortIndex;
49
50 typedef struct {
51   const LV2_Atom_Sequence* midiin;
52   float* outL;
53   float* outR;
54
55   LV2_URID_Map* map;
56   LV2_URID midi_MidiEvent;
57
58   double SampleRateD;
59   void *synth;
60   bool xmas;
61 } RSynth;
62
63 /* main LV2 */
64
65 static LV2_Handle
66 instantiate(const LV2_Descriptor*     descriptor,
67             double                    rate,
68             const char*               bundle_path,
69             const LV2_Feature* const* features)
70 {
71   (void) descriptor; /* unused variable */
72   (void) bundle_path; /* unused variable */
73
74   if (rate < 8000) {
75     fprintf(stderr, "RSynth.lv2 error: unsupported sample-rate (must be > 8k)\n");
76     return NULL;
77   }
78   RSynth* self = (RSynth*)calloc(1, sizeof(RSynth));
79   if(!self) {
80     return NULL;
81   }
82
83   self->SampleRateD = rate;
84
85   int i;
86   for (i=0; features[i]; ++i) {
87     if (!strcmp(features[i]->URI, LV2_URID__map)) {
88       self->map = (LV2_URID_Map*)features[i]->data;
89     }
90   }
91
92   if (!self->map) {
93     fprintf(stderr, "RSynth.lv2 error: Host does not support urid:map\n");
94     free(self);
95     return NULL;
96   }
97
98   self->midi_MidiEvent = self->map->map(self->map->handle, LV2_MIDI__MidiEvent);
99
100   self->synth = synth_alloc();
101   synth_init(self->synth, rate);
102
103 #ifndef PLATFORM_WINDOWS // easter egg is for sane platforms with native support for localtime_r only
104   struct tm date;
105   time_t now;
106   time(&now);
107   localtime_r(&now, &date);
108   if (getenv("ITSXMAS") || (date.tm_mon == 11 /*dec*/ && date.tm_mday == 25)) {
109     printf("reasonable synth.lv2 says: happy holidays!\n");
110     self->xmas = true;
111   }
112 #endif
113
114   return (LV2_Handle)self;
115 }
116
117 static void
118 connect_port(LV2_Handle handle,
119              uint32_t   port,
120              void*      data)
121 {
122   RSynth* self = (RSynth*)handle;
123
124   switch ((PortIndex)port) {
125     case RSY_MIDIIN:
126       self->midiin = (const LV2_Atom_Sequence*)data;
127       break;
128     case RSY_OUTL:
129       self->outL = (float*)data;
130       break;
131     case RSY_OUTR:
132       self->outR = (float*)data;
133       break;
134   }
135 }
136
137 static void
138 run(LV2_Handle handle, uint32_t n_samples)
139 {
140   RSynth* self = (RSynth*)handle;
141   float* audio[2];
142
143   audio[0] = self->outL;
144   audio[1] = self->outR;
145
146   uint32_t written = 0;
147
148   /* Process incoming MIDI events */
149   if (self->midiin) {
150     LV2_Atom_Event const* ev = (LV2_Atom_Event const*)((uintptr_t)((&(self->midiin)->body) + 1)); // lv2_atom_sequence_begin
151     while( // !lv2_atom_sequence_is_end
152         (const uint8_t*)ev < ((const uint8_t*) &(self->midiin)->body + (self->midiin)->atom.size)
153         )
154     {
155       if (ev->body.type == self->midi_MidiEvent) {
156         if (written + BUFFER_SIZE_SAMPLES < ev->time.frames
157             && ev->time.frames < n_samples) {
158           /* first synthesize sound up until the message timestamp */
159           written = synth_sound(self->synth, written, ev->time.frames, audio);
160         }
161         /* send midi message to synth */
162         if (self->xmas) {
163           synth_parse_xmas(self->synth, (const uint8_t*)(ev+1), ev->body.size);
164         } else {
165           synth_parse_midi(self->synth, (const uint8_t*)(ev+1), ev->body.size);
166         }
167       }
168       ev = (LV2_Atom_Event const*) // lv2_atom_sequence_next()
169               ((uintptr_t)((const uint8_t*)ev + sizeof(LV2_Atom_Event) + ((ev->body.size + 7) & ~7)));
170     }
171   }
172
173   /* synthesize [remaining] sound */
174   synth_sound(self->synth, written, n_samples, audio);
175 }
176
177 static void
178 cleanup(LV2_Handle handle)
179 {
180   RSynth* self = (RSynth*)handle;
181   synth_free(self->synth);
182   free(handle);
183 }
184
185 static const void*
186 extension_data(const char* uri)
187 {
188   (void) uri; /* unused variable */
189   return NULL;
190 }
191
192 static const LV2_Descriptor descriptor = {
193   RSY_URI,
194   instantiate,
195   connect_port,
196   NULL,
197   run,
198   NULL,
199   cleanup,
200   extension_data
201 };
202
203 #if defined(COMPILER_MSVC)
204 __declspec(dllexport)
205 #else
206 __attribute__ ((visibility ("default")))
207 #endif
208 const LV2_Descriptor*
209 lv2_descriptor(uint32_t idx)
210 {
211   switch (idx) {
212   case 0:
213     return &descriptor;
214   default:
215     return NULL;
216   }
217 }
218
219 /* vi:set ts=8 sts=2 sw=2 et: */