-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathPlaytune_poll.ino
1214 lines (1053 loc) · 53.4 KB
/
Playtune_poll.ino
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
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
745
746
747
748
749
750
751
752
753
754
755
756
757
758
759
760
761
762
763
764
765
766
767
768
769
770
771
772
773
774
775
776
777
778
779
780
781
782
783
784
785
786
787
788
789
790
791
792
793
794
795
796
797
798
799
800
801
802
803
804
805
806
807
808
809
810
811
812
813
814
815
816
817
818
819
820
821
822
823
824
825
826
827
828
829
830
831
832
833
834
835
836
837
838
839
840
841
842
843
844
845
846
847
848
849
850
851
852
853
854
855
856
857
858
859
860
861
862
863
864
865
866
867
868
869
870
871
872
873
874
875
876
877
878
879
880
881
882
883
884
885
886
887
888
889
890
891
892
893
894
895
896
897
898
899
900
901
902
903
904
905
906
907
908
909
910
911
912
913
914
915
916
917
918
919
920
921
922
923
924
925
926
927
928
929
930
931
932
933
934
935
936
937
938
939
940
941
942
943
944
945
946
947
948
949
950
951
952
953
954
955
956
957
958
959
960
961
962
963
964
965
966
967
968
969
970
971
972
973
974
975
976
977
978
979
980
981
982
983
984
985
986
987
988
989
990
991
992
993
994
995
996
997
998
999
1000
/******************************************************************************
Playtune_poll
An Arduino Tune Generator library that uses interrupt polling
to play a polyphonic musical score.
About Playtune
Playtune is a family of music players for Arduino-like microcontrollers. They
each intepret a bytestream of commands that represent a polyphonic musical
score, and play it using different techniques.
The original Playtune that was first released in 2011 uses a separate hardware timer
to generate a square wave for each note played simultaneously. The timers run at twice
the frequency of the note being played, and the interrupt routine flips the output bit.
It can play only as many simultaneous notes as there are timers available.
https://github.com/LenShustek/arduino-playtune
This second ("polling") version uses only one hardware timer that interrupts often, by
default at 20 Khz, or once every 50 microseconds. The interrupt routine determines
which, if any, of the currently playing notes need to be toggled.
https://github.com/LenShustek/playtune_poll
This third version also uses only one hardware timer interrupting frequently, but uses
the hardware digital-to-analog converter on high-performance microntrollers like the
Teensy to generate an analog wave that is the sum of stored samples of sounds. The
samples scaled to the right frequency and volume, and any number of instrument samples
can be used and mapped to MIDI patches. It currently only supports Teensy.
https://github.com/LenShustek/playtune_samp
The fourth version is an audio object for the PJRC Audio Library.
https://www.pjrc.com/teensy/td_libs_Audio.html
It allows up to 16 simultaneous sound generators that are internally mixed, at
the appropriate volume, to produce one monophonic audio stream.
Sounds are created from sampled one-cycle waveforms for any number of instruments,
each with its own attack-hold-decay-sustain-release envelope. Percussion sounds
(from MIDI channel 10) are generated from longer sampled waveforms of a complete
instrument strike. Each generator's volume is independently adjusted according to
the MIDI velocity of the note being played before all channels are mixed.
www.github.com/LenShustek/Playtune_synth
The fifth version is for the Teensy 3.1/3.2, and uses the four Periodic Interval
Timers in the Cortex M4 processor to support up to 4 simultaneous notes.
It uses less CPU time than the polling version, but is limited to 4 notes at a time.
(This was written to experiment with multi-channel multi-Tesla Coil music playing,
where I use Flexible Timer Module FTM0 for generating precise one-shot pulses.
But I ultimately switched to the polling version to play more simultaneous notes.)
www.github.com/LenShustek/Playtune_Teensy
***** Details about this version: Playtune_poll
The advantage of this polling scheme is that you can play more simultaneous notes
than there are hardware timers. The disadvantages are that it takes more CPU power,
and that notes at higher frequencies have a little bit of jitter that adds some
low frequency "undertones". That problem disappears if you use a microcontroller
like the Teensy, which uses a fast 32-bit ARM processor instead of the slow 8-bit
AVR processors used by most Arduinos so the interrupt frequency can be higher.
The interrupt routine must be really fast, regardless of how many notes are playing
and being toggled. We try hard to be efficient by using the following techniques:
- Do division incrementally by only one subtraction per channel at each interrupt.
- Use direct hardware register bit manipulation instead of digitalWrite().
(That requires assigning output pins at compile time, not at runtime.)
- Unroll the loop for checking on channel transitions at compile time.
(If you are unfamiliar with loop unrolling, see https://en.wikipedia.org/wiki/Loop_unrolling)
Once a score starts playing, all of the processing happens in the interrupt routine.
Any other "real" program can be running at the same time as long as it doesn't
use the timer or the output pins that Playtune is using.
The easiest way to hear the music is to connect each of the output pins to a resistor
(500 ohms, say). Connect other ends of the resistors together and then to one
terminal of an 8-ohm speaker. The other terminal of the speaker is connected to
ground. No other hardware is needed! An external amplifier is good, though.
By default there is no volume modulation. All tones are played at the same volume, which
makes some scores sound strange. This is definitely not a high-quality synthesizer.
There are two optional features you can enable at compile time:
(1) If the DO_VOLUME compile-time switch is set to 1, then we interpret MIDI "velocity"
information and try to modulate the volume of the output by controlling the duty cycle of the
square wave. That doesn't allow much dynamic range, though, so we make the high pulse of
the square wave vary in duration only from 1 to 10 interrupt-times wide.
There's a mapping table from MIDI velocity to pulse width that you can play with. The
MIDI standard actually doesn't specify how velocity is to be interpreted; for an interesting
emprical study of what synthesizers actually do, see this paper by Roger Dannenberg of CMU:
http://www.cs.cmu.edu/~rbd/papers/velocity-icmc2006.pdf
In order for volume modulation to work, the bytestream must include volume information,
which can be generated by Miditones with the -v option. (It's best to also use -d, so
that the self-describing header is generated.)
(2) If the DO_PERCUSSION compile-time switch is also set to 1, then we interpret notes on
MIDI channel 9 (10 if you start counting with 1) to be percussion sounds, which we play
with a high-frequency chirp that is dithered with a pseudo-random number. Tables set
the frequency and duration of the chirp differently for the various notes that represent
different percussion instruments. It will only play one percussion sound at a time.
In order for this to work, the bytestream must have the notes on the percussion track
relocated from 0..127 to 128..255, which Miditones will do with the -pt option in
addition to the -v option. (Again, it best to also use -d.)
****** Using Playtune_poll *****
Unlike the original Playtune, this is not configured as a library because we must
make compile-time changes for pin assignments. You should create a sketch directory
with the following files in it:
Playtune_poll.ino This file, which has most of the code
Playtune_poll.h The header file, which defines the output pin configuration
Playtune_poll_test.ino The main program, which contains the score(s) you wish to
play, and any other code you want to run.
In order to assign the output pins at compile time, you must add lines to the
Playtune_poll.h file. How you do that depends on the kind of microcontroller you
are using; see the instructions and examples in that file.
You can define up to MAX_CHANS channels, which is 8 by default. You can
increase MAX_CHANS up to 16. There is some inefficiency if MAX_CHANS is much
larger than the number of channels actually being used.
We also use the TimerOne library files, which you can get at
http://playground.arduino.cc/Code/Timer1 and put into your Arduino library
directory, or just put in the directory with the other files.
There are seven public functions and one public variable that you can use
in your runtime code in Playtune_poll_test.ino.
void tune_start_timer(int microseconds)
This is optional, and is available only if DO_CONSTANT_POLLTIME is set to 0.
Call it to set how often notes should be checked for transitions,
from 5 to 50 microseconds. If you don't call it, we'll pick something that seems
appropriate from the type of microcontroller and the frequency it's running at.
void tune_playscore(byte *score)
Call this pointing to a "score bytestream" to start playing a tune. It will
only play as many simultaneous notes as you have defined output pins; any
more will be ignored. See below for the format of the score bytestream.
boolean tune_playing
This global variable will be "true" if a score is playing, and "false" if not.
You can use this to see when a score has finished.
void tune_speed(unsigned int percent)
This changes playback speed to the specified percent of normal.
The minimum is percent=20 (5x slowdown),
and the maximum is percent=500 (5x speedup).
void tune_stopscore()
This will stop a currently playing score without waiting for it to end by itself.
void tune_stop_timer()
This stops playing and also stops the timer interrupt.
void tune_resumescore (bool init_pins)
The resumes playing a score that was stopped with tune_stopscore() or tune_stop_timer().
If the I/O pins need to be reinitialized because they were used by something else in the
meantime, pass TRUE for init_pins.
***** The score bytestream *****
The bytestream is a series of commands that can turn notes on and off, and can
start a waiting period until the next note change. Here are the details, with
numbers shown in hexadecimal.
If the high-order bit of the byte is 1, then it is one of the following commands:
9t nn Start playing note nn on tone generator t. Generators are numbered
starting with 0. The notes numbers are the MIDI numbers for the chromatic
scale, with decimal 60 being Middle C, and decimal 69 being Middle A
at 440 Hz. The highest note is decimal 127 at about 12,544 Hz. except
that percussion notes (instruments, really) range from 128 to 255.
[vv] If DO_VOLUME is set to 1, then we expect another byte with the velocity
value from 1 to 127. You can generate this from Miditones with the -v switch.
Everything breaks if vv is present with DO_VOLIUME set 0, and vice versa!
8t Stop playing the note on tone generator t.
Ct ii Change tone generator t to play instrument ii from now on. This version of
Playtune ignores instrument information if it is present.
F0 End of score: stop playing.
E0 End of score: start playing again from the beginning.
If the high-order bit of the byte is 0, it is a command to wait. The other 7 bits
and the 8 bits of the following byte are interpreted as a 15-bit big-endian integer
that is the number of milliseconds to wait before processing the next command.
For example,
07 D0
would cause a wait of 0x07d0 = 2000 decimal millisconds or 2 seconds. Any tones
that were playing before the wait command will continue to play.
The score is stored in Flash memory ("PROGMEM") along with the program, because
there's a lot more of that than data memory.
* ***** Where does the score data come from? *****
Well, you can write the score by hand from the instructions above, but that's
pretty hard. An easier way is to translate MIDI files into these score commands,
and I've written a program called "Miditones" to do that. See the separate
documentation for that program, which is also open source at
https://github.com/lenshustek/miditones
* ***** Nostalgia from me *****
Writing Playtune was a lot of fun, because it essentially duplicates what I did
as a graduate student at Stanford University in about 1973. That project used the
then-new Intel 8008 microprocessor, plus three hardware square-wave generators that
I built out of 7400-series TTL. The music compiler was written in Pascal and read
scores that were hand-written in a notation I made up, which looked something like
this: C Eb 4G 8G+ 2R + F D#
This was before MIDI had been invented, and anyway I wasn't a pianist so I would
not have been able to record my own playing. I could barely read music well enough
to transcribe scores, but I slowly did quite a few of them. MIDI is better!
Len Shustek, originally 4 Feb 2011,
but updated for the polling version in July 2016.
------------------------------------------------------------------------------------
The MIT License (MIT)
Copyright (c) 2011, 2016, Len Shustek
Permission is hereby granted, free of charge, to any person obtaining a copy of
this software and associated documentation files (the "Software"), to deal in
the Software without restriction, including without limitation the rights to use,
copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the
Software, and to permit persons to whom the Software is furnished to do so,
subject to the following conditions:
The above copyright notice and this permission notice shall be included in all
copies or substantial portions of the Software.
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
DEALINGS IN THE SOFTWARE.
**************************************************************************/
/*
Change log
19 January 2011, L.Shustek, V1.0
- Initial release.
23 February 2011, L. Shustek, V1.1
- prevent hang if delay rounds to count of 0
4 December 2011, L. Shustek, V1.2
- add special TESLA_COIL mods
10 June 2013, L. Shustek, V1.3
- change for compatibility with Arduino IDE version 1.0.5
6 April 2015, L. Shustek, V1.4
- change for compatibility with Arduino IDE version 1.6.x
28 May 2016, T. Wasiluk
- added support for ATmega32U4
10 July 2016, Nick Shvelidze
- Fixed include file names for Arduino 1.6 on Linux.
11 July 2017, Len Shustek
- Start this derivative version that uses polling instead of a timer
for each channel. It needs a fast processor, but can play an arbitrary
number of notes simultaneously using only one timer.
19 July 2016, Len Shustek
- Add crude volume modulation and percussion sounds. Thanks go to
Connor Nishijima for providing his code and goading us into doing this.
9 August 2016, Len Shustek
- Support the optional bytestream header, and instrument change data.
23 April 2021, Len Shustek
- Create TESLA_COIL version that only works for Teensy
- Add tune_speed()
12 November 2022, Len Shustek
- Add tune_resumescore(), suggested by GitHub user vsdyachkov.
- Add suggestions for additional pins usable on the Ardino Nano
that avoid, for example, conflicts with SPI.
13 November 2022, Len Shustek
- Major change to allow tables to be in Flash ROM memory when the polling
interval is known at compile time, ie DO_CONSTANT_POLLTIME is set to 1.
That saves about 1200 bytes of RAM, which is a big deal on a Nano with 2K!
14 November 2022, Len Shustek
- Adjust the percussion sound depending on the note volume, if DO_PERCUSSION_VOLUME
*/
#define TESLA_COIL 0 // special version for Tesla coils?
#include <Arduino.h>
#include <TimerOne.h>
#include "Playtune_poll.h" // this contains, among other things, the output pin definitions
#define DO_VOLUME 1 // generate volume-modulating code? Needs -v on Miditones.
#define DO_PERCUSSION 1 // generate percussion sounds? Needs DO_VOLUME, and -pt on Miditones
#define DO_PERCUSSION_VOLUME 1 // adjust percussion based on its volume?
#define DO_CONSTANT_POLLTIME 1 // Is polltime is a constant at compile-time? If so, we save about
// 1200 bytes of RAM. Note that tune_start_timer() can't be called then.
#define CONSTANT_POLLTIME 0 // The constant polltime in usec; if 0, we pick a good one at compile time.
#define ASSUME_VOLUME 0 // if the file has no header, should we assume there is volume info?
#define DBUG 0 // debugging? (screws up timing!)
#if TESLA_COIL
#define DO_VOLUME 1
#define DO_PERCUSSION 0
#ifndef CORE_TEENSY
Tesla coil version is only for teensy
#endif
void teslacoil_rising_edge(byte channel);
byte teslacoil_checknote(byte channel, byte note);
void teslacoil_change_instrument(byte channel, byte instrument);
void teslacoil_change_volume(byte channel, byte volume);
#endif
// A general-purpose macro joiner that allow rescanning of the result
// (Yes, we need both levels! See http://stackoverflow.com/questions/1489932)
#define PASTER(x,y) x ## y
#define JOIN(x,y) PASTER(x,y)
volatile boolean tune_playing = false;
static byte num_chans = // how many channels
#if TESLA_COIL
MAX_CHANS; // can be played on Tesla coils
#else
0; // have been assigned to pins
#endif
static byte num_chans_used = 0;
static boolean pins_initialized = false;
static boolean timer_running = false;
static boolean volume_present = ASSUME_VOLUME;
static const byte *score_start = 0;
static const byte *score_cursor = 0;
static unsigned _tune_speed = 100;
static unsigned /*short*/ int scorewait_interrupt_count;
static unsigned /*short*/ int millisecond_interrupt_count;
static unsigned /*short*/ int interrupts_per_millisecond;
static int32_t accumulator [MAX_CHANS];
static int32_t decrement [MAX_CHANS];
static byte playing [MAX_CHANS];
static byte pins [MAX_CHANS];
#if TESLA_COIL
static byte _tune_volume[MAX_CHANS] = {0};
#endif
#if DO_VOLUME && !TESLA_COIL
static byte max_high_ticks[MAX_CHANS];
static byte current_high_ticks[MAX_CHANS];
#endif
#if DO_PERCUSSION
#define NO_DRUM 0xff
static byte drum_chan = NO_DRUM; // channel playing drum now
static byte drum_tick_count; // count ticks before bit change based on random bit
static byte drum_tick_limit; // count to this
static int drum_duration; // limit on drum duration
#endif
struct file_hdr_t { // the optional bytestream file header
char id1; // 'P'
char id2; // 't'
unsigned char hdr_length; // length of whole file header
unsigned char f1; // flag byte 1
unsigned char f2; // flag byte 2
unsigned char num_tgens; // how many tone generators are used by this score
} file_header;
#define HDR_F1_VOLUME_PRESENT 0x80
#define HDR_F1_INSTRUMENTS_PRESENT 0x40
#define HDR_F1_PERCUSSION_PRESENT 0x20
// note commands in the bytestream
#define CMD_PLAYNOTE 0x90 /* play a note: low nibble is generator #, note is next byte, maybe volume */
#define CMD_STOPNOTE 0x80 /* stop a note: low nibble is generator # */
#define CMD_INSTRUMENT 0xc0 /* change instrument; low nibble is generator #, instrument is next byte */
#define CMD_RESTART 0xe0 /* restart the score from the beginning */
#define CMD_STOP 0xf0 /* stop playing */
/* if CMD < 0x80, then the other 7 bits and the next byte are a 15-bit big-endian number of msec to wait */
/* ------------------------------------------------------------------------------
We use various tables to control how notes are played:
-- accumulator decrement values for note half-periods, to set the frequency
-- number of ticks the high output should last for different volumes
-- approximate number of ticks per note periods, so the high isn't too long
-- number of ticks between transitions for the percussion notes
-- the maximum duration of percussion notes
Some of these depend on the polling frequency and the accumulator restart value.
To generate a tone, we basically do incremental division for each channel in the
polling interrupt routine to determine when the output needs to be flipped:
accum -= decrement
if (accum < 0) {
toggle speaker output
accum += ACCUM_RESTART
}
The first table below contains the maximum decrement values for each note, corresponding
to the maximum delay between interrupts of 50 microseconds (20 Khz). You can use the
companion spreadsheet to calculate other values.
Another table contains the minimum number of ticks for a full period, which we use to
cap the number of ticks that the signal is high for volume modulation. This only has
an effect on notes above about 92.
Another table contains the minimum number of ticks to wait before changing the
percussion output based on a random variable. This essentially sets the frequency
of the percussion chirp.
If the polling interval is specified at compile time (DO_CONSTANT_POLLTIME), then we compile
adjusted tables into ROM, based on a possibly higher interrupt frequency that are told to
use, or we picked based on the type and speed of the processor. The actual decrements will
be smaller because the interrupt is happening more often, so there is no danger of introducing
32-bit overflow. The tick counts will be larger, so more volumes are possible for high notes.
Everything gets better at higher frequencies, including how the notes sound!
If the polling interval can change (DO_CONSTANT_POLLTIME is zero), then
when we start the timer, we create the tables we actually use in RAM, which takes about
1280 bytes in the worst case.
*/
#if DO_CONSTANT_POLLTIME // do all the calculations at compile-time with macros
#if CONSTANT_POLLTIME==0 // we pick a time
#ifndef CORE_TEENSY
#if F_CPU >= 48000000L
#define POLLTIME 25 // fast AVR/AVX
#else
#define POLLTIME 50 // slow AVR/AVX
#endif
#else // Teensy
#if F_CPY < 50000000L
#define POLLTIME 20 // slow Teensies
#else
#define POLLTIME 10 // fast Teensies
#endif
#endif
#else
#define POLLTIME CONSTANT_POLLTIME // the user specified a time
#endif
#if DO_VOLUME
#define SD(x) ((x * (uint64_t)POLLTIME) / MAX_POLLTIME_USEC) >> 1 // scale down the decrement table for a half period
#else
#define SD(x) (x * (uint64_t)POLLTIME) / MAX_POLLTIME_USEC // scale down the decrement table for the full period
#endif
#define SU(x) ((uint32_t)x * MAX_POLLTIME_USEC) / POLLTIME // scale up the four other tables
#else // !DO_CONSTANT_POLLTIME: we will compute new scaled tables during initialization; just compile the max or min
#define SD(x) x
#define SU(x) x
#define POLLTIME 0
#endif
static int polltime = POLLTIME; // microseconds between channel polls
#define ACCUM_RESTART 1073741824L // 2^30. Generates 1-byte arithmetic on 4-byte numbers!
#define MAX_NOTE 123 // the max note number; we thus allow MAX_NOTE+1 notes
#define MAX_POLLTIME_USEC 50
#define MIN_POLLTIME_USEC 5
const int32_t max_decrement_PGM[MAX_NOTE + 1] PROGMEM = {
SD(877870L), SD(930071L), SD(985375L), SD(1043969L), SD(1106047L), SD(1171815L), SD(1241495L), SD(1315318L),
SD(1393531L), SD(1476395L), SD(1564186L), SD(1657197L), SD(1755739L), SD(1860141L), SD(1970751L),
SD(2087938L), SD(2212093L), SD(2343631L), SD(2482991L), SD(2630637L), SD(2787063L), SD(2952790L),
SD(3128372L), SD(3314395L), SD(3511479L), SD(3720282L), SD(3941502L), SD(4175876L), SD(4424186L),
SD(4687262L), SD(4965981L), SD(5261274L), SD(5574125L), SD(5905580L), SD(6256744L), SD(6628789L),
SD(7022958L), SD(7440565L), SD(7883004L), SD(8351751L), SD(8848372L), SD(9374524L), SD(9931962L),
SD(10522547L), SD(11148251L), SD(11811160L), SD(12513488L), SD(13257579L), SD(14045916L), SD(14881129L),
SD(15766007L), SD(16703503L), SD(17696745L), SD(18749048L), SD(19863924L), SD(21045095L), SD(22296501L),
SD(23622320L), SD(25026976L), SD(26515158L), SD(28091831L), SD(29762258L), SD(31532014L), SD(33407005L),
SD(35393489L), SD(37498096L), SD(39727849L), SD(42090189L), SD(44593002L), SD(47244640L), SD(50053953L),
SD(53030316L), SD(56183662L), SD(59524517L), SD(63064029L), SD(66814011L), SD(70786979L), SD(74996192L),
SD(79455697L), SD(84180379L), SD(89186005L), SD(94489281L), SD(100107906L), SD(106060631L),
SD(112367325L), SD(119049034L), SD(126128057L), SD(133628022L), SD(141573958L), SD(149992383L),
SD(158911395L), SD(168360758L), SD(178372009L), SD(188978561L), SD(200215811L), SD(212121263L),
SD(224734649L), SD(238098067L), SD(252256115L), SD(267256044L), SD(283147915L), SD(299984767L),
SD(317822789L), SD(336721516L), SD(356744019L), SD(377957122L), SD(400431622L), SD(424242525L),
SD(449469299L), SD(476196134L), SD(504512230L), SD(534512088L), SD(566295831L), SD(599969533L),
SD(635645578L), SD(673443031L), SD(713488038L), SD(755914244L), SD(800863244L), SD(848485051L),
SD(898938597L), SD(952392268L), SD(1009024459L), SD(1069024176L) };
#if !DO_CONSTANT_POLLTIME
static int32_t decrement_table[MAX_NOTE + 1]; // values may be smaller in this actual table used
#endif
#if DO_VOLUME
const uint16_t min_ticks_per_period_PGM[MAX_NOTE + 1] PROGMEM = {
SU(2446), SU(2308), SU(2178), SU(2056), SU(1940), SU(1832), SU(1728), SU(1632), SU(1540), SU(1454), SU(1372),
SU(1294), SU(1222), SU(1154), SU(1088), SU(1028), SU(970), SU(916), SU(864), SU(816), SU(770), SU(726), SU(686), SU(646),
SU(610), SU(576), SU(544), SU(514), SU(484), SU(458), SU(432), SU(408), SU(384), SU(362), SU(342), SU(322), SU(304), SU(288),
SU(272), SU(256), SU(242), SU(228), SU(216), SU(204), SU(192), SU(180), SU(170), SU(160), SU(152), SU(144), SU(136), SU(128),
SU(120), SU(114), SU(108), SU(102), SU(96), SU(90), SU(84), SU(80), SU(76), SU(72), SU(68), SU(64), SU(60), SU(56), SU(54), SU(50),
SU(48), SU(44), SU(42), SU(40), SU(38), SU(36), SU(34), SU(32), SU(30), SU(28), SU(26), SU(24), SU(24), SU(22), SU(20), SU(20), SU(18),
SU(18), SU(16), SU(16), SU(14), SU(14), SU(12), SU(12), SU(12), SU(10), SU(10), SU(10), SU(8), SU(8), SU(8), SU(8), SU(6), SU(6), SU(6), SU(6), SU(6),
SU(4), SU(4), SU(4), SU(4), SU(4), SU(4), SU(4), SU(2), SU(2), SU(2), SU(2), SU(2), SU(2), SU(2), SU(2), SU(2), SU(2), SU(2), SU(2) };
#if !DO_CONSTANT_POLLTIME
static uint16_t ticks_per_period[MAX_NOTE + 1]; // values may be larger in this actual table used
#endif
const byte min_high_ticks_PGM[128] PROGMEM = {
// A table showing how many ticks output high should last for each volume setting.
// This is pretty arbitrary, based on
// - where MIDI tracks tend to change most
// - that low numbers make quite distinct volumes.
// - that numbers above 10 are too big for even reasonable high notes
// Some synthesizers use this mapping from score notation to the center of a range:
// fff=120, ff=104, f=88, mf=72, mp=56, p=40, pp=24, ppp=8
/* 00- 15 */ SU(1), SU(1), SU(1), SU(1), SU(1), SU(1), SU(1), SU(1), SU(1), SU(1), SU(1), SU(1), SU(1), SU(1), SU(1), SU(1),
/* 16- 31 */ SU(1), SU(1), SU(1), SU(1), SU(1), SU(1), SU(1), SU(1), SU(1), SU(1), SU(1), SU(1), SU(1), SU(1), SU(1), SU(1),
/* 32- 47 */ SU(2), SU(2), SU(2), SU(2), SU(2), SU(2), SU(2), SU(2), SU(2), SU(2), SU(2), SU(2), SU(2), SU(2), SU(2), SU(2),
/* 48- 63 */ SU(3), SU(3), SU(3), SU(3), SU(3), SU(3), SU(3), SU(3), SU(3), SU(3), SU(3), SU(3), SU(3), SU(3), SU(3), SU(3),
/* 64- 79 */ SU(4), SU(4), SU(4), SU(4), SU(4), SU(4), SU(4), SU(4), SU(4), SU(4), SU(4), SU(4), SU(4), SU(4), SU(4), SU(4),
/* 80- 95 */ SU(5), SU(5), SU(5), SU(5), SU(5), SU(5), SU(5), SU(5), SU(6), SU(6), SU(6), SU(6), SU(6), SU(6), SU(6), SU(6),
/* 96-111 */ SU(7), SU(7), SU(7), SU(7), SU(7), SU(7), SU(7), SU(7), SU(8), SU(8), SU(8), SU(8), SU(8), SU(8), SU(8), SU(8),
/* 112-127 */ SU(9), SU(9), SU(9), SU(9), SU(9), SU(9), SU(9), SU(9), SU(10), SU(10), SU(10), SU(10), SU(10), SU(10), SU(10), SU(10) };
#endif
#if !DO_CONSTANT_POLLTIME
static byte high_ticks[128]; // values may be larger in this actual table used
#endif
#if DO_PERCUSSION
const byte min_drum_ticks_PGM[128] PROGMEM = { // how often to output bits, in #ticks
// common values are: 35,36:bass drum, 38,40:snare, 42,44,46:highhat
// This is a pretty random table!
/* 00- 15 */ SU(2), SU(2), SU(2), SU(2), SU(2), SU(2), SU(2), SU(2), SU(2), SU(2), SU(2), SU(2), SU(2), SU(2), SU(2), SU(2),
/* 16- 31 */ SU(2), SU(2), SU(2), SU(2), SU(2), SU(2), SU(2), SU(2), SU(2), SU(2), SU(2), SU(2), SU(2), SU(2), SU(2), SU(2),
/* 32- 47 */ SU(2), SU(2), SU(2), SU(8), SU(8), SU(2), SU(5), SU(3), SU(4), SU(3), SU(3), SU(2), SU(3), SU(2), SU(3), SU(2),
/* 48- 63 */ SU(2), SU(2), SU(2), SU(2), SU(2), SU(2), SU(2), SU(2), SU(2), SU(2), SU(2), SU(2), SU(2), SU(2), SU(2), SU(2),
/* 64- 79 */ SU(2), SU(2), SU(2), SU(2), SU(2), SU(2), SU(2), SU(2), SU(2), SU(2), SU(2), SU(2), SU(2), SU(2), SU(2), SU(2),
/* 80- 95 */ SU(2), SU(2), SU(2), SU(2), SU(2), SU(2), SU(2), SU(2), SU(2), SU(2), SU(2), SU(2), SU(2), SU(2), SU(2), SU(2),
/* 96-111 */ SU(2), SU(2), SU(2), SU(2), SU(2), SU(2), SU(2), SU(2), SU(2), SU(2), SU(2), SU(2), SU(2), SU(2), SU(2), SU(2),
/* 112-127 */ SU(2), SU(2), SU(2), SU(2), SU(2), SU(2), SU(2), SU(2), SU(2), SU(2), SU(2), SU(2), SU(2), SU(2), SU(2), SU(2) };
#if !DO_CONSTANT_POLLTIME
static byte drum_ticks[128]; // values may be larger in this actual table used
#endif
const uint16_t min_drum_cap_PGM[128] PROGMEM = { // cap on drum duration, in #ticks. 500=25 msec
/* 00- 15 */ SU(500), SU(500), SU(500), SU(500), SU(500), SU(500), SU(500), SU(500), SU(500), SU(500), SU(500), SU(500), SU(500), SU(500), SU(500), SU(500),
/* 16- 31 */ SU(500), SU(500), SU(500), SU(500), SU(500), SU(500), SU(500), SU(500), SU(500), SU(500), SU(500), SU(500), SU(500), SU(500), SU(500), SU(500),
/* 32- 47 */ SU(500), SU(500), SU(500), SU(750), SU(750), SU(500), SU(800), SU(500), SU(800), SU(750), SU(750), SU(500), SU(750), SU(500), SU(750), SU(500),
/* 48- 63 */ SU(500), SU(500), SU(500), SU(500), SU(500), SU(500), SU(500), SU(500), SU(500), SU(500), SU(500), SU(500), SU(500), SU(500), SU(500), SU(500),
/* 64- 79 */ SU(500), SU(500), SU(500), SU(500), SU(500), SU(500), SU(500), SU(500), SU(500), SU(500), SU(500), SU(500), SU(500), SU(500), SU(500), SU(500),
/* 80- 95 */ SU(500), SU(500), SU(500), SU(500), SU(500), SU(500), SU(500), SU(500), SU(500), SU(500), SU(500), SU(500), SU(500), SU(500), SU(500), SU(500),
/* 96-111 */ SU(500), SU(500), SU(500), SU(500), SU(500), SU(500), SU(500), SU(500), SU(500), SU(500), SU(500), SU(500), SU(500), SU(500), SU(500), SU(500),
/* 112-127 */ SU(500), SU(500), SU(500), SU(500), SU(500), SU(500), SU(500), SU(500), SU(500), SU(500), SU(500), SU(500), SU(500), SU(500), SU(500), SU(500) };
#if !DO_CONSTANT_POLLTIME
static uint16_t drum_cap[128]; // values may be larger in this actual table used
#endif
#endif
void tune_stopnote (byte chan);
void tune_stepscore (void);
void timer_ISR(void);
//------------------------------------------------------------------------------
// Initialize and start the timer
//------------------------------------------------------------------------------
static void do_start_timer(void) {
interrupts_per_millisecond = 1000 / polltime; // this is a truncated approximation
millisecond_interrupt_count = interrupts_per_millisecond;
#if DBUG
Serial.print("polltime in usec: "); Serial.println(polltime);
#endif
Timer1.initialize(polltime); // start the timer
Timer1.attachInterrupt(timer_ISR);
timer_running = true; }
#if !DO_CONSTANT_POLLTIME
void tune_start_timer(int user_polltime) { // compute RAM tables based on polling rate
// Decide on an interrupt poll time
if (user_polltime) // set by the caller
polltime = max( min(user_polltime, MAX_POLLTIME_USEC), MIN_POLLTIME_USEC);
else { // polltime isn't specified; try to pick a good one
polltime = MAX_POLLTIME_USEC; // assume the worst
if (F_CPU >= 48000000L) polltime = 25; // unless the clock is really fast
#ifdef CORE_TEENSY
if (F_CPU < 50000000L)
polltime = 20; // slow Teensies
else polltime = 10; // fast teensies
#endif
}
// Set up the real tables we use based on our chosen polling rate, which may be faster
// than the 50 microseconds that the table was computed for. We use 32- or 64-bit arithmetic
// for intermediate results to avoid errors, but we only do this during initialization.
for (int note = 0; note <= MAX_NOTE; ++note) {
decrement_table[note] =
((uint64_t) pgm_read_dword(max_decrement_PGM + note) * (uint64_t)polltime) / MAX_POLLTIME_USEC
#if DO_VOLUME
>> 1; // if we're doing volume, use the full note period, not the half period
ticks_per_period[note] = ((uint32_t)pgm_read_word(min_ticks_per_period_PGM + note) * MAX_POLLTIME_USEC) / polltime;
#endif
; }
#if DO_VOLUME
for (int index = 0; index < 128; ++index)
high_ticks[index] = ((uint32_t)pgm_read_byte(min_high_ticks_PGM + index) * MAX_POLLTIME_USEC) / polltime;
#endif
#if DO_PERCUSSION
for (int index = 0; index < 128; ++index) {
drum_ticks[index] = ((uint32_t)pgm_read_byte(min_drum_ticks_PGM + index) * MAX_POLLTIME_USEC) / polltime;
drum_cap[index] = ((uint32_t)pgm_read_word(min_drum_cap_PGM + index) * MAX_POLLTIME_USEC) / polltime; }
#endif
do_start_timer(); }
#endif
//------------------------------------------------------------------------------
// Set up output pins, compile-time macros for manipulating them,
// and runtime bit set and clear routines that are faster than digitalWrite
//------------------------------------------------------------------------------
void tune_init_pins(void) {
#if SCOPE_TEST
pinMode(SCOPE_PIN, OUTPUT);
digitalWrite(SCOPE_PIN, 0);
#endif
#if TESLA_COIL
#define tune_initchan(pin) \
if (num_chans < MAX_CHANS) \
++num_chans;
#else
#define tune_initchan(pin) \
if (num_chans < MAX_CHANS) { \
pins[num_chans] = pin; \
pinMode(pin, OUTPUT); \
++num_chans; \
}
#endif
#ifndef CHAN_0_PIN
#define CHAN_0_PIN 0
#define CHAN_0_REG B
#define CHAN_0_BIT 0
#else
tune_initchan(CHAN_0_PIN)
#endif
#define CHAN_0_PINREG JOIN(PIN,CHAN_0_REG)
#define CHAN_0_PORTREG JOIN(PORT,CHAN_0_REG)
#ifndef CHAN_1_PIN
#define CHAN_1_PIN 0
#define CHAN_1_REG B
#define CHAN_1_BIT 0
#else
tune_initchan(CHAN_1_PIN)
#endif
#define CHAN_1_PINREG JOIN(PIN,CHAN_1_REG)
#define CHAN_1_PORTREG JOIN(PORT,CHAN_1_REG)
#ifndef CHAN_2_PIN
#define CHAN_2_PIN 0
#define CHAN_2_REG B
#define CHAN_2_BIT 0
#else
tune_initchan(CHAN_2_PIN)
#endif
#define CHAN_2_PINREG JOIN(PIN,CHAN_2_REG)
#define CHAN_2_PORTREG JOIN(PORT,CHAN_2_REG)
#ifndef CHAN_3_PIN
#define CHAN_3_PIN 0
#define CHAN_3_REG B
#define CHAN_3_BIT 0
#else
tune_initchan(CHAN_3_PIN)
#endif
#define CHAN_3_PINREG JOIN(PIN,CHAN_3_REG)
#define CHAN_3_PORTREG JOIN(PORT,CHAN_3_REG)
#ifndef CHAN_4_PIN
#define CHAN_4_PIN 0
#define CHAN_4_REG B
#define CHAN_4_BIT 0
#else
tune_initchan(CHAN_4_PIN)
#endif
#define CHAN_4_PINREG JOIN(PIN,CHAN_4_REG)
#define CHAN_4_PORTREG JOIN(PORT,CHAN_4_REG)
#ifndef CHAN_5_PIN
#define CHAN_5_PIN 0
#define CHAN_5_REG B
#define CHAN_5_BIT 0
#else
tune_initchan(CHAN_5_PIN)
#endif
#define CHAN_5_PINREG JOIN(PIN,CHAN_5_REG)
#define CHAN_5_PORTREG JOIN(PORT,CHAN_5_REG)
#ifndef CHAN_6_PIN
#define CHAN_6_PIN 0
#define CHAN_6_REG B
#define CHAN_6_BIT 0
#else
tune_initchan(CHAN_6_PIN)
#endif
#define CHAN_6_PINREG JOIN(PIN,CHAN_6_REG)
#define CHAN_6_PORTREG JOIN(PORT,CHAN_6_REG)
#ifndef CHAN_7_PIN
#define CHAN_7_PIN 0
#define CHAN_7_REG B
#define CHAN_7_BIT 0
#else
tune_initchan(CHAN_7_PIN)
#endif
#define CHAN_7_PINREG JOIN(PIN,CHAN_7_REG)
#define CHAN_7_PORTREG JOIN(PORT,CHAN_7_REG)
#ifndef CHAN_8_PIN
#define CHAN_8_PIN 0
#define CHAN_8_REG B
#define CHAN_8_BIT 0
#else
tune_initchan(CHAN_8_PIN)
#endif
#define CHAN_8_PINREG JOIN(PIN,CHAN_8_REG)
#define CHAN_8_PORTREG JOIN(PORT,CHAN_8_REG)
#ifndef CHAN_9_PIN
#define CHAN_9_PIN 0
#define CHAN_9_REG B
#define CHAN_9_BIT 0
#else
tune_initchan(CHAN_9_PIN)
#endif
#define CHAN_9_PINREG JOIN(PIN,CHAN_9_REG)
#define CHAN_9_PORTREG JOIN(PORT,CHAN_9_REG)
#ifndef CHAN_10_PIN
#define CHAN_10_PIN 0
#define CHAN_10_REG B
#define CHAN_10_BIT 0
#else
tune_initchan(CHAN_10_PIN)
#endif
#define CHAN_10_PINREG JOIN(PIN,CHAN_10_REG)
#define CHAN_10_PORTREG JOIN(PORT,CHAN_10_REG)
#ifndef CHAN_11_PIN
#define CHAN_11_PIN 0
#define CHAN_11_REG B
#define CHAN_11_BIT 0
#else
tune_initchan(CHAN_11_PIN)
#endif
#define CHAN_11_PINREG JOIN(PIN,CHAN_11_REG)
#define CHAN_11_PORTREG JOIN(PORT,CHAN_11_REG)
#ifndef CHAN_12_PIN
#define CHAN_12_PIN 0
#define CHAN_12_REG B
#define CHAN_12_BIT 0
#else
tune_initchan(CHAN_12_PIN)
#endif
#define CHAN_12_PINREG JOIN(PIN,CHAN_12_REG)
#define CHAN_12_PORTREG JOIN(PORT,CHAN_12_REG)
#ifndef CHAN_13_PIN
#define CHAN_13_PIN 0
#define CHAN_13_REG B
#define CHAN_13_BIT 0
#else
tune_initchan(CHAN_13_PIN)
#endif
#define CHAN_13_PINREG JOIN(PIN,CHAN_13_REG)
#define CHAN_13_PORTREG JOIN(PORT,CHAN_13_REG)
#ifndef CHAN_14_PIN
#define CHAN_14_PIN 0
#define CHAN_14_REG B
#define CHAN_14_BIT 0
#else
tune_initchan(CHAN_14_PIN)
#endif
#define CHAN_14_PINREG JOIN(PIN,CHAN_14_REG)
#define CHAN_14_PORTREG JOIN(PORT,CHAN_14_REG)
#ifndef CHAN_15_PIN
#define CHAN_15_PIN 0
#define CHAN_15_REG B
#define CHAN_15_BIT 0
#else
tune_initchan(CHAN_15_PIN)
#endif
#define CHAN_15_PINREG JOIN(PIN,CHAN_15_REG)
#define CHAN_15_PORTREG JOIN(PORT,CHAN_15_REG)
pins_initialized = true; }
#ifdef CORE_TEENSY
// On a Teensy, digitalWrite is not lighting fast, but isn't too bad
#define chan_set_high(chan) digitalWrite(pins[chan], HIGH)
#define chan_set_low(chan) digitalWrite(pins[chan], LOW)
#else // ARDUINO
// See http://garretlab.web.fc2.com/en/arduino/inside/index.html for a great explanation of Arduino I/O definitions
const uint16_t chan_portregs_PGM[16] PROGMEM = { // port addresses for each channel
(uint16_t) &CHAN_0_PORTREG, (uint16_t) &CHAN_1_PORTREG, (uint16_t) &CHAN_2_PORTREG, (uint16_t) &CHAN_3_PORTREG,
(uint16_t) &CHAN_4_PORTREG, (uint16_t) &CHAN_5_PORTREG, (uint16_t) &CHAN_6_PORTREG, (uint16_t) &CHAN_7_PORTREG,
(uint16_t) &CHAN_8_PORTREG, (uint16_t) &CHAN_9_PORTREG, (uint16_t) &CHAN_10_PORTREG, (uint16_t) &CHAN_11_PORTREG,
(uint16_t) &CHAN_12_PORTREG, (uint16_t) &CHAN_13_PORTREG, (uint16_t) &CHAN_14_PORTREG, (uint16_t) &CHAN_15_PORTREG };
const byte chan_bitmask_PGM[16] PROGMEM = { // bit masks for each channel
(1 << CHAN_0_BIT), (1 << CHAN_1_BIT), (1 << CHAN_2_BIT), (1 << CHAN_3_BIT),
(1 << CHAN_4_BIT), (1 << CHAN_5_BIT), (1 << CHAN_6_BIT), (1 << CHAN_7_BIT),
(1 << CHAN_8_BIT), (1 << CHAN_9_BIT), (1 << CHAN_10_BIT), (1 << CHAN_11_BIT),
(1 << CHAN_12_BIT), (1 << CHAN_13_BIT), (1 << CHAN_14_BIT), (1 << CHAN_15_BIT) };
// macros to control channel output that are faster than digitalWrite
#define chan_set_high(chan) *(volatile byte *)pgm_read_word(chan_portregs_PGM + chan) |= pgm_read_byte(chan_bitmask_PGM + chan)
#define chan_set_low(chan) *(volatile byte *)pgm_read_word(chan_portregs_PGM + chan) &= ~pgm_read_byte(chan_bitmask_PGM + chan);
#endif // ARDUINO
//------------------------------------------------------------------------------
// Start playing a note on a particular channel
//------------------------------------------------------------------------------
#if DO_VOLUME
void tune_playnote (byte chan, byte note, byte volume) {
#if DBUG
Serial.print("chan="); Serial.print(chan);
Serial.print(" note="); Serial.print(note);
Serial.print(" vol="); Serial.println(volume);
#endif
if (chan < num_chans) {
if (note <= MAX_NOTE) {
if (volume > 127) volume = 127;
#if TESLA_COIL
note = teslacoil_checknote(chan, note); // let teslacoil modify the note
if (chan < num_chans && _tune_volume[chan] != volume) { // if volume is changing
_tune_volume[chan] = volume;
teslacoil_change_volume(chan, volume); }
#else
max_high_ticks[chan] =
#if DO_CONSTANT_POLLTIME
min(pgm_read_byte(min_high_ticks_PGM + volume), pgm_read_byte(min_ticks_per_period_PGM + note));
#else
min(high_ticks[volume], ticks_per_period[note]);
#endif
current_high_ticks[chan] = 0; // we're starting with output low
#if DBUG
Serial.print(" max high ticks="); Serial.println(max_high_ticks[chan]);
#endif
#endif
decrement[chan] =
#if DO_CONSTANT_POLLTIME
pgm_read_dword(max_decrement_PGM + note);
#else
decrement_table[note];
#endif
accumulator[chan] = ACCUM_RESTART;
playing[chan] = true; }
#if DO_PERCUSSION
else if (note > 127) { // Notes above 127 are percussion sounds.
// Following Connor Nishijima's suggestion, we only play one at a time.
if (drum_chan == NO_DRUM) { // drum player is free
drum_chan = chan; // assign it
#if DO_CONSTANT_POLLTIME
drum_tick_limit = pgm_read_byte(min_drum_ticks_PGM + note - 128);
drum_duration = pgm_read_word(min_drum_cap_PGM + note - 128); // cap on drum note duration
#else
drum_tick_limit = drum_ticks[note - 128];
drum_duration = drum_cap[note - 128]; // cap on drum note duration
#endif
#if DO_PERCUSSION_VOLUME
drum_tick_limit = ((uint16_t)drum_tick_limit * volume) >> 8;
if (drum_tick_limit == 0) drum_tick_limit = 1;
drum_duration = ((uint32_t)drum_duration * volume) >> 8;
if (drum_duration == 0) drum_duration = 1;
#endif
#if DBUG
Serial.print(" drum tick limit="); Serial.println(drum_tick_limit);
Serial.print(" drum tick duration="); Serial.println(drum_duration);
#endif
drum_tick_count = drum_tick_limit;
} }
#endif
} }
#else // non-volume version
void tune_playnote (byte chan, byte note) {
if (chan < num_chans) {
if (note > MAX_NOTE) note = MAX_NOTE;
decrement[chan] =
#if DO_CONSTANT_POLLTIME
pgm_read_dword(max_decrement_PGM + note);
#else
decrement_table[note];
#endif
accumulator[chan] = ACCUM_RESTART;
playing[chan] = true; } }
#endif
//------------------------------------------------------------------------------
// Stop playing a note on a particular channel
//------------------------------------------------------------------------------
void tune_stopnote (byte chan) {
if (chan < num_chans) {
#if DO_PERCUSSION
if (chan == drum_chan) { // if this is currently the drum channel
drum_chan = NO_DRUM; // stop playing drum
#if DBUG
Serial.print(" stopping drum chan "); Serial.println(chan);
#endif
}
#endif
if (playing[chan]) {
playing[chan] = false;
chan_set_low(chan);
#if DBUG
Serial.print(" stopping chan "); Serial.println(chan);
#endif
} } }
//------------------------------------------------------------------------------
// Start playing a score
//------------------------------------------------------------------------------
void tune_playscore (const byte * score) {
if (!pins_initialized)
tune_init_pins();
if (tune_playing) tune_stopscore();
if (!timer_running)
#if DO_CONSTANT_POLLTIME
do_start_timer(); // using compile-time tables, so just get the timer going
#else
tune_start_timer(0); // compute the RAM tables, then get the timer going
#endif
score_start = score;
volume_present = ASSUME_VOLUME;
num_chans_used = MAX_CHANS;
// look for the optional file header
memcpy_P(&file_header, score, sizeof(file_hdr_t)); // copy possible header from PROGMEM to RAM
if (file_header.id1 == 'P' && file_header.id2 == 't') { // validate it
volume_present = file_header.f1 & HDR_F1_VOLUME_PRESENT;
num_chans_used = max(1, min(MAX_CHANS, file_header.num_tgens));
#if DBUG
Serial.print("header: volume_present="); Serial.print(volume_present);
Serial.print(", #chans="); Serial.println(num_chans_used);
#endif
score_start += file_header.hdr_length; // skip the whole header
}
score_cursor = score_start;
#if DBUG
Serial.println("playing score");
#endif
tune_stepscore(); /* execute initial commands */
tune_playing = true; }
void tune_speed (unsigned speed) {
if (speed > 500) speed = 500; // pin between 1/5 and 5x speed
if (speed < 20) speed = 20;
_tune_speed = speed; }
void tune_stepscore (void) { // continue in the score
byte cmd, opcode, chan, note, vol;
/* Do score commands until a "wait" is found, or the score is stopped.
This is called initially from tune_playscore, but then is called
from the interrupt routine when waits expire. */
while (1) {
cmd = pgm_read_byte(score_cursor++);
if (cmd < 0x80) { /* wait count in msec. */
scorewait_interrupt_count = ((unsigned)cmd << 8) | (pgm_read_byte(score_cursor++));
if (_tune_speed != 100) {
scorewait_interrupt_count =
(unsigned) (((unsigned long)scorewait_interrupt_count * 100UL) / _tune_speed);
if (scorewait_interrupt_count == 0) scorewait_interrupt_count = 1; }
//Serial.println((int)(score_cursor-score_start+6),HEX);
#if DBUG
Serial.print("waitcount="); Serial.println(scorewait_interrupt_count);
#endif
break; }
opcode = cmd & 0xf0;
chan = cmd & 0x0f;
if (opcode == CMD_STOPNOTE) { /* stop note */
tune_stopnote (chan); }
else if (opcode == CMD_PLAYNOTE) { /* play note */
note = pgm_read_byte(score_cursor++); // argument evaluation order is undefined in C!
#if DBUG
Serial.print("chan "); Serial.print(chan); Serial.print(" note "); Serial.println(note);
#endif