comparison win95kggui/dep/ft2play/pmp_main.c @ 126:8e4ee43d3b81

remove submodules
author Paper <mrpapersonic@gmail.com>
date Sun, 01 Oct 2023 03:48:43 -0400
parents
children
comparison
equal deleted inserted replaced
125:5cc85ef3a675 126:8e4ee43d3b81
1 /* - main XM replayer -
2 **
3 ** NOTE: Effect handling is slightly different because
4 ** I've removed the channel muting logic.
5 ** Muted channels would only process *some* effects, but
6 ** since we can't mute channels, we don't care about this.
7 **
8 ** In FT2, the only way to mute a channel is through the
9 ** tracker itself, so this is not really needed in a replayer.
10 */
11
12 #include <stdio.h>
13 #include <stdint.h>
14 #include <stdbool.h>
15 #include "pmplay.h"
16 #include "pmp_mix.h"
17 #include "snd_masm.h"
18 #include "tables.h"
19
20 #define MAX_FRQ 32000
21 #define MAX_NOTES (10*12*16+16)
22
23 static tonTyp nilPatternLine[32]; // 8bb: used for non-allocated (empty) patterns
24
25 typedef void (*volKolEfxRoutine)(stmTyp *ch);
26 typedef void (*volKolEfxRoutine2)(stmTyp *ch, uint8_t *volKol);
27 typedef void (*efxRoutine)(stmTyp *ch, uint8_t param);
28
29 static void retrigVolume(stmTyp *ch)
30 {
31 ch->realVol = ch->oldVol;
32 ch->outVol = ch->oldVol;
33 ch->outPan = ch->oldPan;
34 ch->status |= IS_Vol + IS_Pan + IS_QuickVol;
35 }
36
37 static void retrigEnvelopeVibrato(stmTyp *ch)
38 {
39 // 8bb: reset vibrato position
40 if (!(ch->waveCtrl & 0x04))
41 ch->vibPos = 0;
42
43 /*
44 ** 8bb:
45 ** In FT2.00 .. FT2.09, if the sixth bit of "ch->waveCtrl" is set
46 ** (from effect E7x where x is $4..$7 or $C..$F) and you trigger a note,
47 ** the replayer interrupt will freeze / lock up. This is because of a
48 ** label bug in the original code, causing it to jump back to itself
49 ** indefinitely.
50 */
51
52 // 8bb: safely reset tremolo position
53 if (!(ch->waveCtrl & 0x40))
54 ch->tremPos = 0;
55
56 ch->retrigCnt = 0;
57 ch->tremorPos = 0;
58
59 ch->envSustainActive = true;
60
61 instrTyp *ins = ch->instrSeg;
62
63 if (ins->envVTyp & ENV_ENABLED)
64 {
65 ch->envVCnt = 65535;
66 ch->envVPos = 0;
67 }
68
69 if (ins->envPTyp & ENV_ENABLED)
70 {
71 ch->envPCnt = 65535;
72 ch->envPPos = 0;
73 }
74
75 ch->fadeOutSpeed = ins->fadeOut; // 8bb: ranges 0..4095 (FT2 doesn't check if it's higher than 4095!)
76
77 // 8bb: final fadeout range is in fact 0..32768, and not 0..65536 like the XM format doc says
78 ch->fadeOutAmp = 32768;
79
80 if (ins->vibDepth > 0)
81 {
82 ch->eVibPos = 0;
83
84 if (ins->vibSweep > 0)
85 {
86 ch->eVibAmp = 0;
87 ch->eVibSweep = (ins->vibDepth << 8) / ins->vibSweep;
88 }
89 else
90 {
91 ch->eVibAmp = ins->vibDepth << 8;
92 ch->eVibSweep = 0;
93 }
94 }
95 }
96
97 static void keyOff(stmTyp *ch)
98 {
99 instrTyp *ins = ch->instrSeg;
100
101 if (!(ins->envPTyp & ENV_ENABLED)) // 8bb: probably an FT2 bug
102 {
103 if (ch->envPCnt >= (uint16_t)ins->envPP[ch->envPPos][0])
104 ch->envPCnt = ins->envPP[ch->envPPos][0]-1;
105 }
106
107 if (ins->envVTyp & ENV_ENABLED)
108 {
109 if (ch->envVCnt >= (uint16_t)ins->envVP[ch->envVPos][0])
110 ch->envVCnt = ins->envVP[ch->envVPos][0]-1;
111 }
112 else
113 {
114 ch->realVol = 0;
115 ch->outVol = 0;
116 ch->status |= IS_Vol + IS_QuickVol;
117 }
118
119 ch->envSustainActive = false;
120 }
121
122 uint32_t getFrequenceValue(uint16_t period) // 8bb: converts period to 16.16fp resampling delta
123 {
124 uint32_t delta;
125
126 if (period == 0)
127 return 0;
128
129 if (linearFrqTab)
130 {
131 const uint16_t invPeriod = (12 * 192 * 4) - period; // 8bb: this intentionally underflows uint16_t to be accurate to FT2
132
133 const uint32_t quotient = invPeriod / 768;
134 const uint32_t remainder = invPeriod % 768;
135
136 const int32_t octShift = 14 - quotient;
137
138 delta = (uint32_t)(((int64_t)logTab[remainder] * (int32_t)frequenceMulFactor) >> 24);
139 delta >>= (octShift & 31); // 8bb: added needed 32-bit bitshift mask
140 }
141 else
142 {
143 delta = frequenceDivFactor / (uint32_t)period;
144 }
145
146 return delta;
147 }
148
149 static void startTone(uint8_t ton, uint8_t effTyp, uint8_t eff, stmTyp *ch)
150 {
151 if (ton == NOTE_KEYOFF)
152 {
153 keyOff(ch);
154 return;
155 }
156
157 // 8bb: if we came from Rxy (retrig), we didn't check note (Ton) yet
158 if (ton == 0)
159 {
160 ton = ch->tonNr;
161 if (ton == 0)
162 return; // 8bb: if still no note, return
163 }
164
165 ch->tonNr = ton;
166
167 instrTyp *ins = instr[ch->instrNr];
168 if (ins == NULL)
169 ins = instr[0];
170
171 ch->instrSeg = ins;
172 ch->mute = ins->mute;
173
174 uint8_t smp = ins->ta[ton-1] & 0xF; // 8bb: added for safety
175 ch->sampleNr = smp;
176
177 sampleTyp *s = &ins->samp[smp];
178 ch->relTonNr = s->relTon;
179
180 ton += ch->relTonNr;
181 if (ton >= 10*12)
182 return;
183
184 ch->oldVol = s->vol;
185 ch->oldPan = s->pan;
186
187 if (effTyp == 0x0E && (eff & 0xF0) == 0x50) // 8bb: EFx - Set Finetune
188 ch->fineTune = ((eff & 0x0F) << 4) - 128;
189 else
190 ch->fineTune = s->fine;
191
192 if (ton != 0)
193 {
194 const uint16_t tmpTon = ((ton-1) << 4) + (((int8_t)ch->fineTune >> 3) + 16); // 8bb: 0..1935
195 if (tmpTon < MAX_NOTES) // 8bb: tmpTon is *always* below MAX_NOTES here, this check is not needed
196 ch->outPeriod = ch->realPeriod = note2Period[tmpTon];
197 }
198
199 ch->status |= IS_Period + IS_Vol + IS_Pan + IS_NyTon + IS_QuickVol;
200
201 if (effTyp == 9) // 8bb: 9xx - Set Sample Offset
202 {
203 if (eff)
204 ch->smpOffset = ch->eff;
205
206 ch->smpStartPos = ch->smpOffset << 8;
207 }
208 else
209 {
210 ch->smpStartPos = 0;
211 }
212
213 P_StartTone(s, ch->smpStartPos);
214 }
215
216 static void volume(stmTyp *ch, uint8_t param); // 8bb: volume slide
217 static void vibrato2(stmTyp *ch);
218 static void tonePorta(stmTyp *ch, uint8_t param);
219
220 static void dummy(stmTyp *ch, uint8_t param)
221 {
222 return;
223
224 (void)ch;
225 (void)param;
226 }
227
228 static void finePortaUp(stmTyp *ch, uint8_t param)
229 {
230 if (param == 0)
231 param = ch->fPortaUpSpeed;
232
233 ch->fPortaUpSpeed = param;
234
235 ch->realPeriod -= param << 2;
236 if ((int16_t)ch->realPeriod < 1)
237 ch->realPeriod = 1;
238
239 ch->outPeriod = ch->realPeriod;
240 ch->status |= IS_Period;
241 }
242
243 static void finePortaDown(stmTyp *ch, uint8_t param)
244 {
245 if (param == 0)
246 param = ch->fPortaDownSpeed;
247
248 ch->fPortaDownSpeed = param;
249
250 ch->realPeriod += param << 2;
251 if ((int16_t)ch->realPeriod > MAX_FRQ-1) // 8bb: FT2 bug, should've been unsigned comparison!
252 ch->realPeriod = MAX_FRQ-1;
253
254 ch->outPeriod = ch->realPeriod;
255 ch->status |= IS_Period;
256 }
257
258 static void setGlissCtrl(stmTyp *ch, uint8_t param)
259 {
260 ch->glissFunk = param;
261 }
262
263 static void setVibratoCtrl(stmTyp *ch, uint8_t param)
264 {
265 ch->waveCtrl = (ch->waveCtrl & 0xF0) | param;
266 }
267
268 static void jumpLoop(stmTyp *ch, uint8_t param)
269 {
270 if (param == 0)
271 {
272 ch->pattPos = song.pattPos & 0xFF;
273 }
274 else if (ch->loopCnt == 0)
275 {
276 ch->loopCnt = param;
277
278 song.pBreakPos = ch->pattPos;
279 song.pBreakFlag = true;
280 }
281 else if (--ch->loopCnt > 0)
282 {
283 song.pBreakPos = ch->pattPos;
284 song.pBreakFlag = true;
285 }
286 }
287
288 static void setTremoloCtrl(stmTyp *ch, uint8_t param)
289 {
290 ch->waveCtrl = (param << 4) | (ch->waveCtrl & 0x0F);
291 }
292
293 static void volFineUp(stmTyp *ch, uint8_t param)
294 {
295 if (param == 0)
296 param = ch->fVolSlideUpSpeed;
297
298 ch->fVolSlideUpSpeed = param;
299
300 ch->realVol += param;
301 if (ch->realVol > 64)
302 ch->realVol = 64;
303
304 ch->outVol = ch->realVol;
305 ch->status |= IS_Vol;
306 }
307
308 static void volFineDown(stmTyp *ch, uint8_t param)
309 {
310 if (param == 0)
311 param = ch->fVolSlideDownSpeed;
312
313 ch->fVolSlideDownSpeed = param;
314
315 ch->realVol -= param;
316 if ((int8_t)ch->realVol < 0)
317 ch->realVol = 0;
318
319 ch->outVol = ch->realVol;
320 ch->status |= IS_Vol;
321 }
322
323 static void noteCut0(stmTyp *ch, uint8_t param)
324 {
325 if (param == 0) // 8bb: only a parameter of zero is handled here
326 {
327 ch->realVol = 0;
328 ch->outVol = 0;
329 ch->status |= IS_Vol + IS_QuickVol;
330 }
331 }
332
333 static void pattDelay(stmTyp *ch, uint8_t param)
334 {
335 if (song.pattDelTime2 == 0)
336 song.pattDelTime = param + 1;
337
338 (void)ch;
339 }
340
341 static const efxRoutine EJumpTab_TickZero[16] =
342 {
343 dummy, // 0
344 finePortaUp, // 1
345 finePortaDown, // 2
346 setGlissCtrl, // 3
347 setVibratoCtrl, // 4
348 dummy, // 5
349 jumpLoop, // 6
350 setTremoloCtrl, // 7
351 dummy, // 8
352 dummy, // 9
353 volFineUp, // A
354 volFineDown, // B
355 noteCut0, // C
356 dummy, // D
357 pattDelay, // E
358 dummy // F
359 };
360
361 static void E_Effects_TickZero(stmTyp *ch, uint8_t param)
362 {
363 EJumpTab_TickZero[param >> 4](ch, param & 0x0F);
364 }
365
366 static void posJump(stmTyp *ch, uint8_t param)
367 {
368 song.songPos = (int16_t)param - 1;
369 song.pBreakPos = 0;
370 song.posJumpFlag = true;
371
372 (void)ch;
373 }
374
375 static void pattBreak(stmTyp *ch, uint8_t param)
376 {
377 song.posJumpFlag = true;
378
379 param = ((param >> 4) * 10) + (param & 0x0F);
380 if (param <= 63)
381 song.pBreakPos = param;
382 else
383 song.pBreakPos = 0;
384
385 (void)ch;
386 }
387
388 static void setSpeed(stmTyp *ch, uint8_t param)
389 {
390 if (param >= 32)
391 {
392 song.speed = param;
393 P_SetSpeed(song.speed);
394 }
395 else
396 {
397 song.timer = song.tempo = param;
398 }
399
400 (void)ch;
401 }
402
403 static void setGlobaVol(stmTyp *ch, uint8_t param)
404 {
405 if (param > 64)
406 param = 64;
407
408 song.globVol = param;
409
410 stmTyp *c = stm;
411 for (int32_t i = 0; i < song.antChn; i++, c++) // 8bb: this updates the volume for all voices
412 c->status |= IS_Vol;
413
414 (void)ch;
415 }
416
417 static void setEnvelopePos(stmTyp *ch, uint8_t param)
418 {
419 int8_t envPos;
420 bool envUpdate;
421 int16_t newEnvPos;
422
423 instrTyp *ins = ch->instrSeg;
424
425 // *** VOLUME ENVELOPE ***
426 if (ins->envVTyp & ENV_ENABLED)
427 {
428 ch->envVCnt = param - 1;
429
430 envPos = 0;
431 envUpdate = true;
432 newEnvPos = param;
433
434 if (ins->envVPAnt > 1)
435 {
436 envPos++;
437 for (int32_t i = 0; i < ins->envVPAnt-1; i++)
438 {
439 if (newEnvPos < ins->envVP[envPos][0])
440 {
441 envPos--;
442
443 newEnvPos -= ins->envVP[envPos][0];
444 if (newEnvPos == 0)
445 {
446 envUpdate = false;
447 break;
448 }
449
450 if (ins->envVP[envPos+1][0] <= ins->envVP[envPos][0])
451 {
452 envUpdate = true;
453 break;
454 }
455
456 ch->envVIPValue = ((ins->envVP[envPos+1][1] - ins->envVP[envPos][1]) & 0xFF) << 8;
457 ch->envVIPValue /= (ins->envVP[envPos+1][0] - ins->envVP[envPos][0]);
458
459 ch->envVAmp = (ch->envVIPValue * (newEnvPos - 1)) + ((ins->envVP[envPos][1] & 0xFF) << 8);
460
461 envPos++;
462
463 envUpdate = false;
464 break;
465 }
466
467 envPos++;
468 }
469
470 if (envUpdate)
471 envPos--;
472 }
473
474 if (envUpdate)
475 {
476 ch->envVIPValue = 0;
477 ch->envVAmp = (ins->envVP[envPos][1] & 0xFF) << 8;
478 }
479
480 if (envPos >= ins->envVPAnt)
481 {
482 envPos = ins->envVPAnt - 1;
483 if (envPos < 0)
484 envPos = 0;
485 }
486
487 ch->envVPos = envPos;
488 }
489
490 // *** PANNING ENVELOPE ***
491 if (ins->envVTyp & ENV_SUSTAIN) // 8bb: FT2 bug? (should probably have been "ins->envPTyp & ENV_ENABLED")
492 {
493 ch->envPCnt = param - 1;
494
495 envPos = 0;
496 envUpdate = true;
497 newEnvPos = param;
498
499 if (ins->envPPAnt > 1)
500 {
501 envPos++;
502 for (int32_t i = 0; i < ins->envPPAnt-1; i++)
503 {
504 if (newEnvPos < ins->envPP[envPos][0])
505 {
506 envPos--;
507
508 newEnvPos -= ins->envPP[envPos][0];
509 if (newEnvPos == 0)
510 {
511 envUpdate = false;
512 break;
513 }
514
515 if (ins->envPP[envPos + 1][0] <= ins->envPP[envPos][0])
516 {
517 envUpdate = true;
518 break;
519 }
520
521 ch->envPIPValue = ((ins->envPP[envPos+1][1] - ins->envPP[envPos][1]) & 0xFF) << 8;
522 ch->envPIPValue /= (ins->envPP[envPos+1][0] - ins->envPP[envPos][0]);
523
524 ch->envPAmp = (ch->envPIPValue * (newEnvPos - 1)) + ((ins->envPP[envPos][1] & 0xFF) << 8);
525
526 envPos++;
527
528 envUpdate = false;
529 break;
530 }
531
532 envPos++;
533 }
534
535 if (envUpdate)
536 envPos--;
537 }
538
539 if (envUpdate)
540 {
541 ch->envPIPValue = 0;
542 ch->envPAmp = (ins->envPP[envPos][1] & 0xFF) << 8;
543 }
544
545 if (envPos >= ins->envPPAnt)
546 {
547 envPos = ins->envPPAnt - 1;
548 if (envPos < 0)
549 envPos = 0;
550 }
551
552 ch->envPPos = envPos;
553 }
554 }
555
556 /* -- tick-zero volume column effects --
557 ** 2nd parameter is used for a volume column quirk with the Rxy command (multiretrig)
558 */
559
560 static void v_SetVibSpeed(stmTyp *ch, uint8_t *volKol)
561 {
562 *volKol = (ch->volKolVol & 0x0F) << 2;
563 if (*volKol != 0)
564 ch->vibSpeed = *volKol;
565 }
566
567 static void v_Volume(stmTyp *ch, uint8_t *volKol)
568 {
569 *volKol -= 16;
570 if (*volKol > 64) // 8bb: no idea why FT2 has this check, this can't happen...
571 *volKol = 64;
572
573 ch->outVol = ch->realVol = *volKol;
574 ch->status |= IS_Vol + IS_QuickVol;
575 }
576
577 static void v_FineSlideDown(stmTyp *ch, uint8_t *volKol)
578 {
579 *volKol = (uint8_t)(0 - (ch->volKolVol & 0x0F)) + ch->realVol;
580 if ((int8_t)*volKol < 0)
581 *volKol = 0;
582
583 ch->outVol = ch->realVol = *volKol;
584 ch->status |= IS_Vol;
585 }
586
587 static void v_FineSlideUp(stmTyp *ch, uint8_t *volKol)
588 {
589 *volKol = (ch->volKolVol & 0x0F) + ch->realVol;
590 if (*volKol > 64)
591 *volKol = 64;
592
593 ch->outVol = ch->realVol = *volKol;
594 ch->status |= IS_Vol;
595 }
596
597 static void v_SetPan(stmTyp *ch, uint8_t *volKol)
598 {
599 *volKol <<= 4;
600
601 ch->outPan = *volKol;
602 ch->status |= IS_Pan;
603 }
604
605 // -- non-tick-zero volume column effects --
606
607 static void v_SlideDown(stmTyp *ch)
608 {
609 uint8_t newVol = (uint8_t)(0 - (ch->volKolVol & 0x0F)) + ch->realVol;
610 if ((int8_t)newVol < 0)
611 newVol = 0;
612
613 ch->outVol = ch->realVol = newVol;
614 ch->status |= IS_Vol;
615 }
616
617 static void v_SlideUp(stmTyp *ch)
618 {
619 uint8_t newVol = (ch->volKolVol & 0x0F) + ch->realVol;
620 if (newVol > 64)
621 newVol = 64;
622
623 ch->outVol = ch->realVol = newVol;
624 ch->status |= IS_Vol;
625 }
626
627 static void v_Vibrato(stmTyp *ch)
628 {
629 const uint8_t param = ch->volKolVol & 0xF;
630 if (param > 0)
631 ch->vibDepth = param;
632
633 vibrato2(ch);
634 }
635
636 static void v_PanSlideLeft(stmTyp *ch)
637 {
638 uint16_t tmp16 = (uint8_t)(0 - (ch->volKolVol & 0x0F)) + ch->outPan;
639 if (tmp16 < 256) // 8bb: includes an FT2 bug: pan-slide-left of 0 = set pan to 0
640 tmp16 = 0;
641
642 ch->outPan = (uint8_t)tmp16;
643 ch->status |= IS_Pan;
644 }
645
646 static void v_PanSlideRight(stmTyp *ch)
647 {
648 uint16_t tmp16 = (ch->volKolVol & 0x0F) + ch->outPan;
649 if (tmp16 > 255)
650 tmp16 = 255;
651
652 ch->outPan = (uint8_t)tmp16;
653 ch->status |= IS_Pan;
654 }
655
656 static void v_TonePorta(stmTyp *ch)
657 {
658 tonePorta(ch, 0); // 8bb: the last parameter is actually not used in tonePorta()
659 }
660
661 static void v_dummy(stmTyp *ch)
662 {
663 (void)ch;
664 return;
665 }
666
667 static void v_dummy2(stmTyp *ch, uint8_t *volKol)
668 {
669 (void)ch;
670 (void)volKol;
671 return;
672 }
673
674 static const volKolEfxRoutine VJumpTab_TickNonZero[16] =
675 {
676 v_dummy, v_dummy, v_dummy, v_dummy,
677 v_dummy, v_dummy, v_SlideDown, v_SlideUp,
678 v_dummy, v_dummy, v_dummy, v_Vibrato,
679 v_dummy, v_PanSlideLeft, v_PanSlideRight, v_TonePorta
680 };
681
682 static const volKolEfxRoutine2 VJumpTab_TickZero[16] =
683 {
684 v_dummy2, v_Volume, v_Volume, v_Volume,
685 v_Volume, v_Volume, v_dummy2, v_dummy2,
686 v_FineSlideDown, v_FineSlideUp, v_SetVibSpeed, v_dummy2,
687 v_SetPan, v_dummy2, v_dummy2, v_dummy2
688 };
689
690 static void setPan(stmTyp *ch, uint8_t param)
691 {
692 ch->outPan = param;
693 ch->status |= IS_Pan;
694 }
695
696 static void setVol(stmTyp *ch, uint8_t param)
697 {
698 if (param > 64)
699 param = 64;
700
701 ch->outVol = ch->realVol = param;
702 ch->status |= IS_Vol + IS_QuickVol;
703 }
704
705 static void xFinePorta(stmTyp *ch, uint8_t param)
706 {
707 const uint8_t type = param >> 4;
708 param &= 0x0F;
709
710 if (type == 0x1) // extra fine porta up
711 {
712 if (param == 0)
713 param = ch->ePortaUpSpeed;
714
715 ch->ePortaUpSpeed = param;
716
717 uint16_t newPeriod = ch->realPeriod;
718
719 newPeriod -= param;
720 if ((int16_t)newPeriod < 1)
721 newPeriod = 1;
722
723 ch->outPeriod = ch->realPeriod = newPeriod;
724 ch->status |= IS_Period;
725 }
726 else if (type == 0x2) // extra fine porta down
727 {
728 if (param == 0)
729 param = ch->ePortaDownSpeed;
730
731 ch->ePortaDownSpeed = param;
732
733 uint16_t newPeriod = ch->realPeriod;
734
735 newPeriod += param;
736 if ((int16_t)newPeriod > MAX_FRQ-1) // 8bb: FT2 bug, should've been unsigned comparison!
737 newPeriod = MAX_FRQ-1;
738
739 ch->outPeriod = ch->realPeriod = newPeriod;
740 ch->status |= IS_Period;
741 }
742 }
743
744 static void doMultiRetrig(stmTyp *ch, uint8_t param) // 8bb: "param" is never used (needed for efx jumptable structure)
745 {
746 uint8_t cnt = ch->retrigCnt + 1;
747 if (cnt < ch->retrigSpeed)
748 {
749 ch->retrigCnt = cnt;
750 return;
751 }
752
753 ch->retrigCnt = 0;
754
755 int16_t vol = ch->realVol;
756 switch (ch->retrigVol)
757 {
758 case 0x1: vol -= 1; break;
759 case 0x2: vol -= 2; break;
760 case 0x3: vol -= 4; break;
761 case 0x4: vol -= 8; break;
762 case 0x5: vol -= 16; break;
763 case 0x6: vol = (vol >> 1) + (vol >> 3) + (vol >> 4); break;
764 case 0x7: vol >>= 1; break;
765 case 0x8: break; // 8bb: does not change the volume
766 case 0x9: vol += 1; break;
767 case 0xA: vol += 2; break;
768 case 0xB: vol += 4; break;
769 case 0xC: vol += 8; break;
770 case 0xD: vol += 16; break;
771 case 0xE: vol = (vol >> 1) + vol; break;
772 case 0xF: vol += vol; break;
773 default: break;
774 }
775 vol = CLAMP(vol, 0, 64);
776
777 ch->realVol = (uint8_t)vol;
778 ch->outVol = ch->realVol;
779
780 if (ch->volKolVol >= 0x10 && ch->volKolVol <= 0x50) // 8bb: Set Volume (volume column)
781 {
782 ch->outVol = ch->volKolVol - 0x10;
783 ch->realVol = ch->outVol;
784 }
785 else if (ch->volKolVol >= 0xC0 && ch->volKolVol <= 0xCF) // 8bb: Set Panning (volume column)
786 {
787 ch->outPan = (ch->volKolVol & 0x0F) << 4;
788 }
789
790 startTone(0, 0, 0, ch);
791
792 (void)param;
793 }
794
795 static void multiRetrig(stmTyp *ch, uint8_t param, uint8_t volumeColumnData)
796 {
797 uint8_t tmpParam;
798
799 tmpParam = param & 0x0F;
800 if (tmpParam == 0)
801 tmpParam = ch->retrigSpeed;
802
803 ch->retrigSpeed = tmpParam;
804
805 tmpParam = param >> 4;
806 if (tmpParam == 0)
807 tmpParam = ch->retrigVol;
808
809 ch->retrigVol = tmpParam;
810
811 if (volumeColumnData == 0)
812 doMultiRetrig(ch, 0); // 8bb: the second parameter is never used (needed for efx jumptable structure)
813 }
814
815 static const efxRoutine JumpTab_TickZero[36] =
816 {
817 dummy, // 0
818 dummy, // 1
819 dummy, // 2
820 dummy, // 3
821 dummy, // 4
822 dummy, // 5
823 dummy, // 6
824 dummy, // 7
825 setPan, // 8
826 dummy, // 9
827 dummy, // A
828 posJump, // B
829 setVol, // C
830 pattBreak, // D
831 E_Effects_TickZero, // E
832 setSpeed, // F
833 setGlobaVol, // G
834 dummy, // H
835 dummy, // I
836 dummy, // J
837 dummy, // K
838 setEnvelopePos, // L
839 dummy, // M
840 dummy, // N
841 dummy, // O
842 dummy, // P
843 dummy, // Q
844 dummy, // R
845 dummy, // S
846 dummy, // T
847 dummy, // U
848 dummy, // V
849 dummy, // W
850 xFinePorta, // X
851 dummy, // Y
852 dummy // Z
853 };
854
855 static void checkEffects(stmTyp *ch) // tick0 effect handling
856 {
857 // volume column effects
858 uint8_t newVolKol = ch->volKolVol; // 8bb: manipulated by vol. column effects, then used for multiretrig check (FT2 quirk)
859 VJumpTab_TickZero[ch->volKolVol >> 4](ch, &newVolKol);
860
861 // normal effects
862 const uint8_t param = ch->eff;
863
864 if ((ch->effTyp == 0 && param == 0) || ch->effTyp > 35)
865 return;
866
867 // 8bb: this one has to be done here instead of in the jumptable, as it needs the "newVolKol" parameter (FT2 quirk)
868 if (ch->effTyp == 27) // 8bb: Rxy - Multi Retrig
869 {
870 multiRetrig(ch, param, newVolKol);
871 return;
872 }
873
874 JumpTab_TickZero[ch->effTyp](ch, ch->eff);
875 }
876
877 static void fixTonePorta(stmTyp *ch, const tonTyp *p, uint8_t inst)
878 {
879 if (p->ton > 0)
880 {
881 if (p->ton == NOTE_KEYOFF)
882 {
883 keyOff(ch);
884 }
885 else
886 {
887 const uint16_t portaTmp = (((p->ton-1) + ch->relTonNr) << 4) + (((int8_t)ch->fineTune >> 3) + 16);
888 if (portaTmp < MAX_NOTES)
889 {
890 ch->wantPeriod = note2Period[portaTmp];
891
892 if (ch->wantPeriod == ch->realPeriod)
893 ch->portaDir = 0;
894 else if (ch->wantPeriod > ch->realPeriod)
895 ch->portaDir = 1;
896 else
897 ch->portaDir = 2;
898 }
899 }
900 }
901
902 if (inst > 0)
903 {
904 retrigVolume(ch);
905
906 if (p->ton != NOTE_KEYOFF)
907 retrigEnvelopeVibrato(ch);
908 }
909 }
910
911 static void getNewNote(stmTyp *ch, const tonTyp *p)
912 {
913 ch->volKolVol = p->vol;
914
915 if (ch->effTyp == 0)
916 {
917 if (ch->eff != 0) // 8bb: we have an arpeggio (0xy) running, set period back
918 {
919 ch->outPeriod = ch->realPeriod;
920 ch->status |= IS_Period;
921 }
922 }
923 else
924 {
925 // 8bb: if we have a vibrato (4xy/6xy) on previous row (ch) that ends at current row (p), set period back
926 if ((ch->effTyp == 4 || ch->effTyp == 6) && (p->effTyp != 4 && p->effTyp != 6))
927 {
928 ch->outPeriod = ch->realPeriod;
929 ch->status |= IS_Period;
930 }
931 }
932
933 ch->effTyp = p->effTyp;
934 ch->eff = p->eff;
935 ch->tonTyp = (p->instr << 8) | p->ton;
936
937 // 8bb: 'inst' var is used for later if-checks
938 uint8_t inst = p->instr;
939 if (inst > 0)
940 {
941 if (inst <= 128)
942 ch->instrNr = inst;
943 else
944 inst = 0;
945 }
946
947 bool checkEfx = true;
948 if (p->effTyp == 0x0E) // 8bb: check for EDx (Note Delay) and E90 (Retrigger Note)
949 {
950 if (p->eff >= 0xD1 && p->eff <= 0xDF) // 8bb: ED1..EDF (Note Delay)
951 return;
952 else if (p->eff == 0x90) // 8bb: E90 (Retrigger Note)
953 checkEfx = false;
954 }
955
956 if (checkEfx)
957 {
958 if ((ch->volKolVol & 0xF0) == 0xF0) // 8bb: Portamento (volume column)
959 {
960 const uint8_t volKolParam = ch->volKolVol & 0x0F;
961 if (volKolParam > 0)
962 ch->portaSpeed = volKolParam << 6;
963
964 fixTonePorta(ch, p, inst);
965 checkEffects(ch);
966 return;
967 }
968
969 if (p->effTyp == 3 || p->effTyp == 5) // 8bb: Portamento (3xx/5xx)
970 {
971 if (p->effTyp != 5 && p->eff != 0)
972 ch->portaSpeed = p->eff << 2;
973
974 fixTonePorta(ch, p, inst);
975 checkEffects(ch);
976 return;
977 }
978
979 if (p->effTyp == 0x14 && p->eff == 0) // 8bb: K00 (Key Off - only handle tick 0 here)
980 {
981 keyOff(ch);
982
983 if (inst)
984 retrigVolume(ch);
985
986 checkEffects(ch);
987 return;
988 }
989
990 if (p->ton == 0)
991 {
992 if (inst > 0)
993 {
994 retrigVolume(ch);
995 retrigEnvelopeVibrato(ch);
996 }
997
998 checkEffects(ch);
999 return;
1000 }
1001 }
1002
1003 if (p->ton == NOTE_KEYOFF)
1004 keyOff(ch);
1005 else
1006 startTone(p->ton, p->effTyp, p->eff, ch);
1007
1008 if (inst > 0)
1009 {
1010 retrigVolume(ch);
1011 if (p->ton != NOTE_KEYOFF)
1012 retrigEnvelopeVibrato(ch);
1013 }
1014
1015 checkEffects(ch);
1016 }
1017
1018 static void fixaEnvelopeVibrato(stmTyp *ch)
1019 {
1020 bool envInterpolateFlag, envDidInterpolate;
1021 uint8_t envPos;
1022 int16_t autoVibVal;
1023 uint16_t autoVibAmp, envVal;
1024 uint32_t vol;
1025
1026 instrTyp *ins = ch->instrSeg;
1027
1028 // *** FADEOUT ***
1029 if (!ch->envSustainActive)
1030 {
1031 ch->status |= IS_Vol;
1032
1033 if (ch->fadeOutAmp >= ch->fadeOutSpeed)
1034 {
1035 ch->fadeOutAmp -= ch->fadeOutSpeed;
1036 }
1037 else
1038 {
1039 ch->fadeOutAmp = 0;
1040 ch->fadeOutSpeed = 0;
1041 }
1042 }
1043
1044 if (!ch->mute)
1045 {
1046 // *** VOLUME ENVELOPE ***
1047 envVal = 0;
1048 if (ins->envVTyp & ENV_ENABLED)
1049 {
1050 envDidInterpolate = false;
1051 envPos = ch->envVPos;
1052
1053 if (++ch->envVCnt == ins->envVP[envPos][0])
1054 {
1055 ch->envVAmp = ins->envVP[envPos][1] << 8;
1056
1057 envPos++;
1058 if (ins->envVTyp & ENV_LOOP)
1059 {
1060 envPos--;
1061
1062 if (envPos == ins->envVRepE)
1063 {
1064 if (!(ins->envVTyp & ENV_SUSTAIN) || envPos != ins->envVSust || ch->envSustainActive)
1065 {
1066 envPos = ins->envVRepS;
1067
1068 ch->envVCnt = ins->envVP[envPos][0];
1069 ch->envVAmp = ins->envVP[envPos][1] << 8;
1070 }
1071 }
1072
1073 envPos++;
1074 }
1075
1076 if (envPos < ins->envVPAnt)
1077 {
1078 envInterpolateFlag = true;
1079 if ((ins->envVTyp & ENV_SUSTAIN) && ch->envSustainActive)
1080 {
1081 if (envPos-1 == ins->envVSust)
1082 {
1083 envPos--;
1084 ch->envVIPValue = 0;
1085 envInterpolateFlag = false;
1086 }
1087 }
1088
1089 if (envInterpolateFlag)
1090 {
1091 ch->envVPos = envPos;
1092
1093 ch->envVIPValue = 0;
1094 if (ins->envVP[envPos][0] > ins->envVP[envPos-1][0])
1095 {
1096 ch->envVIPValue = (ins->envVP[envPos][1] - ins->envVP[envPos-1][1]) << 8;
1097 ch->envVIPValue /= (ins->envVP[envPos][0] - ins->envVP[envPos-1][0]);
1098
1099 envVal = ch->envVAmp;
1100 envDidInterpolate = true;
1101 }
1102 }
1103 }
1104 else
1105 {
1106 ch->envVIPValue = 0;
1107 }
1108 }
1109
1110 if (!envDidInterpolate)
1111 {
1112 ch->envVAmp += ch->envVIPValue;
1113
1114 envVal = ch->envVAmp;
1115 if (envVal > 64*256)
1116 {
1117 if (envVal > 128*256)
1118 envVal = 64*256;
1119 else
1120 envVal = 0;
1121
1122 ch->envVIPValue = 0;
1123 }
1124 }
1125
1126 envVal >>= 8;
1127
1128 vol = (envVal * ch->outVol * ch->fadeOutAmp) >> (16+2);
1129 vol = (vol * song.globVol) >> 7;
1130
1131 ch->status |= IS_Vol; // 8bb: this updates vol on every tick (because vol envelope is enabled)
1132 }
1133 else
1134 {
1135 vol = ((ch->outVol << 4) * ch->fadeOutAmp) >> 16;
1136 vol = (vol * song.globVol) >> 7;
1137 }
1138
1139 ch->finalVol = (uint16_t)vol; // 0..256
1140 }
1141 else
1142 {
1143 ch->finalVol = 0;
1144 }
1145
1146 // *** PANNING ENVELOPE ***
1147
1148 envVal = 0;
1149 if (ins->envPTyp & ENV_ENABLED)
1150 {
1151 envDidInterpolate = false;
1152 envPos = ch->envPPos;
1153
1154 if (++ch->envPCnt == ins->envPP[envPos][0])
1155 {
1156 ch->envPAmp = ins->envPP[envPos][1] << 8;
1157
1158 envPos++;
1159 if (ins->envPTyp & ENV_LOOP)
1160 {
1161 envPos--;
1162
1163 if (envPos == ins->envPRepE)
1164 {
1165 if (!(ins->envPTyp & ENV_SUSTAIN) || envPos != ins->envPSust || ch->envSustainActive)
1166 {
1167 envPos = ins->envPRepS;
1168
1169 ch->envPCnt = ins->envPP[envPos][0];
1170 ch->envPAmp = ins->envPP[envPos][1] << 8;
1171 }
1172 }
1173
1174 envPos++;
1175 }
1176
1177 if (envPos < ins->envPPAnt)
1178 {
1179 envInterpolateFlag = true;
1180 if ((ins->envPTyp & ENV_SUSTAIN) && ch->envSustainActive)
1181 {
1182 if (envPos-1 == ins->envPSust)
1183 {
1184 envPos--;
1185 ch->envPIPValue = 0;
1186 envInterpolateFlag = false;
1187 }
1188 }
1189
1190 if (envInterpolateFlag)
1191 {
1192 ch->envPPos = envPos;
1193
1194 ch->envPIPValue = 0;
1195 if (ins->envPP[envPos][0] > ins->envPP[envPos-1][0])
1196 {
1197 ch->envPIPValue = (ins->envPP[envPos][1] - ins->envPP[envPos-1][1]) << 8;
1198 ch->envPIPValue /= (ins->envPP[envPos][0] - ins->envPP[envPos-1][0]);
1199
1200 envVal = ch->envPAmp;
1201 envDidInterpolate = true;
1202 }
1203 }
1204 }
1205 else
1206 {
1207 ch->envPIPValue = 0;
1208 }
1209 }
1210
1211 if (!envDidInterpolate)
1212 {
1213 ch->envPAmp += ch->envPIPValue;
1214
1215 envVal = ch->envPAmp;
1216 if (envVal > 64*256)
1217 {
1218 if (envVal > 128*256)
1219 envVal = 64*256;
1220 else
1221 envVal = 0;
1222
1223 ch->envPIPValue = 0;
1224 }
1225 }
1226
1227 int16_t panTmp = ch->outPan - 128;
1228 if (panTmp > 0)
1229 panTmp = 0 - panTmp;
1230 panTmp += 128;
1231
1232 panTmp <<= 3;
1233 envVal -= 32*256;
1234
1235 ch->finalPan = ch->outPan + (uint8_t)(((int16_t)envVal * panTmp) >> 16);
1236 ch->status |= IS_Pan;
1237 }
1238 else
1239 {
1240 ch->finalPan = ch->outPan;
1241 }
1242
1243 // *** AUTO VIBRATO ***
1244 if (ins->vibDepth > 0)
1245 {
1246 if (ch->eVibSweep > 0)
1247 {
1248 autoVibAmp = ch->eVibSweep;
1249 if (ch->envSustainActive)
1250 {
1251 autoVibAmp += ch->eVibAmp;
1252 if ((autoVibAmp >> 8) > ins->vibDepth)
1253 {
1254 autoVibAmp = ins->vibDepth << 8;
1255 ch->eVibSweep = 0;
1256 }
1257
1258 ch->eVibAmp = autoVibAmp;
1259 }
1260 }
1261 else
1262 {
1263 autoVibAmp = ch->eVibAmp;
1264 }
1265
1266 ch->eVibPos += ins->vibRate;
1267
1268 if (ins->vibTyp == 1) autoVibVal = (ch->eVibPos > 127) ? 64 : -64; // square
1269 else if (ins->vibTyp == 2) autoVibVal = (((ch->eVibPos >> 1) + 64) & 127) - 64; // ramp up
1270 else if (ins->vibTyp == 3) autoVibVal = ((-(ch->eVibPos >> 1) + 64) & 127) - 64; // ramp down
1271 else autoVibVal = vibSineTab[ch->eVibPos]; // sine
1272
1273 autoVibVal <<= 2;
1274 uint16_t tmpPeriod = (autoVibVal * (int16_t)autoVibAmp) >> 16;
1275
1276 tmpPeriod += ch->outPeriod;
1277 if (tmpPeriod >= MAX_FRQ)
1278 tmpPeriod = 0; // 8bb: yes, FT2 does this (!)
1279
1280 ch->finalPeriod = tmpPeriod;
1281 ch->status |= IS_Period;
1282 }
1283 else
1284 {
1285 ch->finalPeriod = ch->outPeriod;
1286 }
1287 }
1288
1289 // 8bb: converts period to note number, for arpeggio and portamento (in semitone-slide mode)
1290 static uint16_t relocateTon(uint16_t period, uint8_t arpNote, stmTyp *ch)
1291 {
1292 int32_t tmpPeriod;
1293
1294 const int32_t fineTune = ((int8_t)ch->fineTune >> 3) + 16;
1295
1296 /* 8bb: FT2 bug, should've been 10*12*16. Notes above B-7 (95) will have issues.
1297 ** You can only achieve such high notes by having a high relative note value
1298 ** in the sample.
1299 */
1300 int32_t hiPeriod = 8*12*16;
1301
1302 int32_t loPeriod = 0;
1303
1304 for (int32_t i = 0; i < 8; i++)
1305 {
1306 tmpPeriod = (((loPeriod + hiPeriod) >> 1) & ~15) + fineTune;
1307
1308 int32_t lookUp = tmpPeriod - 8;
1309 if (lookUp < 0)
1310 lookUp = 0; // 8bb: safety fix (C-0 w/ ftune <= -65). This buggy read seems to return 0 in FT2 (TODO: verify)
1311
1312 if (period >= note2Period[lookUp])
1313 hiPeriod = (tmpPeriod - fineTune) & ~15;
1314 else
1315 loPeriod = (tmpPeriod - fineTune) & ~15;
1316 }
1317
1318 tmpPeriod = loPeriod + fineTune + (arpNote << 4);
1319 if (tmpPeriod >= (8*12*16+15)-1) // 8bb: FT2 bug, should've been 10*12*16+16 (also notice the +2 difference)
1320 tmpPeriod = (8*12*16+16)-1;
1321
1322 return note2Period[tmpPeriod];
1323 }
1324
1325 static void vibrato2(stmTyp *ch)
1326 {
1327 uint8_t tmpVib = (ch->vibPos >> 2) & 0x1F;
1328
1329 switch (ch->waveCtrl & 3)
1330 {
1331 // 0: sine
1332 case 0: tmpVib = vibTab[tmpVib]; break;
1333
1334 // 1: ramp
1335 case 1:
1336 {
1337 tmpVib <<= 3;
1338 if ((int8_t)ch->vibPos < 0)
1339 tmpVib = ~tmpVib;
1340 }
1341 break;
1342
1343 // 2/3: square
1344 default: tmpVib = 255; break;
1345 }
1346
1347 tmpVib = (tmpVib * ch->vibDepth) >> 5;
1348
1349 if ((int8_t)ch->vibPos < 0)
1350 ch->outPeriod = ch->realPeriod - tmpVib;
1351 else
1352 ch->outPeriod = ch->realPeriod + tmpVib;
1353
1354 ch->status |= IS_Period;
1355 ch->vibPos += ch->vibSpeed;
1356 }
1357
1358 static void arp(stmTyp *ch, uint8_t param)
1359 {
1360 /* 8bb: The original arpTab table only supports 16 ticks, so it can and will overflow.
1361 ** I have added overflown values to the table so that we can handle up to 256 ticks.
1362 ** The added overflow entries are accurate to the overflow-read in FT2.08/FT2.09.
1363 */
1364 const uint8_t tick = arpTab[song.timer & 0xFF];
1365
1366 if (tick == 0)
1367 {
1368 ch->outPeriod = ch->realPeriod;
1369 }
1370 else
1371 {
1372 const uint8_t note = (tick == 1) ? (param >> 4) : (param & 0x0F);
1373 ch->outPeriod = relocateTon(ch->realPeriod, note, ch);
1374 }
1375
1376 ch->status |= IS_Period;
1377 }
1378
1379 static void portaUp(stmTyp *ch, uint8_t param)
1380 {
1381 if (param == 0)
1382 param = ch->portaUpSpeed;
1383
1384 ch->portaUpSpeed = param;
1385
1386 ch->realPeriod -= param << 2;
1387 if ((int16_t)ch->realPeriod < 1)
1388 ch->realPeriod = 1;
1389
1390 ch->outPeriod = ch->realPeriod;
1391 ch->status |= IS_Period;
1392 }
1393
1394 static void portaDown(stmTyp *ch, uint8_t param)
1395 {
1396 if (param == 0)
1397 param = ch->portaDownSpeed;
1398
1399 ch->portaDownSpeed = param;
1400
1401 ch->realPeriod += param << 2;
1402 if ((int16_t)ch->realPeriod > MAX_FRQ-1) // 8bb: FT2 bug, should've been unsigned comparison!
1403 ch->realPeriod = MAX_FRQ-1;
1404
1405 ch->outPeriod = ch->realPeriod;
1406 ch->status |= IS_Period;
1407 }
1408
1409 static void tonePorta(stmTyp *ch, uint8_t param) // 8bb: param is a placeholder (not used)
1410 {
1411 if (ch->portaDir == 0)
1412 return;
1413
1414 if (ch->portaDir > 1)
1415 {
1416 ch->realPeriod -= ch->portaSpeed;
1417 if ((int16_t)ch->realPeriod <= (int16_t)ch->wantPeriod)
1418 {
1419 ch->portaDir = 1;
1420 ch->realPeriod = ch->wantPeriod;
1421 }
1422 }
1423 else
1424 {
1425 ch->realPeriod += ch->portaSpeed;
1426 if (ch->realPeriod >= ch->wantPeriod)
1427 {
1428 ch->portaDir = 1;
1429 ch->realPeriod = ch->wantPeriod;
1430 }
1431 }
1432
1433 if (ch->glissFunk) // 8bb: semitone-slide flag
1434 ch->outPeriod = relocateTon(ch->realPeriod, 0, ch);
1435 else
1436 ch->outPeriod = ch->realPeriod;
1437
1438 ch->status |= IS_Period;
1439
1440 (void)param;
1441 }
1442
1443 static void vibrato(stmTyp *ch, uint8_t param)
1444 {
1445 uint8_t tmp8;
1446
1447 if (ch->eff > 0)
1448 {
1449 tmp8 = param & 0x0F;
1450 if (tmp8 > 0)
1451 ch->vibDepth = tmp8;
1452
1453 tmp8 = (param & 0xF0) >> 2;
1454 if (tmp8 > 0)
1455 ch->vibSpeed = tmp8;
1456 }
1457
1458 vibrato2(ch);
1459 }
1460
1461 static void tonePlusVol(stmTyp *ch, uint8_t param)
1462 {
1463 tonePorta(ch, 0); // 8bb: the last parameter is not used in tonePorta()
1464 volume(ch, param);
1465
1466 (void)param;
1467 }
1468
1469 static void vibratoPlusVol(stmTyp *ch, uint8_t param)
1470 {
1471 vibrato2(ch);
1472 volume(ch, param);
1473
1474 (void)param;
1475 }
1476
1477 static void tremolo(stmTyp *ch, uint8_t param)
1478 {
1479 uint8_t tmp8;
1480 int16_t tremVol;
1481
1482 const uint8_t tmpEff = param;
1483 if (tmpEff > 0)
1484 {
1485 tmp8 = tmpEff & 0x0F;
1486 if (tmp8 > 0)
1487 ch->tremDepth = tmp8;
1488
1489 tmp8 = (tmpEff & 0xF0) >> 2;
1490 if (tmp8 > 0)
1491 ch->tremSpeed = tmp8;
1492 }
1493
1494 uint8_t tmpTrem = (ch->tremPos >> 2) & 0x1F;
1495 switch ((ch->waveCtrl >> 4) & 3)
1496 {
1497 // 0: sine
1498 case 0: tmpTrem = vibTab[tmpTrem]; break;
1499
1500 // 1: ramp
1501 case 1:
1502 {
1503 tmpTrem <<= 3;
1504 if ((int8_t)ch->vibPos < 0) // 8bb: FT2 bug, should've been ch->tremPos
1505 tmpTrem = ~tmpTrem;
1506 }
1507 break;
1508
1509 // 2/3: square
1510 default: tmpTrem = 255; break;
1511 }
1512 tmpTrem = (tmpTrem * ch->tremDepth) >> 6;
1513
1514 if ((int8_t)ch->tremPos < 0)
1515 {
1516 tremVol = ch->realVol - tmpTrem;
1517 if (tremVol < 0)
1518 tremVol = 0;
1519 }
1520 else
1521 {
1522 tremVol = ch->realVol + tmpTrem;
1523 if (tremVol > 64)
1524 tremVol = 64;
1525 }
1526
1527 ch->outVol = (uint8_t)tremVol;
1528 ch->status |= IS_Vol;
1529 ch->tremPos += ch->tremSpeed;
1530 }
1531
1532 static void volume(stmTyp *ch, uint8_t param) // 8bb: volume slide
1533 {
1534 if (param == 0)
1535 param = ch->volSlideSpeed;
1536
1537 ch->volSlideSpeed = param;
1538
1539 uint8_t newVol = ch->realVol;
1540 if ((param & 0xF0) == 0)
1541 {
1542 newVol -= param;
1543 if ((int8_t)newVol < 0)
1544 newVol = 0;
1545 }
1546 else
1547 {
1548 param >>= 4;
1549
1550 newVol += param;
1551 if (newVol > 64)
1552 newVol = 64;
1553 }
1554
1555 ch->outVol = ch->realVol = newVol;
1556 ch->status |= IS_Vol;
1557 }
1558
1559 static void globalVolSlide(stmTyp *ch, uint8_t param)
1560 {
1561 if (param == 0)
1562 param = ch->globVolSlideSpeed;
1563
1564 ch->globVolSlideSpeed = param;
1565
1566 uint8_t newVol = (uint8_t)song.globVol;
1567 if ((param & 0xF0) == 0)
1568 {
1569 newVol -= param;
1570 if ((int8_t)newVol < 0)
1571 newVol = 0;
1572 }
1573 else
1574 {
1575 param >>= 4;
1576
1577 newVol += param;
1578 if (newVol > 64)
1579 newVol = 64;
1580 }
1581
1582 song.globVol = newVol;
1583
1584 stmTyp *c = stm;
1585 for (int32_t i = 0; i < song.antChn; i++, c++) // 8bb: this updates the volume for all voices
1586 c->status |= IS_Vol;
1587 }
1588
1589 static void keyOffCmd(stmTyp *ch, uint8_t param)
1590 {
1591 if ((uint8_t)(song.tempo-song.timer) == (param & 31))
1592 keyOff(ch);
1593 }
1594
1595 static void panningSlide(stmTyp *ch, uint8_t param)
1596 {
1597 if (param == 0)
1598 param = ch->panningSlideSpeed;
1599
1600 ch->panningSlideSpeed = param;
1601
1602 int16_t newPan = (int16_t)ch->outPan;
1603 if ((param & 0xF0) == 0)
1604 {
1605 newPan -= param;
1606 if (newPan < 0)
1607 newPan = 0;
1608 }
1609 else
1610 {
1611 param >>= 4;
1612
1613 newPan += param;
1614 if (newPan > 255)
1615 newPan = 255;
1616 }
1617
1618 ch->outPan = (uint8_t)newPan;
1619 ch->status |= IS_Pan;
1620 }
1621
1622 static void tremor(stmTyp *ch, uint8_t param)
1623 {
1624 if (param == 0)
1625 param = ch->tremorSave;
1626
1627 ch->tremorSave = param;
1628
1629 uint8_t tremorSign = ch->tremorPos & 0x80;
1630 uint8_t tremorData = ch->tremorPos & 0x7F;
1631
1632 tremorData--;
1633 if ((int8_t)tremorData < 0)
1634 {
1635 if (tremorSign == 0x80)
1636 {
1637 tremorSign = 0x00;
1638 tremorData = param & 0x0F;
1639 }
1640 else
1641 {
1642 tremorSign = 0x80;
1643 tremorData = param >> 4;
1644 }
1645 }
1646
1647 ch->tremorPos = tremorSign | tremorData;
1648 ch->outVol = (tremorSign == 0x80) ? ch->realVol : 0;
1649 ch->status |= IS_Vol + IS_QuickVol;
1650 }
1651
1652 static void retrigNote(stmTyp *ch, uint8_t param)
1653 {
1654 if (param == 0) // 8bb: E9x with a param of zero is handled in getNewNote()
1655 return;
1656
1657 if ((song.tempo-song.timer) % param == 0)
1658 {
1659 startTone(0, 0, 0, ch);
1660 retrigEnvelopeVibrato(ch);
1661 }
1662 }
1663
1664 static void noteCut(stmTyp *ch, uint8_t param)
1665 {
1666 if ((uint8_t)(song.tempo-song.timer) == param)
1667 {
1668 ch->outVol = ch->realVol = 0;
1669 ch->status |= IS_Vol + IS_QuickVol;
1670 }
1671 }
1672
1673 static void noteDelay(stmTyp *ch, uint8_t param)
1674 {
1675 if ((uint8_t)(song.tempo-song.timer) == param)
1676 {
1677 startTone(ch->tonTyp & 0xFF, 0, 0, ch);
1678
1679 if ((ch->tonTyp & 0xFF00) > 0) // 8bb: do we have an instrument number?
1680 retrigVolume(ch);
1681
1682 retrigEnvelopeVibrato(ch);
1683
1684 if (ch->volKolVol >= 0x10 && ch->volKolVol <= 0x50) // 8bb: Set Volume (volume column)
1685 {
1686 ch->outVol = ch->volKolVol - 16;
1687 ch->realVol = ch->outVol;
1688 }
1689 else if (ch->volKolVol >= 0xC0 && ch->volKolVol <= 0xCF) // 8bb: Set Panning (volume column)
1690 {
1691 ch->outPan = (ch->volKolVol & 0x0F) << 4;
1692 }
1693 }
1694 }
1695
1696 static const efxRoutine EJumpTab_TickNonZero[16] =
1697 {
1698 dummy, // 0
1699 dummy, // 1
1700 dummy, // 2
1701 dummy, // 3
1702 dummy, // 4
1703 dummy, // 5
1704 dummy, // 6
1705 dummy, // 7
1706 dummy, // 8
1707 retrigNote, // 9
1708 dummy, // A
1709 dummy, // B
1710 noteCut, // C
1711 noteDelay, // D
1712 dummy, // E
1713 dummy // F
1714 };
1715
1716 static void E_Effects_TickNonZero(stmTyp *ch, uint8_t param)
1717 {
1718 EJumpTab_TickNonZero[param >> 4](ch, param & 0xF);
1719 }
1720
1721 static const efxRoutine JumpTab_TickNonZero[36] =
1722 {
1723 arp, // 0
1724 portaUp, // 1
1725 portaDown, // 2
1726 tonePorta, // 3
1727 vibrato, // 4
1728 tonePlusVol, // 5
1729 vibratoPlusVol, // 6
1730 tremolo, // 7
1731 dummy, // 8
1732 dummy, // 9
1733 volume, // A
1734 dummy, // B
1735 dummy, // C
1736 dummy, // D
1737 E_Effects_TickNonZero, // E
1738 dummy, // F
1739 dummy, // G
1740 globalVolSlide, // H
1741 dummy, // I
1742 dummy, // J
1743 keyOffCmd, // K
1744 dummy, // L
1745 dummy, // M
1746 dummy, // N
1747 dummy, // O
1748 panningSlide, // P
1749 dummy, // Q
1750 doMultiRetrig, // R
1751 dummy, // S
1752 tremor, // T
1753 dummy, // U
1754 dummy, // V
1755 dummy, // W
1756 dummy, // X
1757 dummy, // Y
1758 dummy // Z
1759 };
1760
1761 static void doEffects(stmTyp *ch) // tick>0 effect handling
1762 {
1763 const uint8_t volKolEfx = ch->volKolVol >> 4;
1764 if (volKolEfx > 0)
1765 VJumpTab_TickNonZero[volKolEfx](ch);
1766
1767 if ((ch->eff == 0 && ch->effTyp == 0) || ch->effTyp > 35)
1768 return;
1769
1770 JumpTab_TickNonZero[ch->effTyp](ch, ch->eff);
1771 }
1772
1773 static void getNextPos(void)
1774 {
1775 song.pattPos++;
1776
1777 if (song.pattDelTime > 0)
1778 {
1779 song.pattDelTime2 = song.pattDelTime;
1780 song.pattDelTime = 0;
1781 }
1782
1783 if (song.pattDelTime2 > 0)
1784 {
1785 song.pattDelTime2--;
1786 if (song.pattDelTime2 > 0)
1787 song.pattPos--;
1788 }
1789
1790 if (song.pBreakFlag)
1791 {
1792 song.pBreakFlag = false;
1793 song.pattPos = song.pBreakPos;
1794 }
1795
1796 if (song.pattPos >= song.pattLen || song.posJumpFlag)
1797 {
1798 song.pattPos = song.pBreakPos;
1799 song.pBreakPos = 0;
1800 song.posJumpFlag = false;
1801
1802 song.songPos++;
1803 if (song.songPos >= song.len)
1804 song.songPos = song.repS;
1805
1806 song.pattNr = song.songTab[(uint8_t)song.songPos];
1807 song.pattLen = pattLens[(uint8_t)song.pattNr];
1808 }
1809 }
1810
1811 void mainPlayer(void)
1812 {
1813 if (musicPaused)
1814 return;
1815
1816 bool tickZero = false;
1817
1818 song.timer--;
1819 if (song.timer == 0)
1820 {
1821 song.timer = song.tempo;
1822 tickZero = true;
1823 }
1824
1825 const bool readNewNote = tickZero && (song.pattDelTime2 == 0);
1826 if (readNewNote)
1827 {
1828 const tonTyp *pattPtr = nilPatternLine;
1829 if (patt[song.pattNr] != NULL)
1830 pattPtr = &patt[song.pattNr][song.pattPos * song.antChn];
1831
1832 stmTyp *c = stm;
1833 for (uint8_t i = 0; i < song.antChn; i++, c++, pattPtr++)
1834 {
1835 PMPTmpActiveChannel = i; // 8bb: for P_StartTone()
1836 getNewNote(c, pattPtr);
1837 fixaEnvelopeVibrato(c);
1838 }
1839 }
1840 else
1841 {
1842 stmTyp *c = stm;
1843 for (uint8_t i = 0; i < song.antChn; i++, c++)
1844 {
1845 PMPTmpActiveChannel = i; // 8bb: for P_StartTone()
1846 doEffects(c);
1847 fixaEnvelopeVibrato(c);
1848 }
1849 }
1850
1851 if (song.timer == 1)
1852 getNextPos();
1853 }