Mercurial > codedump
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 } |
