126
|
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 }
|