| 
									
										
										
										
											2022-02-14 22:12:20 -05:00
										 |  |  |  | /**
 | 
					
						
							|  |  |  |  |  * Furnace Tracker - multi-system chiptune tracker | 
					
						
							| 
									
										
										
										
											2025-01-28 18:49:19 -05:00
										 |  |  |  |  * Copyright (C) 2021-2025 tildearrow and contributors | 
					
						
							| 
									
										
										
										
											2022-02-14 22:12:20 -05:00
										 |  |  |  |  * | 
					
						
							|  |  |  |  |  * This program is free software; you can redistribute it and/or modify | 
					
						
							|  |  |  |  |  * it under the terms of the GNU General Public License as published by | 
					
						
							|  |  |  |  |  * the Free Software Foundation; either version 2 of the License, or | 
					
						
							|  |  |  |  |  * (at your option) any later version. | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * This program is distributed in the hope that it will be useful, | 
					
						
							|  |  |  |  |  * but WITHOUT ANY WARRANTY; without even the implied warranty of | 
					
						
							|  |  |  |  |  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | 
					
						
							|  |  |  |  |  * GNU General Public License for more details. | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * You should have received a copy of the GNU General Public License along | 
					
						
							|  |  |  |  |  * with this program; if not, write to the Free Software Foundation, Inc., | 
					
						
							|  |  |  |  |  * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA. | 
					
						
							|  |  |  |  |  */ | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-08-21 16:07:28 -04:00
										 |  |  |  | #define _USE_MATH_DEFINES
 | 
					
						
							| 
									
										
										
										
											2022-08-04 01:51:47 -04:00
										 |  |  |  | #include "dispatch.h"
 | 
					
						
							| 
									
										
										
										
											2022-08-26 04:03:36 -04:00
										 |  |  |  | #include "song.h"
 | 
					
						
							| 
									
										
										
										
											2021-05-11 16:08:08 -04:00
										 |  |  |  | #include "engine.h"
 | 
					
						
							| 
									
										
										
										
											2021-12-09 02:38:55 -05:00
										 |  |  |  | #include "instrument.h"
 | 
					
						
							| 
									
										
										
										
											2021-05-11 16:08:08 -04:00
										 |  |  |  | #include "safeReader.h"
 | 
					
						
							| 
									
										
										
										
											2023-09-06 15:23:47 -04:00
										 |  |  |  | #include "workPool.h"
 | 
					
						
							| 
									
										
										
										
											2021-05-11 16:08:08 -04:00
										 |  |  |  | #include "../ta-log.h"
 | 
					
						
							| 
									
										
										
										
											2022-01-20 05:04:03 -05:00
										 |  |  |  | #include "../fileutils.h"
 | 
					
						
							| 
									
										
										
										
											2022-05-22 20:01:50 -04:00
										 |  |  |  | #ifdef HAVE_SDL2
 | 
					
						
							| 
									
										
										
										
											2022-06-23 17:25:51 -04:00
										 |  |  |  | #include "../audio/sdlAudio.h"
 | 
					
						
							| 
									
										
										
										
											2022-05-22 20:01:50 -04:00
										 |  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2021-12-19 16:52:04 -05:00
										 |  |  |  | #include <stdexcept>
 | 
					
						
							| 
									
										
										
										
											2021-06-09 04:33:03 -04:00
										 |  |  |  | #ifdef HAVE_JACK
 | 
					
						
							|  |  |  |  | #include "../audio/jack.h"
 | 
					
						
							|  |  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2023-08-30 18:32:51 -04:00
										 |  |  |  | #ifdef HAVE_PA
 | 
					
						
							|  |  |  |  | #include "../audio/pa.h"
 | 
					
						
							|  |  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2024-04-23 05:38:08 -04:00
										 |  |  |  | #include "../audio/pipe.h"
 | 
					
						
							| 
									
										
										
										
											2021-05-14 15:16:48 -04:00
										 |  |  |  | #include <math.h>
 | 
					
						
							| 
									
										
										
										
											2022-07-19 18:01:19 -04:00
										 |  |  |  | #include <float.h>
 | 
					
						
							| 
									
										
										
										
											2021-12-17 03:33:12 -05:00
										 |  |  |  | #include <fmt/printf.h>
 | 
					
						
							| 
									
										
										
										
											2025-02-19 17:33:06 -05:00
										 |  |  |  | #include <chrono>
 | 
					
						
							| 
									
										
										
										
											2021-05-11 16:08:08 -04:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-12 04:58:55 -04:00
										 |  |  |  | void process(void* u, float** in, float** out, int inChans, int outChans, unsigned int size) { | 
					
						
							|  |  |  |  |   ((DivEngine*)u)->nextBuf(in,out,inChans,outChans,size); | 
					
						
							| 
									
										
										
										
											2021-05-11 16:08:08 -04:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-19 19:44:05 -04:00
										 |  |  |  | const char* DivEngine::getEffectDesc(unsigned char effect, int chan, bool notNull) { | 
					
						
							| 
									
										
										
										
											2022-01-20 13:48:20 -05:00
										 |  |  |  |   switch (effect) { | 
					
						
							|  |  |  |  |     case 0x00: | 
					
						
							| 
									
										
										
										
											2024-05-27 05:02:07 -04:00
										 |  |  |  |       return _("00xy: Arpeggio"); | 
					
						
							| 
									
										
										
										
											2022-01-20 13:48:20 -05:00
										 |  |  |  |     case 0x01: | 
					
						
							| 
									
										
										
										
											2024-05-27 05:02:07 -04:00
										 |  |  |  |       return _("01xx: Pitch slide up"); | 
					
						
							| 
									
										
										
										
											2022-01-20 13:48:20 -05:00
										 |  |  |  |     case 0x02: | 
					
						
							| 
									
										
										
										
											2024-05-27 05:02:07 -04:00
										 |  |  |  |       return _("02xx: Pitch slide down"); | 
					
						
							| 
									
										
										
										
											2022-01-20 13:48:20 -05:00
										 |  |  |  |     case 0x03: | 
					
						
							| 
									
										
										
										
											2024-05-27 05:02:07 -04:00
										 |  |  |  |       return _("03xx: Portamento"); | 
					
						
							| 
									
										
										
										
											2022-01-20 13:48:20 -05:00
										 |  |  |  |     case 0x04: | 
					
						
							| 
									
										
										
										
											2024-05-27 05:02:07 -04:00
										 |  |  |  |       return _("04xy: Vibrato (x: speed; y: depth)"); | 
					
						
							| 
									
										
										
										
											2023-04-30 14:46:09 -04:00
										 |  |  |  |     case 0x05: | 
					
						
							| 
									
										
										
										
											2024-05-27 05:02:07 -04:00
										 |  |  |  |       return _("05xy: Volume slide + vibrato (compatibility only!)"); | 
					
						
							| 
									
										
										
										
											2023-04-30 14:46:09 -04:00
										 |  |  |  |     case 0x06: | 
					
						
							| 
									
										
										
										
											2024-05-27 05:02:07 -04:00
										 |  |  |  |       return _("06xy: Volume slide + portamento (compatibility only!)"); | 
					
						
							| 
									
										
										
										
											2022-03-14 10:50:52 -04:00
										 |  |  |  |     case 0x07: | 
					
						
							| 
									
										
										
										
											2024-05-27 05:02:07 -04:00
										 |  |  |  |       return _("07xy: Tremolo (x: speed; y: depth)"); | 
					
						
							| 
									
										
										
										
											2022-01-20 13:48:20 -05:00
										 |  |  |  |     case 0x08: | 
					
						
							| 
									
										
										
										
											2024-05-27 05:02:07 -04:00
										 |  |  |  |       return _("08xy: Set panning (x: left; y: right)"); | 
					
						
							| 
									
										
										
										
											2022-01-20 13:48:20 -05:00
										 |  |  |  |     case 0x09: | 
					
						
							| 
									
										
										
										
											2024-05-27 05:02:07 -04:00
										 |  |  |  |       return _("09xx: Set groove pattern (speed 1 if no grooves exist)"); | 
					
						
							| 
									
										
										
										
											2022-01-20 13:48:20 -05:00
										 |  |  |  |     case 0x0a: | 
					
						
							| 
									
										
										
										
											2024-05-27 05:02:07 -04:00
										 |  |  |  |       return _("0Axy: Volume slide (0y: down; x0: up)"); | 
					
						
							| 
									
										
										
										
											2022-01-20 13:48:20 -05:00
										 |  |  |  |     case 0x0b: | 
					
						
							| 
									
										
										
										
											2024-05-27 05:02:07 -04:00
										 |  |  |  |       return _("0Bxx: Jump to pattern"); | 
					
						
							| 
									
										
										
										
											2022-01-20 13:48:20 -05:00
										 |  |  |  |     case 0x0c: | 
					
						
							| 
									
										
										
										
											2024-05-27 05:02:07 -04:00
										 |  |  |  |       return _("0Cxx: Retrigger"); | 
					
						
							| 
									
										
										
										
											2022-01-20 13:48:20 -05:00
										 |  |  |  |     case 0x0d: | 
					
						
							| 
									
										
										
										
											2024-05-27 05:02:07 -04:00
										 |  |  |  |       return _("0Dxx: Jump to next pattern"); | 
					
						
							| 
									
										
										
										
											2022-01-20 13:48:20 -05:00
										 |  |  |  |     case 0x0f: | 
					
						
							| 
									
										
										
										
											2024-05-27 05:02:07 -04:00
										 |  |  |  |       return _("0Fxx: Set speed (speed 2 if no grooves exist)"); | 
					
						
							| 
									
										
										
										
											2022-04-30 00:41:14 -04:00
										 |  |  |  |     case 0x80: | 
					
						
							| 
									
										
										
										
											2024-05-27 05:02:07 -04:00
										 |  |  |  |       return _("80xx: Set panning (00: left; 80: center; FF: right)"); | 
					
						
							| 
									
										
										
										
											2022-04-30 00:41:14 -04:00
										 |  |  |  |     case 0x81: | 
					
						
							| 
									
										
										
										
											2024-05-27 05:02:07 -04:00
										 |  |  |  |       return _("81xx: Set panning (left channel)"); | 
					
						
							| 
									
										
										
										
											2022-04-30 00:41:14 -04:00
										 |  |  |  |     case 0x82: | 
					
						
							| 
									
										
										
										
											2024-05-27 05:02:07 -04:00
										 |  |  |  |       return _("82xx: Set panning (right channel)"); | 
					
						
							| 
									
										
										
										
											2024-07-07 19:55:22 -04:00
										 |  |  |  |     case 0x83: | 
					
						
							|  |  |  |  |       return _("83xy: Panning slide (x0: left; 0y: right)"); | 
					
						
							|  |  |  |  |     case 0x84: | 
					
						
							|  |  |  |  |       return _("84xy: Panbrello (x: speed; y: depth)"); | 
					
						
							| 
									
										
										
										
											2023-01-05 03:08:57 -05:00
										 |  |  |  |     case 0x88: | 
					
						
							| 
									
										
										
										
											2024-05-27 05:02:07 -04:00
										 |  |  |  |       return _("88xy: Set panning (rear channels; x: left; y: right)"); | 
					
						
							| 
									
										
										
										
											2023-01-05 03:08:57 -05:00
										 |  |  |  |       break; | 
					
						
							|  |  |  |  |     case 0x89: | 
					
						
							| 
									
										
										
										
											2024-05-27 05:02:07 -04:00
										 |  |  |  |       return _("89xx: Set panning (rear left channel)"); | 
					
						
							| 
									
										
										
										
											2023-01-05 03:08:57 -05:00
										 |  |  |  |       break; | 
					
						
							|  |  |  |  |     case 0x8a: | 
					
						
							| 
									
										
										
										
											2024-05-27 05:02:07 -04:00
										 |  |  |  |       return _("8Axx: Set panning (rear right channel)"); | 
					
						
							| 
									
										
										
										
											2023-01-05 03:08:57 -05:00
										 |  |  |  |       break; | 
					
						
							| 
									
										
										
										
											2022-02-15 01:46:03 -05:00
										 |  |  |  |     case 0xc0: case 0xc1: case 0xc2: case 0xc3: | 
					
						
							| 
									
										
										
										
											2024-05-27 05:02:07 -04:00
										 |  |  |  |       return _("Cxxx: Set tick rate (hz)"); | 
					
						
							| 
									
										
										
										
											2024-08-23 13:39:14 -04:00
										 |  |  |  |     case 0xd3: | 
					
						
							|  |  |  |  |       return _("D3xx: Volume portamento"); | 
					
						
							|  |  |  |  |     case 0xd4: | 
					
						
							|  |  |  |  |       return _("D4xx: Volume portamento (fast)"); | 
					
						
							| 
									
										
										
										
											2024-07-17 05:11:24 -04:00
										 |  |  |  |     case 0xdc: | 
					
						
							|  |  |  |  |       return _("DCxx: Delayed mute"); | 
					
						
							| 
									
										
										
										
											2022-01-20 13:48:20 -05:00
										 |  |  |  |     case 0xe0: | 
					
						
							| 
									
										
										
										
											2024-05-27 05:02:07 -04:00
										 |  |  |  |       return _("E0xx: Set arp speed"); | 
					
						
							| 
									
										
										
										
											2022-01-20 13:48:20 -05:00
										 |  |  |  |     case 0xe1: | 
					
						
							| 
									
										
										
										
											2024-05-27 05:02:07 -04:00
										 |  |  |  |       return _("E1xy: Note slide up (x: speed; y: semitones)"); | 
					
						
							| 
									
										
										
										
											2022-01-20 13:48:20 -05:00
										 |  |  |  |     case 0xe2: | 
					
						
							| 
									
										
										
										
											2024-05-27 05:02:07 -04:00
										 |  |  |  |       return _("E2xy: Note slide down (x: speed; y: semitones)"); | 
					
						
							| 
									
										
										
										
											2022-01-20 13:48:20 -05:00
										 |  |  |  |     case 0xe3: | 
					
						
							| 
									
										
										
										
											2024-06-24 07:24:14 -04:00
										 |  |  |  |       return _("E3xx: Set vibrato shape"); | 
					
						
							| 
									
										
										
										
											2022-01-20 13:48:20 -05:00
										 |  |  |  |     case 0xe4: | 
					
						
							| 
									
										
										
										
											2024-05-27 05:02:07 -04:00
										 |  |  |  |       return _("E4xx: Set vibrato range"); | 
					
						
							| 
									
										
										
										
											2022-01-20 13:48:20 -05:00
										 |  |  |  |     case 0xe5: | 
					
						
							| 
									
										
										
										
											2024-05-27 05:02:07 -04:00
										 |  |  |  |       return _("E5xx: Set pitch (80: center)"); | 
					
						
							| 
									
										
										
										
											2024-03-16 20:41:08 -04:00
										 |  |  |  |     case 0xe6: | 
					
						
							| 
									
										
										
										
											2024-05-27 05:02:07 -04:00
										 |  |  |  |       return _("E6xy: Quick legato (x: time (0-7 up; 8-F down); y: semitones)"); | 
					
						
							| 
									
										
										
										
											2024-03-15 14:45:57 -04:00
										 |  |  |  |     case 0xe7: | 
					
						
							| 
									
										
										
										
											2024-05-27 05:02:07 -04:00
										 |  |  |  |       return _("E7xx: Macro release"); | 
					
						
							| 
									
										
										
										
											2024-03-16 20:41:08 -04:00
										 |  |  |  |     case 0xe8: | 
					
						
							| 
									
										
										
										
											2024-05-27 05:02:07 -04:00
										 |  |  |  |       return _("E8xy: Quick legato up (x: time; y: semitones)"); | 
					
						
							| 
									
										
										
										
											2024-03-16 20:41:08 -04:00
										 |  |  |  |     case 0xe9: | 
					
						
							| 
									
										
										
										
											2024-05-27 05:02:07 -04:00
										 |  |  |  |       return _("E9xy: Quick legato down (x: time; y: semitones)"); | 
					
						
							| 
									
										
										
										
											2022-01-20 13:48:20 -05:00
										 |  |  |  |     case 0xea: | 
					
						
							| 
									
										
										
										
											2024-05-27 05:02:07 -04:00
										 |  |  |  |       return _("EAxx: Legato"); | 
					
						
							| 
									
										
										
										
											2022-01-20 13:48:20 -05:00
										 |  |  |  |     case 0xeb: | 
					
						
							| 
									
										
										
										
											2024-05-27 05:02:07 -04:00
										 |  |  |  |       return _("EBxx: Set LEGACY sample mode bank"); | 
					
						
							| 
									
										
										
										
											2022-01-20 13:48:20 -05:00
										 |  |  |  |     case 0xec: | 
					
						
							| 
									
										
										
										
											2024-05-27 05:02:07 -04:00
										 |  |  |  |       return _("ECxx: Note cut"); | 
					
						
							| 
									
										
										
										
											2022-01-20 13:48:20 -05:00
										 |  |  |  |     case 0xed: | 
					
						
							| 
									
										
										
										
											2024-05-27 05:02:07 -04:00
										 |  |  |  |       return _("EDxx: Note delay"); | 
					
						
							| 
									
										
										
										
											2022-01-20 13:48:20 -05:00
										 |  |  |  |     case 0xee: | 
					
						
							| 
									
										
										
										
											2024-05-27 05:02:07 -04:00
										 |  |  |  |       return _("EExx: Send external command"); | 
					
						
							| 
									
										
										
										
											2022-03-14 16:59:42 -04:00
										 |  |  |  |     case 0xf0: | 
					
						
							| 
									
										
										
										
											2024-05-27 05:02:07 -04:00
										 |  |  |  |       return _("F0xx: Set tick rate (bpm)"); | 
					
						
							| 
									
										
										
										
											2022-03-14 10:50:52 -04:00
										 |  |  |  |     case 0xf1: | 
					
						
							| 
									
										
										
										
											2024-08-25 17:13:17 -04:00
										 |  |  |  |       return _("F1xx: Single tick pitch up"); | 
					
						
							| 
									
										
										
										
											2022-03-14 10:50:52 -04:00
										 |  |  |  |     case 0xf2: | 
					
						
							| 
									
										
										
										
											2024-08-25 17:13:17 -04:00
										 |  |  |  |       return _("F2xx: Single tick pitch down"); | 
					
						
							| 
									
										
										
										
											2022-03-26 21:58:33 -04:00
										 |  |  |  |     case 0xf3: | 
					
						
							| 
									
										
										
										
											2024-05-27 05:02:07 -04:00
										 |  |  |  |       return _("F3xx: Fine volume slide up"); | 
					
						
							| 
									
										
										
										
											2022-03-26 21:58:33 -04:00
										 |  |  |  |     case 0xf4: | 
					
						
							| 
									
										
										
										
											2024-05-27 05:02:07 -04:00
										 |  |  |  |       return _("F4xx: Fine volume slide down"); | 
					
						
							| 
									
										
										
										
											2022-12-17 00:09:56 -05:00
										 |  |  |  |     case 0xf5: | 
					
						
							| 
									
										
										
										
											2024-05-27 05:02:07 -04:00
										 |  |  |  |       return _("F5xx: Disable macro (see manual)"); | 
					
						
							| 
									
										
										
										
											2022-12-17 00:09:56 -05:00
										 |  |  |  |     case 0xf6: | 
					
						
							| 
									
										
										
										
											2024-05-27 05:02:07 -04:00
										 |  |  |  |       return _("F6xx: Enable macro (see manual)"); | 
					
						
							| 
									
										
										
										
											2024-01-17 07:28:29 -05:00
										 |  |  |  |     case 0xf7: | 
					
						
							| 
									
										
										
										
											2024-05-27 05:02:07 -04:00
										 |  |  |  |       return _("F7xx: Restart macro (see manual)"); | 
					
						
							| 
									
										
										
										
											2022-03-14 10:50:52 -04:00
										 |  |  |  |     case 0xf8: | 
					
						
							| 
									
										
										
										
											2024-08-25 17:13:17 -04:00
										 |  |  |  |       return _("F8xx: Single tick volume up"); | 
					
						
							| 
									
										
										
										
											2022-03-14 10:50:52 -04:00
										 |  |  |  |     case 0xf9: | 
					
						
							| 
									
										
										
										
											2024-08-25 17:13:17 -04:00
										 |  |  |  |       return _("F9xx: Single tick volume down"); | 
					
						
							| 
									
										
										
										
											2022-03-14 10:50:52 -04:00
										 |  |  |  |     case 0xfa: | 
					
						
							| 
									
										
										
										
											2024-05-27 05:02:07 -04:00
										 |  |  |  |       return _("FAxx: Fast volume slide (0y: down; x0: up)"); | 
					
						
							| 
									
										
										
										
											2024-03-15 14:45:57 -04:00
										 |  |  |  |     case 0xfc: | 
					
						
							| 
									
										
										
										
											2024-05-27 05:02:07 -04:00
										 |  |  |  |       return _("FCxx: Note release"); | 
					
						
							| 
									
										
										
										
											2024-03-15 15:56:55 -04:00
										 |  |  |  |     case 0xfd: | 
					
						
							| 
									
										
										
										
											2024-05-27 05:02:07 -04:00
										 |  |  |  |       return _("FDxx: Set virtual tempo numerator"); | 
					
						
							| 
									
										
										
										
											2024-03-15 15:56:55 -04:00
										 |  |  |  |     case 0xfe: | 
					
						
							| 
									
										
										
										
											2024-05-27 05:02:07 -04:00
										 |  |  |  |       return _("FExx: Set virtual tempo denominator"); | 
					
						
							| 
									
										
										
										
											2022-02-04 14:43:57 -05:00
										 |  |  |  |     case 0xff: | 
					
						
							| 
									
										
										
										
											2024-05-27 05:02:07 -04:00
										 |  |  |  |       return _("FFxx: Stop song"); | 
					
						
							| 
									
										
										
										
											2022-01-20 13:48:20 -05:00
										 |  |  |  |     default: | 
					
						
							| 
									
										
										
										
											2022-03-14 10:50:52 -04:00
										 |  |  |  |       if ((effect&0xf0)==0x90) { | 
					
						
							| 
									
										
										
										
											2024-04-23 15:36:06 -04:00
										 |  |  |  |         if (song.oldSampleOffset) { | 
					
						
							| 
									
										
										
										
											2024-05-27 05:02:07 -04:00
										 |  |  |  |           return _("9xxx: Set sample offset*256"); | 
					
						
							| 
									
										
										
										
											2024-04-23 15:36:06 -04:00
										 |  |  |  |         } | 
					
						
							|  |  |  |  |         switch (effect) { | 
					
						
							|  |  |  |  |           case 0x90: | 
					
						
							| 
									
										
										
										
											2024-05-27 05:02:07 -04:00
										 |  |  |  |             return _("90xx: Set sample offset (first byte)"); | 
					
						
							| 
									
										
										
										
											2024-04-23 15:36:06 -04:00
										 |  |  |  |           case 0x91: | 
					
						
							| 
									
										
										
										
											2024-05-27 05:02:07 -04:00
										 |  |  |  |             return _("91xx: Set sample offset (second byte, ×256)"); | 
					
						
							| 
									
										
										
										
											2024-04-23 15:36:06 -04:00
										 |  |  |  |           case 0x92: | 
					
						
							| 
									
										
										
										
											2024-05-27 05:02:07 -04:00
										 |  |  |  |             return _("92xx: Set sample offset (third byte, ×65536)"); | 
					
						
							| 
									
										
										
										
											2024-04-23 15:36:06 -04:00
										 |  |  |  |         } | 
					
						
							| 
									
										
										
										
											2022-04-19 19:44:05 -04:00
										 |  |  |  |       } else if (chan>=0 && chan<chans) { | 
					
						
							| 
									
										
										
										
											2022-08-18 02:26:22 -04:00
										 |  |  |  |         DivSysDef* sysDef=sysDefs[sysOfChan[chan]]; | 
					
						
							|  |  |  |  |         auto iter=sysDef->effectHandlers.find(effect); | 
					
						
							|  |  |  |  |         if (iter!=sysDef->effectHandlers.end()) { | 
					
						
							|  |  |  |  |           return iter->second.description; | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  |         iter=sysDef->postEffectHandlers.find(effect); | 
					
						
							|  |  |  |  |         if (iter!=sysDef->postEffectHandlers.end()) { | 
					
						
							|  |  |  |  |           return iter->second.description; | 
					
						
							|  |  |  |  |         } | 
					
						
							| 
									
										
										
										
											2023-10-06 01:34:51 -04:00
										 |  |  |  |         iter=sysDef->preEffectHandlers.find(effect); | 
					
						
							|  |  |  |  |         if (iter!=sysDef->preEffectHandlers.end()) { | 
					
						
							|  |  |  |  |           return iter->second.description; | 
					
						
							|  |  |  |  |         } | 
					
						
							| 
									
										
										
										
											2022-01-20 13:48:20 -05:00
										 |  |  |  |       } | 
					
						
							|  |  |  |  |       break; | 
					
						
							|  |  |  |  |   } | 
					
						
							| 
									
										
										
										
											2024-05-27 05:02:07 -04:00
										 |  |  |  |   return notNull?_("Invalid effect"):NULL; | 
					
						
							| 
									
										
										
										
											2022-01-20 13:48:20 -05:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-01-27 01:04:26 -05:00
										 |  |  |  | void DivEngine::walkSong(int& loopOrder, int& loopRow, int& loopEnd) { | 
					
						
							| 
									
										
										
										
											2024-06-26 05:03:49 -04:00
										 |  |  |  |   if (curSubSong!=NULL) { | 
					
						
							|  |  |  |  |     curSubSong->walk(loopOrder,loopRow,loopEnd,chans,song.jumpTreatment,song.ignoreJumpAtEnd); | 
					
						
							| 
									
										
										
										
											2022-01-25 15:06:29 -05:00
										 |  |  |  |   } | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-08-24 18:41:41 -04:00
										 |  |  |  | void DivEngine::findSongLength(int loopOrder, int loopRow, double fadeoutLen, int& rowsForFadeout, bool& hasFFxx, std::vector<int>& orders, int& length) { | 
					
						
							|  |  |  |  |   if (curSubSong!=NULL) { | 
					
						
							|  |  |  |  |     curSubSong->findLength(loopOrder,loopRow,fadeoutLen,rowsForFadeout,hasFFxx,orders,song.grooves,length,chans,song.jumpTreatment,song.ignoreJumpAtEnd); | 
					
						
							| 
									
										
										
										
											2024-08-20 15:02:07 -04:00
										 |  |  |  |   } | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-07-19 18:01:19 -04:00
										 |  |  |  | #define EXPORT_BUFSIZE 2048
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | double DivEngine::benchmarkPlayback() { | 
					
						
							|  |  |  |  |   float* outBuf[2]; | 
					
						
							|  |  |  |  |   outBuf[0]=new float[EXPORT_BUFSIZE]; | 
					
						
							|  |  |  |  |   outBuf[1]=new float[EXPORT_BUFSIZE]; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   curOrder=0; | 
					
						
							|  |  |  |  |   prevOrder=0; | 
					
						
							|  |  |  |  |   remainingLoops=1; | 
					
						
							|  |  |  |  |   playSub(false); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   std::chrono::high_resolution_clock::time_point timeStart=std::chrono::high_resolution_clock::now(); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   // benchmark
 | 
					
						
							|  |  |  |  |   while (playing) { | 
					
						
							|  |  |  |  |     nextBuf(NULL,outBuf,0,2,EXPORT_BUFSIZE); | 
					
						
							|  |  |  |  |   } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   std::chrono::high_resolution_clock::time_point timeEnd=std::chrono::high_resolution_clock::now(); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   delete[] outBuf[0]; | 
					
						
							|  |  |  |  |   delete[] outBuf[1]; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   double t=(double)(std::chrono::duration_cast<std::chrono::microseconds>(timeEnd-timeStart).count())/1000000.0; | 
					
						
							|  |  |  |  |   printf("[RESULT] %fs\n",t); | 
					
						
							|  |  |  |  |   return t; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | double DivEngine::benchmarkSeek() { | 
					
						
							|  |  |  |  |   double t[20]; | 
					
						
							|  |  |  |  |   curOrder=curSubSong->ordersLen-1; | 
					
						
							|  |  |  |  |   prevOrder=curSubSong->ordersLen-1; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   // benchmark
 | 
					
						
							|  |  |  |  |   for (int i=0; i<20; i++) { | 
					
						
							|  |  |  |  |     std::chrono::high_resolution_clock::time_point timeStart=std::chrono::high_resolution_clock::now(); | 
					
						
							|  |  |  |  |     playSub(false);      | 
					
						
							|  |  |  |  |     std::chrono::high_resolution_clock::time_point timeEnd=std::chrono::high_resolution_clock::now(); | 
					
						
							|  |  |  |  |     t[i]=(double)(std::chrono::duration_cast<std::chrono::microseconds>(timeEnd-timeStart).count())/1000000.0; | 
					
						
							|  |  |  |  |     printf("[#%d] %fs\n",i+1,t[i]); | 
					
						
							|  |  |  |  |   } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   double tMin=DBL_MAX; | 
					
						
							|  |  |  |  |   double tMax=0.0; | 
					
						
							|  |  |  |  |   double tAvg=0.0; | 
					
						
							|  |  |  |  |   for (int i=0; i<20; i++) { | 
					
						
							|  |  |  |  |     if (t[i]<tMin) tMin=t[i]; | 
					
						
							|  |  |  |  |     if (t[i]>tMax) tMax=t[i]; | 
					
						
							|  |  |  |  |     tAvg+=t[i]; | 
					
						
							|  |  |  |  |   } | 
					
						
							|  |  |  |  |   tAvg/=20.0; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   printf("[RESULT] min %fs max %fs average %fs\n",tMin,tMax,tAvg); | 
					
						
							|  |  |  |  |   return tAvg; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-01-17 23:59:52 -05:00
										 |  |  |  | void DivEngine::notifyInsChange(int ins) { | 
					
						
							| 
									
										
										
										
											2022-03-23 22:38:28 -04:00
										 |  |  |  |   BUSY_BEGIN; | 
					
						
							| 
									
										
										
										
											2022-01-17 23:59:52 -05:00
										 |  |  |  |   for (int i=0; i<song.systemLen; i++) { | 
					
						
							|  |  |  |  |     disCont[i].dispatch->notifyInsChange(ins); | 
					
						
							|  |  |  |  |   } | 
					
						
							| 
									
										
										
										
											2022-03-23 22:38:28 -04:00
										 |  |  |  |   BUSY_END; | 
					
						
							| 
									
										
										
										
											2022-01-17 23:59:52 -05:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-01-18 00:25:10 -05:00
										 |  |  |  | void DivEngine::notifyWaveChange(int wave) { | 
					
						
							| 
									
										
										
										
											2022-03-23 22:38:28 -04:00
										 |  |  |  |   BUSY_BEGIN; | 
					
						
							| 
									
										
										
										
											2022-01-18 00:25:10 -05:00
										 |  |  |  |   for (int i=0; i<song.systemLen; i++) { | 
					
						
							|  |  |  |  |     disCont[i].dispatch->notifyWaveChange(wave); | 
					
						
							|  |  |  |  |   } | 
					
						
							| 
									
										
										
										
											2022-03-23 22:38:28 -04:00
										 |  |  |  |   BUSY_END; | 
					
						
							| 
									
										
										
										
											2022-01-18 00:25:10 -05:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-05-12 23:15:03 -04:00
										 |  |  |  | int DivEngine::loadSampleROM(String path, ssize_t expectedSize, unsigned char*& ret) { | 
					
						
							| 
									
										
										
										
											2023-08-24 16:33:53 -04:00
										 |  |  |  |   ret=NULL; | 
					
						
							| 
									
										
										
										
											2022-05-11 17:39:36 -04:00
										 |  |  |  |   if (path.empty()) { | 
					
						
							|  |  |  |  |     return 0; | 
					
						
							|  |  |  |  |   } | 
					
						
							|  |  |  |  |   logI("loading ROM %s...",path); | 
					
						
							|  |  |  |  |   FILE* f=ps_fopen(path.c_str(),"rb"); | 
					
						
							|  |  |  |  |   if (f==NULL) { | 
					
						
							|  |  |  |  |     logE("error: %s",strerror(errno)); | 
					
						
							|  |  |  |  |     lastError=strerror(errno); | 
					
						
							|  |  |  |  |     return -1; | 
					
						
							|  |  |  |  |   } | 
					
						
							|  |  |  |  |   if (fseek(f,0,SEEK_END)<0) { | 
					
						
							|  |  |  |  |     logE("size error: %s",strerror(errno)); | 
					
						
							| 
									
										
										
										
											2024-05-27 05:02:07 -04:00
										 |  |  |  |     lastError=fmt::sprintf(_("on seek: %s"),strerror(errno)); | 
					
						
							| 
									
										
										
										
											2022-05-11 17:39:36 -04:00
										 |  |  |  |     fclose(f); | 
					
						
							|  |  |  |  |     return -1; | 
					
						
							|  |  |  |  |   } | 
					
						
							|  |  |  |  |   ssize_t len=ftell(f); | 
					
						
							|  |  |  |  |   if (len==(SIZE_MAX>>1)) { | 
					
						
							|  |  |  |  |     logE("could not get file length: %s",strerror(errno)); | 
					
						
							| 
									
										
										
										
											2024-05-27 05:02:07 -04:00
										 |  |  |  |     lastError=fmt::sprintf(_("on pre tell: %s"),strerror(errno)); | 
					
						
							| 
									
										
										
										
											2022-05-11 17:39:36 -04:00
										 |  |  |  |     fclose(f); | 
					
						
							|  |  |  |  |     return -1; | 
					
						
							|  |  |  |  |   } | 
					
						
							|  |  |  |  |   if (len<1) { | 
					
						
							|  |  |  |  |     if (len==0) { | 
					
						
							|  |  |  |  |       logE("that file is empty!"); | 
					
						
							| 
									
										
										
										
											2024-05-27 05:02:07 -04:00
										 |  |  |  |       lastError=_("file is empty"); | 
					
						
							| 
									
										
										
										
											2022-05-11 17:39:36 -04:00
										 |  |  |  |     } else { | 
					
						
							|  |  |  |  |       logE("tell error: %s",strerror(errno)); | 
					
						
							| 
									
										
										
										
											2024-05-27 05:02:07 -04:00
										 |  |  |  |       lastError=fmt::sprintf(_("on tell: %s"),strerror(errno)); | 
					
						
							| 
									
										
										
										
											2022-05-11 17:39:36 -04:00
										 |  |  |  |     } | 
					
						
							|  |  |  |  |     fclose(f); | 
					
						
							|  |  |  |  |     return -1; | 
					
						
							|  |  |  |  |   } | 
					
						
							|  |  |  |  |   if (len!=expectedSize) { | 
					
						
							|  |  |  |  |     logE("ROM size mismatch, expected: %d bytes, was: %d bytes", expectedSize, len); | 
					
						
							| 
									
										
										
										
											2024-05-27 05:02:07 -04:00
										 |  |  |  |     lastError=fmt::sprintf(_("ROM size mismatch, expected: %d bytes, was: %d"), expectedSize, len); | 
					
						
							| 
									
										
										
										
											2022-05-11 17:39:36 -04:00
										 |  |  |  |     return -1; | 
					
						
							|  |  |  |  |   } | 
					
						
							|  |  |  |  |   if (fseek(f,0,SEEK_SET)<0) { | 
					
						
							|  |  |  |  |     logE("size error: %s",strerror(errno)); | 
					
						
							| 
									
										
										
										
											2024-05-27 05:02:07 -04:00
										 |  |  |  |     lastError=fmt::sprintf(_("on get size: %s"),strerror(errno)); | 
					
						
							| 
									
										
										
										
											2022-05-11 17:39:36 -04:00
										 |  |  |  |     fclose(f); | 
					
						
							|  |  |  |  |     return -1; | 
					
						
							|  |  |  |  |   } | 
					
						
							|  |  |  |  |   unsigned char* file=new unsigned char[len]; | 
					
						
							|  |  |  |  |   if (fread(file,1,(size_t)len,f)!=(size_t)len) { | 
					
						
							|  |  |  |  |     logE("read error: %s",strerror(errno)); | 
					
						
							| 
									
										
										
										
											2024-05-27 05:02:07 -04:00
										 |  |  |  |     lastError=fmt::sprintf(_("on read: %s"),strerror(errno)); | 
					
						
							| 
									
										
										
										
											2022-05-11 17:39:36 -04:00
										 |  |  |  |     fclose(f); | 
					
						
							|  |  |  |  |     delete[] file; | 
					
						
							|  |  |  |  |     return -1; | 
					
						
							|  |  |  |  |   } | 
					
						
							|  |  |  |  |   fclose(f); | 
					
						
							| 
									
										
										
										
											2023-08-24 16:33:53 -04:00
										 |  |  |  |   ret=file; | 
					
						
							| 
									
										
										
										
											2022-05-11 17:39:36 -04:00
										 |  |  |  |   return 0; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-10-02 02:32:12 -04:00
										 |  |  |  | unsigned int DivEngine::getSampleFormatMask() { | 
					
						
							|  |  |  |  |   unsigned int formatMask=1U<<16; // 16-bit is always on
 | 
					
						
							|  |  |  |  |   for (int i=0; i<song.systemLen; i++) { | 
					
						
							|  |  |  |  |     const DivSysDef* s=getSystemDef(song.system[i]); | 
					
						
							|  |  |  |  |     if (s==NULL) continue; | 
					
						
							|  |  |  |  |     formatMask|=s->sampleFormatMask; | 
					
						
							|  |  |  |  |   } | 
					
						
							|  |  |  |  |   return formatMask; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-05-12 23:15:03 -04:00
										 |  |  |  | int DivEngine::loadSampleROMs() { | 
					
						
							|  |  |  |  |   if (yrw801ROM!=NULL) { | 
					
						
							|  |  |  |  |     delete[] yrw801ROM; | 
					
						
							|  |  |  |  |     yrw801ROM=NULL; | 
					
						
							|  |  |  |  |   } | 
					
						
							|  |  |  |  |   if (tg100ROM!=NULL) { | 
					
						
							|  |  |  |  |     delete[] tg100ROM; | 
					
						
							|  |  |  |  |     tg100ROM=NULL; | 
					
						
							|  |  |  |  |   } | 
					
						
							|  |  |  |  |   if (mu5ROM!=NULL) { | 
					
						
							|  |  |  |  |     delete[] mu5ROM; | 
					
						
							|  |  |  |  |     mu5ROM=NULL; | 
					
						
							|  |  |  |  |   } | 
					
						
							| 
									
										
										
										
											2023-08-24 16:33:53 -04:00
										 |  |  |  |   int error=0; | 
					
						
							|  |  |  |  |   error+=loadSampleROM(getConfString("yrw801Path",""), 0x200000, yrw801ROM); | 
					
						
							|  |  |  |  |   error+=loadSampleROM(getConfString("tg100Path",""), 0x200000, tg100ROM); | 
					
						
							|  |  |  |  |   error+=loadSampleROM(getConfString("mu5Path",""), 0x200000, mu5ROM); | 
					
						
							| 
									
										
										
										
											2022-05-11 17:39:36 -04:00
										 |  |  |  |   return error; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-09-08 02:19:48 -04:00
										 |  |  |  | void DivEngine::renderSamplesP(int whichSample) { | 
					
						
							| 
									
										
										
										
											2022-03-23 22:38:28 -04:00
										 |  |  |  |   BUSY_BEGIN; | 
					
						
							| 
									
										
										
										
											2023-09-08 02:19:48 -04:00
										 |  |  |  |   renderSamples(whichSample); | 
					
						
							| 
									
										
										
										
											2022-03-23 22:38:28 -04:00
										 |  |  |  |   BUSY_END; | 
					
						
							| 
									
										
										
										
											2021-12-17 03:33:12 -05:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-09-08 02:19:48 -04:00
										 |  |  |  | void DivEngine::renderSamples(int whichSample) { | 
					
						
							| 
									
										
										
										
											2021-12-21 16:02:31 -05:00
										 |  |  |  |   sPreview.sample=-1; | 
					
						
							|  |  |  |  |   sPreview.pos=0; | 
					
						
							| 
									
										
										
										
											2022-08-11 09:21:54 -04:00
										 |  |  |  |   sPreview.dir=false; | 
					
						
							| 
									
										
										
										
											2021-12-10 04:22:13 -05:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-10-02 02:32:12 -04:00
										 |  |  |  |   logD("rendering samples..."); | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-25 18:18:04 -04:00
										 |  |  |  |   // step 0: make sample format mask
 | 
					
						
							|  |  |  |  |   unsigned int formatMask=1U<<16; // 16-bit is always on
 | 
					
						
							|  |  |  |  |   for (int i=0; i<song.systemLen; i++) { | 
					
						
							|  |  |  |  |     const DivSysDef* s=getSystemDef(song.system[i]); | 
					
						
							|  |  |  |  |     if (s==NULL) continue; | 
					
						
							|  |  |  |  |     formatMask|=s->sampleFormatMask; | 
					
						
							|  |  |  |  |   } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-02-24 16:16:02 -05:00
										 |  |  |  |   // step 1: render samples
 | 
					
						
							| 
									
										
										
										
											2023-09-08 02:19:48 -04:00
										 |  |  |  |   if (whichSample==-1) { | 
					
						
							|  |  |  |  |     for (int i=0; i<song.sampleLen; i++) { | 
					
						
							|  |  |  |  |       song.sample[i]->render(formatMask); | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  |   } else if (whichSample>=0 && whichSample<song.sampleLen) { | 
					
						
							|  |  |  |  |     song.sample[whichSample]->render(formatMask); | 
					
						
							| 
									
										
										
										
											2021-05-13 03:39:26 -04:00
										 |  |  |  |   } | 
					
						
							| 
									
										
										
										
											2021-12-10 23:41:00 -05:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-05-01 13:57:44 -04:00
										 |  |  |  |   // step 2: render samples to dispatch
 | 
					
						
							|  |  |  |  |   for (int i=0; i<song.systemLen; i++) { | 
					
						
							|  |  |  |  |     if (disCont[i].dispatch!=NULL) { | 
					
						
							| 
									
										
										
										
											2022-11-26 23:50:20 -05:00
										 |  |  |  |       disCont[i].dispatch->renderSamples(i); | 
					
						
							| 
									
										
										
										
											2022-03-06 12:31:03 -05:00
										 |  |  |  |     } | 
					
						
							|  |  |  |  |   } | 
					
						
							| 
									
										
										
										
											2021-05-13 03:39:26 -04:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-30 01:26:54 -04:00
										 |  |  |  | String DivEngine::decodeSysDesc(String desc) { | 
					
						
							|  |  |  |  |   DivConfig newDesc; | 
					
						
							| 
									
										
										
										
											2022-04-30 02:37:37 -04:00
										 |  |  |  |   bool hasVal=false; | 
					
						
							|  |  |  |  |   bool negative=false; | 
					
						
							|  |  |  |  |   int val=0; | 
					
						
							|  |  |  |  |   int curStage=0; | 
					
						
							| 
									
										
										
										
											2022-04-30 05:30:33 -04:00
										 |  |  |  |   int sysID=0; | 
					
						
							| 
									
										
										
										
											2023-01-06 17:43:08 -05:00
										 |  |  |  |   float sysVol=0; | 
					
						
							|  |  |  |  |   float sysPan=0; | 
					
						
							| 
									
										
										
										
											2022-04-30 05:30:33 -04:00
										 |  |  |  |   int sysFlags=0; | 
					
						
							| 
									
										
										
										
											2022-09-30 01:26:54 -04:00
										 |  |  |  |   int curSys=0; | 
					
						
							| 
									
										
										
										
											2022-04-30 02:37:37 -04:00
										 |  |  |  |   desc+=' '; // ha
 | 
					
						
							|  |  |  |  |   for (char i: desc) { | 
					
						
							|  |  |  |  |     switch (i) { | 
					
						
							|  |  |  |  |       case ' ': | 
					
						
							|  |  |  |  |         if (hasVal) { | 
					
						
							|  |  |  |  |           if (negative) val=-val; | 
					
						
							|  |  |  |  |           switch (curStage) { | 
					
						
							|  |  |  |  |             case 0: | 
					
						
							|  |  |  |  |               sysID=val; | 
					
						
							|  |  |  |  |               curStage++; | 
					
						
							|  |  |  |  |               break; | 
					
						
							|  |  |  |  |             case 1: | 
					
						
							| 
									
										
										
										
											2023-01-06 17:43:08 -05:00
										 |  |  |  |               sysVol=(float)val/64.0f; | 
					
						
							| 
									
										
										
										
											2022-04-30 02:37:37 -04:00
										 |  |  |  |               curStage++; | 
					
						
							|  |  |  |  |               break; | 
					
						
							|  |  |  |  |             case 2: | 
					
						
							| 
									
										
										
										
											2023-01-06 17:43:08 -05:00
										 |  |  |  |               sysPan=(float)val/127.0f; | 
					
						
							| 
									
										
										
										
											2022-04-30 02:37:37 -04:00
										 |  |  |  |               curStage++; | 
					
						
							|  |  |  |  |               break; | 
					
						
							|  |  |  |  |             case 3: | 
					
						
							|  |  |  |  |               sysFlags=val; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-30 01:26:54 -04:00
										 |  |  |  |               if (sysID!=0) { | 
					
						
							| 
									
										
										
										
											2023-01-06 17:43:08 -05:00
										 |  |  |  |                 if (sysVol<-1.0f) sysVol=-1.0f; | 
					
						
							|  |  |  |  |                 if (sysVol>1.0f) sysVol=1.0f; | 
					
						
							|  |  |  |  |                 if (sysPan<-1.0f) sysPan=-1.0f; | 
					
						
							|  |  |  |  |                 if (sysPan>1.0f) sysPan=1.0f; | 
					
						
							| 
									
										
										
										
											2022-09-30 01:26:54 -04:00
										 |  |  |  |                 newDesc.set(fmt::sprintf("id%d",curSys),sysID); | 
					
						
							|  |  |  |  |                 newDesc.set(fmt::sprintf("vol%d",curSys),sysVol); | 
					
						
							|  |  |  |  |                 newDesc.set(fmt::sprintf("pan%d",curSys),sysPan); | 
					
						
							| 
									
										
										
										
											2023-01-06 17:43:08 -05:00
										 |  |  |  |                 newDesc.set(fmt::sprintf("fr%d",curSys),0.0f); | 
					
						
							| 
									
										
										
										
											2022-09-30 01:26:54 -04:00
										 |  |  |  |                 DivConfig newFlagsC; | 
					
						
							|  |  |  |  |                 newFlagsC.clear(); | 
					
						
							|  |  |  |  |                 convertOldFlags((unsigned int)sysFlags,newFlagsC,systemFromFileFur(sysID)); | 
					
						
							|  |  |  |  |                 newDesc.set(fmt::sprintf("flags%d",curSys),newFlagsC.toBase64()); | 
					
						
							|  |  |  |  |                 curSys++; | 
					
						
							| 
									
										
										
										
											2022-04-30 02:37:37 -04:00
										 |  |  |  |               } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |               curStage=0; | 
					
						
							|  |  |  |  |               break; | 
					
						
							|  |  |  |  |           } | 
					
						
							|  |  |  |  |           hasVal=false; | 
					
						
							|  |  |  |  |           negative=false; | 
					
						
							|  |  |  |  |           val=0; | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  |         break; | 
					
						
							|  |  |  |  |       case '0': case '1': case '2': case '3': case '4': | 
					
						
							|  |  |  |  |       case '5': case '6': case '7': case '8': case '9': | 
					
						
							|  |  |  |  |         val=(val*10)+(i-'0'); | 
					
						
							|  |  |  |  |         hasVal=true; | 
					
						
							|  |  |  |  |         break; | 
					
						
							|  |  |  |  |       case '-': | 
					
						
							|  |  |  |  |         if (!hasVal) negative=true; | 
					
						
							|  |  |  |  |         break; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  |   } | 
					
						
							| 
									
										
										
										
											2022-09-30 01:26:54 -04:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |   return newDesc.toBase64(); | 
					
						
							| 
									
										
										
										
											2022-04-30 02:37:37 -04:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-01-06 17:43:08 -05:00
										 |  |  |  | void DivEngine::initSongWithDesc(const char* description, bool inBase64, bool oldVol) { | 
					
						
							| 
									
										
										
										
											2022-04-30 02:37:37 -04:00
										 |  |  |  |   int chanCount=0; | 
					
						
							| 
									
										
										
										
											2022-09-30 01:26:54 -04:00
										 |  |  |  |   DivConfig c; | 
					
						
							| 
									
										
										
										
											2022-11-13 16:57:47 -05:00
										 |  |  |  |   if (inBase64) { | 
					
						
							|  |  |  |  |     c.loadFromBase64(description); | 
					
						
							|  |  |  |  |   } else { | 
					
						
							|  |  |  |  |     c.loadFromMemory(description); | 
					
						
							|  |  |  |  |   } | 
					
						
							| 
									
										
										
										
											2022-09-30 01:26:54 -04:00
										 |  |  |  |   int index=0; | 
					
						
							| 
									
										
										
										
											2022-12-09 00:30:27 -05:00
										 |  |  |  |   for (; index<DIV_MAX_CHIPS; index++) { | 
					
						
							| 
									
										
										
										
											2022-09-30 01:26:54 -04:00
										 |  |  |  |     song.system[index]=systemFromFileFur(c.getInt(fmt::sprintf("id%d",index),0)); | 
					
						
							|  |  |  |  |     if (song.system[index]==DIV_SYSTEM_NULL) { | 
					
						
							|  |  |  |  |       break; | 
					
						
							| 
									
										
										
										
											2022-04-30 02:37:37 -04:00
										 |  |  |  |     } | 
					
						
							| 
									
										
										
										
											2022-09-30 01:26:54 -04:00
										 |  |  |  |     chanCount+=getChannelCount(song.system[index]); | 
					
						
							|  |  |  |  |     if (chanCount>=DIV_MAX_CHANS) { | 
					
						
							|  |  |  |  |       song.system[index]=DIV_SYSTEM_NULL; | 
					
						
							|  |  |  |  |       break; | 
					
						
							|  |  |  |  |     } | 
					
						
							| 
									
										
										
										
											2023-01-06 17:43:08 -05:00
										 |  |  |  |     song.systemVol[index]=c.getFloat(fmt::sprintf("vol%d",index),1.0f); | 
					
						
							|  |  |  |  |     song.systemPan[index]=c.getFloat(fmt::sprintf("pan%d",index),0.0f); | 
					
						
							|  |  |  |  |     song.systemPanFR[index]=c.getFloat(fmt::sprintf("fr%d",index),0.0f); | 
					
						
							| 
									
										
										
										
											2022-09-30 01:26:54 -04:00
										 |  |  |  |     song.systemFlags[index].clear(); | 
					
						
							| 
									
										
										
										
											2023-01-06 17:43:08 -05:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |     if (oldVol) { | 
					
						
							|  |  |  |  |       song.systemVol[index]/=64.0f; | 
					
						
							|  |  |  |  |       song.systemPan[index]/=127.0f; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-30 01:26:54 -04:00
										 |  |  |  |     String flags=c.getString(fmt::sprintf("flags%d",index),""); | 
					
						
							|  |  |  |  |     song.systemFlags[index].loadFromBase64(flags.c_str()); | 
					
						
							| 
									
										
										
										
											2022-04-30 02:37:37 -04:00
										 |  |  |  |   } | 
					
						
							| 
									
										
										
										
											2022-09-30 01:26:54 -04:00
										 |  |  |  |   song.systemLen=index; | 
					
						
							| 
									
										
										
										
											2022-12-04 01:11:32 -05:00
										 |  |  |  |    | 
					
						
							|  |  |  |  |   // extra attributes
 | 
					
						
							|  |  |  |  |   song.subsong[0]->hz=c.getDouble("tickRate",60.0); | 
					
						
							| 
									
										
										
										
											2024-04-23 15:36:06 -04:00
										 |  |  |  |   if (song.subsong[0]->hz<1.0) song.subsong[0]->hz=1.0; | 
					
						
							|  |  |  |  |   if (song.subsong[0]->hz>999.0) song.subsong[0]->hz=999.0; | 
					
						
							| 
									
										
										
										
											2024-04-23 14:36:32 -04:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-02 16:39:35 -04:00
										 |  |  |  |   curChanMask=c.getIntList("chanMask",{}); | 
					
						
							|  |  |  |  |   for (unsigned char i:curChanMask) { | 
					
						
							|  |  |  |  |     int j=i-1; | 
					
						
							|  |  |  |  |     if (j<0) j=0; | 
					
						
							|  |  |  |  |     if (j>DIV_MAX_CHANS) j=DIV_MAX_CHANS-1; | 
					
						
							|  |  |  |  |     curSubSong->chanShow[j]=false; | 
					
						
							|  |  |  |  |     curSubSong->chanShowChanOsc[j]=false; | 
					
						
							|  |  |  |  |   } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-10-11 10:16:46 -04:00
										 |  |  |  |   song.author=getConfString("defaultAuthorName",""); | 
					
						
							| 
									
										
										
										
											2022-04-30 02:37:37 -04:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-13 16:57:47 -05:00
										 |  |  |  | void DivEngine::createNew(const char* description, String sysName, bool inBase64) { | 
					
						
							| 
									
										
										
										
											2021-12-24 18:23:01 -05:00
										 |  |  |  |   quitDispatch(); | 
					
						
							| 
									
										
										
										
											2022-03-23 22:38:28 -04:00
										 |  |  |  |   BUSY_BEGIN; | 
					
						
							| 
									
										
										
										
											2022-03-21 17:17:51 -04:00
										 |  |  |  |   saveLock.lock(); | 
					
						
							| 
									
										
										
										
											2021-12-24 18:23:01 -05:00
										 |  |  |  |   song.unload(); | 
					
						
							|  |  |  |  |   song=DivSong(); | 
					
						
							| 
									
										
										
										
											2022-05-15 02:42:49 -04:00
										 |  |  |  |   changeSong(0); | 
					
						
							| 
									
										
										
										
											2022-03-01 17:19:52 -05:00
										 |  |  |  |   if (description!=NULL) { | 
					
						
							| 
									
										
										
										
											2022-11-13 16:57:47 -05:00
										 |  |  |  |     initSongWithDesc(description,inBase64); | 
					
						
							| 
									
										
										
										
											2022-03-01 17:19:52 -05:00
										 |  |  |  |   } | 
					
						
							| 
									
										
										
										
											2022-07-23 18:02:03 -04:00
										 |  |  |  |   if (sysName=="") { | 
					
						
							| 
									
										
										
										
											2022-07-27 03:36:48 -04:00
										 |  |  |  |     song.systemName=getSongSystemLegacyName(song,!getConfInt("noMultiSystem",0)); | 
					
						
							| 
									
										
										
										
											2022-07-23 18:02:03 -04:00
										 |  |  |  |   } else { | 
					
						
							|  |  |  |  |     song.systemName=sysName; | 
					
						
							|  |  |  |  |   } | 
					
						
							| 
									
										
										
										
											2022-01-08 16:03:32 -05:00
										 |  |  |  |   recalcChans(); | 
					
						
							| 
									
										
										
										
											2022-03-21 17:17:51 -04:00
										 |  |  |  |   saveLock.unlock(); | 
					
						
							| 
									
										
										
										
											2022-03-23 22:38:28 -04:00
										 |  |  |  |   BUSY_END; | 
					
						
							| 
									
										
										
										
											2021-12-24 18:23:01 -05:00
										 |  |  |  |   initDispatch(); | 
					
						
							| 
									
										
										
										
											2022-03-23 22:38:28 -04:00
										 |  |  |  |   BUSY_BEGIN; | 
					
						
							| 
									
										
										
										
											2022-05-01 13:57:44 -04:00
										 |  |  |  |   renderSamples(); | 
					
						
							| 
									
										
										
										
											2021-12-24 18:23:01 -05:00
										 |  |  |  |   reset(); | 
					
						
							| 
									
										
										
										
											2022-03-23 22:38:28 -04:00
										 |  |  |  |   BUSY_END; | 
					
						
							| 
									
										
										
										
											2021-12-24 18:23:01 -05:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-23 05:56:48 -05:00
										 |  |  |  | void DivEngine::createNewFromDefaults() { | 
					
						
							|  |  |  |  |   quitDispatch(); | 
					
						
							|  |  |  |  |   BUSY_BEGIN; | 
					
						
							|  |  |  |  |   saveLock.lock(); | 
					
						
							|  |  |  |  |   song.unload(); | 
					
						
							|  |  |  |  |   song=DivSong(); | 
					
						
							|  |  |  |  |   changeSong(0); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   String preset=getConfString("initialSys2",""); | 
					
						
							|  |  |  |  |   bool oldVol=getConfInt("configVersion",DIV_ENGINE_VERSION)<135; | 
					
						
							|  |  |  |  |   if (preset.empty()) { | 
					
						
							|  |  |  |  |     // try loading old preset
 | 
					
						
							| 
									
										
										
										
											2023-02-25 00:43:27 -05:00
										 |  |  |  |     logD("trying to load old preset"); | 
					
						
							| 
									
										
										
										
											2023-02-23 05:56:48 -05:00
										 |  |  |  |     preset=decodeSysDesc(getConfString("initialSys","")); | 
					
						
							|  |  |  |  |     oldVol=false; | 
					
						
							|  |  |  |  |   } | 
					
						
							|  |  |  |  |   logD("preset size %ld",preset.size()); | 
					
						
							|  |  |  |  |   if (preset.size()>0 && (preset.size()&3)==0) { | 
					
						
							|  |  |  |  |     initSongWithDesc(preset.c_str(),true,oldVol); | 
					
						
							|  |  |  |  |   } | 
					
						
							|  |  |  |  |   String sysName=getConfString("initialSysName",""); | 
					
						
							|  |  |  |  |   if (sysName=="") { | 
					
						
							|  |  |  |  |     song.systemName=getSongSystemLegacyName(song,!getConfInt("noMultiSystem",0)); | 
					
						
							|  |  |  |  |   } else { | 
					
						
							|  |  |  |  |     song.systemName=sysName; | 
					
						
							|  |  |  |  |   } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   recalcChans(); | 
					
						
							|  |  |  |  |   saveLock.unlock(); | 
					
						
							|  |  |  |  |   BUSY_END; | 
					
						
							|  |  |  |  |   initDispatch(); | 
					
						
							|  |  |  |  |   BUSY_BEGIN; | 
					
						
							|  |  |  |  |   renderSamples(); | 
					
						
							|  |  |  |  |   reset(); | 
					
						
							|  |  |  |  |   BUSY_END; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-28 04:36:15 -04:00
										 |  |  |  | void DivEngine::swapChannels(int src, int dest) { | 
					
						
							|  |  |  |  |   logV("swapping channel %d with %d",src,dest); | 
					
						
							|  |  |  |  |   if (src==dest) { | 
					
						
							|  |  |  |  |     logV("not swapping channels because it's the same channel!",src,dest); | 
					
						
							|  |  |  |  |     return; | 
					
						
							|  |  |  |  |   } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-12-09 00:30:27 -05:00
										 |  |  |  |   for (int i=0; i<DIV_MAX_PATTERNS; i++) { | 
					
						
							| 
									
										
										
										
											2022-05-15 02:42:49 -04:00
										 |  |  |  |     curOrders->ord[dest][i]^=curOrders->ord[src][i]; | 
					
						
							|  |  |  |  |     curOrders->ord[src][i]^=curOrders->ord[dest][i]; | 
					
						
							|  |  |  |  |     curOrders->ord[dest][i]^=curOrders->ord[src][i]; | 
					
						
							| 
									
										
										
										
											2022-04-28 04:36:15 -04:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-05-15 02:42:49 -04:00
										 |  |  |  |     DivPattern* prev=curPat[src].data[i]; | 
					
						
							|  |  |  |  |     curPat[src].data[i]=curPat[dest].data[i]; | 
					
						
							|  |  |  |  |     curPat[dest].data[i]=prev; | 
					
						
							| 
									
										
										
										
											2022-04-28 04:36:15 -04:00
										 |  |  |  |   } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-05-15 02:42:49 -04:00
										 |  |  |  |   curPat[src].effectCols^=curPat[dest].effectCols; | 
					
						
							|  |  |  |  |   curPat[dest].effectCols^=curPat[src].effectCols; | 
					
						
							|  |  |  |  |   curPat[src].effectCols^=curPat[dest].effectCols; | 
					
						
							| 
									
										
										
										
											2022-04-28 15:03:58 -04:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-05-15 02:42:49 -04:00
										 |  |  |  |   String prevChanName=curSubSong->chanName[src]; | 
					
						
							|  |  |  |  |   String prevChanShortName=curSubSong->chanShortName[src]; | 
					
						
							|  |  |  |  |   bool prevChanShow=curSubSong->chanShow[src]; | 
					
						
							| 
									
										
										
										
											2024-02-24 18:12:02 -05:00
										 |  |  |  |   bool prevChanShowChanOsc=curSubSong->chanShowChanOsc[src]; | 
					
						
							| 
									
										
										
										
											2022-08-31 03:52:35 -04:00
										 |  |  |  |   unsigned char prevChanCollapse=curSubSong->chanCollapse[src]; | 
					
						
							| 
									
										
										
										
											2022-04-28 15:03:58 -04:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-05-15 02:42:49 -04:00
										 |  |  |  |   curSubSong->chanName[src]=curSubSong->chanName[dest]; | 
					
						
							|  |  |  |  |   curSubSong->chanShortName[src]=curSubSong->chanShortName[dest]; | 
					
						
							|  |  |  |  |   curSubSong->chanShow[src]=curSubSong->chanShow[dest]; | 
					
						
							| 
									
										
										
										
											2024-02-24 18:35:26 -05:00
										 |  |  |  |   curSubSong->chanShowChanOsc[src]=curSubSong->chanShowChanOsc[dest]; | 
					
						
							| 
									
										
										
										
											2022-05-15 02:42:49 -04:00
										 |  |  |  |   curSubSong->chanCollapse[src]=curSubSong->chanCollapse[dest]; | 
					
						
							|  |  |  |  |   curSubSong->chanName[dest]=prevChanName; | 
					
						
							|  |  |  |  |   curSubSong->chanShortName[dest]=prevChanShortName; | 
					
						
							|  |  |  |  |   curSubSong->chanShow[dest]=prevChanShow; | 
					
						
							| 
									
										
										
										
											2024-02-24 18:12:02 -05:00
										 |  |  |  |   curSubSong->chanShowChanOsc[dest]=prevChanShowChanOsc; | 
					
						
							| 
									
										
										
										
											2022-05-15 02:42:49 -04:00
										 |  |  |  |   curSubSong->chanCollapse[dest]=prevChanCollapse; | 
					
						
							| 
									
										
										
										
											2022-04-28 04:36:15 -04:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | void DivEngine::stompChannel(int ch) { | 
					
						
							|  |  |  |  |   logV("stomping channel %d",ch); | 
					
						
							| 
									
										
										
										
											2022-12-09 00:30:27 -05:00
										 |  |  |  |   for (int i=0; i<DIV_MAX_PATTERNS; i++) { | 
					
						
							| 
									
										
										
										
											2022-05-15 02:42:49 -04:00
										 |  |  |  |     curOrders->ord[ch][i]=0; | 
					
						
							|  |  |  |  |   } | 
					
						
							|  |  |  |  |   curPat[ch].wipePatterns(); | 
					
						
							|  |  |  |  |   curPat[ch].effectCols=1; | 
					
						
							|  |  |  |  |   curSubSong->chanName[ch]=""; | 
					
						
							|  |  |  |  |   curSubSong->chanShortName[ch]=""; | 
					
						
							|  |  |  |  |   curSubSong->chanShow[ch]=true; | 
					
						
							| 
									
										
										
										
											2024-02-24 18:12:02 -05:00
										 |  |  |  |   curSubSong->chanShowChanOsc[ch]=true; | 
					
						
							| 
									
										
										
										
											2022-05-15 02:42:49 -04:00
										 |  |  |  |   curSubSong->chanCollapse[ch]=false; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | void DivEngine::changeSong(size_t songIndex) { | 
					
						
							|  |  |  |  |   if (songIndex>=song.subsong.size()) return; | 
					
						
							|  |  |  |  |   curSubSong=song.subsong[songIndex]; | 
					
						
							|  |  |  |  |   curPat=song.subsong[songIndex]->pat; | 
					
						
							|  |  |  |  |   curOrders=&song.subsong[songIndex]->orders; | 
					
						
							|  |  |  |  |   curSubSongIndex=songIndex; | 
					
						
							|  |  |  |  |   curOrder=0; | 
					
						
							|  |  |  |  |   curRow=0; | 
					
						
							| 
									
										
										
										
											2022-06-06 02:05:06 -04:00
										 |  |  |  |   prevOrder=0; | 
					
						
							|  |  |  |  |   prevRow=0; | 
					
						
							| 
									
										
										
										
											2022-04-28 04:36:15 -04:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-05-16 04:04:16 -04:00
										 |  |  |  | void DivEngine::moveAsset(std::vector<DivAssetDir>& dir, int before, int after) { | 
					
						
							|  |  |  |  |   if (before<0 || after<0) return; | 
					
						
							|  |  |  |  |   for (DivAssetDir& i: dir) { | 
					
						
							|  |  |  |  |     for (size_t j=0; j<i.entries.size(); j++) { | 
					
						
							|  |  |  |  |       // erase matching entry
 | 
					
						
							|  |  |  |  |       if (i.entries[j]==before) { | 
					
						
							|  |  |  |  |         i.entries[j]=after; | 
					
						
							|  |  |  |  |       } else if (i.entries[j]==after) { | 
					
						
							|  |  |  |  |         i.entries[j]=before; | 
					
						
							|  |  |  |  |       } | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  |   } | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-05-16 00:27:45 -04:00
										 |  |  |  | void DivEngine::removeAsset(std::vector<DivAssetDir>& dir, int entry) { | 
					
						
							|  |  |  |  |   if (entry<0) return; | 
					
						
							|  |  |  |  |   for (DivAssetDir& i: dir) { | 
					
						
							|  |  |  |  |     for (size_t j=0; j<i.entries.size(); j++) { | 
					
						
							|  |  |  |  |       // erase matching entry
 | 
					
						
							|  |  |  |  |       if (i.entries[j]==entry) { | 
					
						
							|  |  |  |  |         i.entries.erase(i.entries.begin()+j); | 
					
						
							|  |  |  |  |         j--; | 
					
						
							|  |  |  |  |       } else if (i.entries[j]>entry) { | 
					
						
							|  |  |  |  |         i.entries[j]--; | 
					
						
							|  |  |  |  |       } | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  |   } | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-03-12 05:10:46 -04:00
										 |  |  |  | void DivEngine::checkAssetDir(std::vector<DivAssetDir>& dir, size_t entries) { | 
					
						
							|  |  |  |  |   bool* inAssetDir=new bool[entries]; | 
					
						
							|  |  |  |  |   memset(inAssetDir,0,entries*sizeof(bool)); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   for (DivAssetDir& i: dir) { | 
					
						
							|  |  |  |  |     for (size_t j=0; j<i.entries.size(); j++) { | 
					
						
							|  |  |  |  |       // erase invalid entry
 | 
					
						
							|  |  |  |  |       if (i.entries[j]<0 || i.entries[j]>=(int)entries) { | 
					
						
							|  |  |  |  |         i.entries.erase(i.entries.begin()+j); | 
					
						
							|  |  |  |  |         j--; | 
					
						
							|  |  |  |  |         continue; | 
					
						
							|  |  |  |  |       } | 
					
						
							| 
									
										
										
										
											2023-05-16 01:04:26 -04:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |       // erase duplicate entry
 | 
					
						
							| 
									
										
										
										
											2023-05-16 03:44:46 -04:00
										 |  |  |  |       if (inAssetDir[i.entries[j]]) { | 
					
						
							| 
									
										
										
										
											2023-05-16 01:04:26 -04:00
										 |  |  |  |         i.entries.erase(i.entries.begin()+j); | 
					
						
							|  |  |  |  |         j--; | 
					
						
							|  |  |  |  |         continue; | 
					
						
							|  |  |  |  |       } | 
					
						
							| 
									
										
										
										
											2023-03-12 05:10:46 -04:00
										 |  |  |  |        | 
					
						
							|  |  |  |  |       // mark entry as present
 | 
					
						
							| 
									
										
										
										
											2023-05-16 03:44:46 -04:00
										 |  |  |  |       inAssetDir[i.entries[j]]=true; | 
					
						
							| 
									
										
										
										
											2023-03-12 05:10:46 -04:00
										 |  |  |  |     } | 
					
						
							|  |  |  |  |   } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   // get unsorted directory
 | 
					
						
							|  |  |  |  |   DivAssetDir* unsortedDir=NULL; | 
					
						
							|  |  |  |  |   for (DivAssetDir& i: dir) { | 
					
						
							| 
									
										
										
										
											2023-03-12 17:13:00 -04:00
										 |  |  |  |     if (i.name.empty()) { | 
					
						
							| 
									
										
										
										
											2023-03-12 05:10:46 -04:00
										 |  |  |  |       unsortedDir=&i; | 
					
						
							|  |  |  |  |       break; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  |   } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   // add missing items to unsorted directory
 | 
					
						
							|  |  |  |  |   for (size_t i=0; i<entries; i++) { | 
					
						
							|  |  |  |  |     if (!inAssetDir[i]) { | 
					
						
							| 
									
										
										
										
											2023-05-16 20:00:05 -04:00
										 |  |  |  |       // create unsorted directory if it doesn't exist
 | 
					
						
							|  |  |  |  |       if (unsortedDir==NULL) { | 
					
						
							|  |  |  |  |         dir.push_back(DivAssetDir("")); | 
					
						
							|  |  |  |  |         unsortedDir=&(*dir.rbegin()); | 
					
						
							|  |  |  |  |       } | 
					
						
							| 
									
										
										
										
											2023-03-12 05:10:46 -04:00
										 |  |  |  |       unsortedDir->entries.push_back(i); | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  |   } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   delete[] inAssetDir; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-28 04:36:15 -04:00
										 |  |  |  | void DivEngine::swapChannelsP(int src, int dest) { | 
					
						
							|  |  |  |  |   if (src<0 || src>=chans) return; | 
					
						
							|  |  |  |  |   if (dest<0 || dest>=chans) return; | 
					
						
							|  |  |  |  |   BUSY_BEGIN; | 
					
						
							|  |  |  |  |   saveLock.lock(); | 
					
						
							|  |  |  |  |   swapChannels(src,dest); | 
					
						
							|  |  |  |  |   saveLock.unlock(); | 
					
						
							|  |  |  |  |   BUSY_END; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-05-15 02:42:49 -04:00
										 |  |  |  | void DivEngine::changeSongP(size_t index) { | 
					
						
							|  |  |  |  |   if (index>=song.subsong.size()) return; | 
					
						
							|  |  |  |  |   if (index==curSubSongIndex) return; | 
					
						
							|  |  |  |  |   stop(); | 
					
						
							|  |  |  |  |   BUSY_BEGIN; | 
					
						
							|  |  |  |  |   saveLock.lock(); | 
					
						
							|  |  |  |  |   changeSong(index); | 
					
						
							|  |  |  |  |   saveLock.unlock(); | 
					
						
							|  |  |  |  |   BUSY_END; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | int DivEngine::addSubSong() { | 
					
						
							|  |  |  |  |   if (song.subsong.size()>=127) return -1; | 
					
						
							|  |  |  |  |   BUSY_BEGIN; | 
					
						
							|  |  |  |  |   saveLock.lock(); | 
					
						
							|  |  |  |  |   song.subsong.push_back(new DivSubSong); | 
					
						
							| 
									
										
										
										
											2024-09-02 16:39:35 -04:00
										 |  |  |  |   for (unsigned char i:curChanMask) { | 
					
						
							|  |  |  |  |     int j=i-1; | 
					
						
							|  |  |  |  |     if (j<0) j=0; | 
					
						
							|  |  |  |  |     if (j>DIV_MAX_CHANS) j=DIV_MAX_CHANS-1; | 
					
						
							|  |  |  |  |     song.subsong.back()->chanShow[j]=false; | 
					
						
							|  |  |  |  |     song.subsong.back()->chanShowChanOsc[j]=false; | 
					
						
							|  |  |  |  |   } | 
					
						
							| 
									
										
										
										
											2022-05-15 02:42:49 -04:00
										 |  |  |  |   saveLock.unlock(); | 
					
						
							|  |  |  |  |   BUSY_END; | 
					
						
							|  |  |  |  |   return song.subsong.size()-1; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-07-01 00:37:22 -04:00
										 |  |  |  | int DivEngine::duplicateSubSong(int index) { | 
					
						
							|  |  |  |  |   if (song.subsong.size()>=127) return -1; | 
					
						
							|  |  |  |  |   BUSY_BEGIN; | 
					
						
							|  |  |  |  |   saveLock.lock(); | 
					
						
							|  |  |  |  |   DivSubSong* theCopy=new DivSubSong; | 
					
						
							|  |  |  |  |   DivSubSong* theOrig=song.subsong[index]; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   theCopy->name=theOrig->name; | 
					
						
							|  |  |  |  |   theCopy->notes=theOrig->notes; | 
					
						
							|  |  |  |  |   theCopy->hilightA=theOrig->hilightA; | 
					
						
							|  |  |  |  |   theCopy->hilightB=theOrig->hilightB; | 
					
						
							|  |  |  |  |   theCopy->timeBase=theOrig->timeBase; | 
					
						
							|  |  |  |  |   theCopy->arpLen=theOrig->arpLen; | 
					
						
							|  |  |  |  |   theCopy->speeds=theOrig->speeds; | 
					
						
							|  |  |  |  |   theCopy->virtualTempoN=theOrig->virtualTempoN; | 
					
						
							|  |  |  |  |   theCopy->virtualTempoD=theOrig->virtualTempoD; | 
					
						
							|  |  |  |  |   theCopy->hz=theOrig->hz; | 
					
						
							|  |  |  |  |   theCopy->patLen=theOrig->patLen; | 
					
						
							|  |  |  |  |   theCopy->ordersLen=theOrig->ordersLen; | 
					
						
							|  |  |  |  |   theCopy->orders=theOrig->orders; | 
					
						
							|  |  |  |  |    | 
					
						
							|  |  |  |  |   memcpy(theCopy->chanShow,theOrig->chanShow,DIV_MAX_CHANS*sizeof(bool)); | 
					
						
							| 
									
										
										
										
											2024-02-24 18:12:02 -05:00
										 |  |  |  |   memcpy(theCopy->chanShowChanOsc,theOrig->chanShowChanOsc,DIV_MAX_CHANS*sizeof(bool)); | 
					
						
							| 
									
										
										
										
											2023-07-01 00:37:22 -04:00
										 |  |  |  |   memcpy(theCopy->chanCollapse,theOrig->chanCollapse,DIV_MAX_CHANS); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   for (int i=0; i<DIV_MAX_CHANS; i++) { | 
					
						
							|  |  |  |  |     theCopy->chanName[i]=theOrig->chanName[i]; | 
					
						
							|  |  |  |  |     theCopy->chanShortName[i]=theOrig->chanShortName[i]; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     theCopy->pat[i].effectCols=theOrig->pat[i].effectCols; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     for (int j=0; j<DIV_MAX_PATTERNS; j++) { | 
					
						
							|  |  |  |  |       if (theOrig->pat[i].data[j]==NULL) continue; | 
					
						
							|  |  |  |  |       DivPattern* origPat=theOrig->pat[i].getPattern(j,false); | 
					
						
							|  |  |  |  |       DivPattern* copyPat=theCopy->pat[i].getPattern(j,true); | 
					
						
							|  |  |  |  |       origPat->copyOn(copyPat); | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  |   } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   song.subsong.push_back(theCopy); | 
					
						
							|  |  |  |  |    | 
					
						
							|  |  |  |  |   saveLock.unlock(); | 
					
						
							|  |  |  |  |   BUSY_END; | 
					
						
							|  |  |  |  |   return song.subsong.size()-1; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-05-15 02:42:49 -04:00
										 |  |  |  | bool DivEngine::removeSubSong(int index) { | 
					
						
							|  |  |  |  |   if (song.subsong.size()<=1) return false; | 
					
						
							|  |  |  |  |   stop(); | 
					
						
							|  |  |  |  |   BUSY_BEGIN; | 
					
						
							|  |  |  |  |   saveLock.lock(); | 
					
						
							|  |  |  |  |   song.subsong[index]->clearData(); | 
					
						
							|  |  |  |  |   delete song.subsong[index]; | 
					
						
							|  |  |  |  |   song.subsong.erase(song.subsong.begin()+index); | 
					
						
							|  |  |  |  |   changeSong(0); | 
					
						
							|  |  |  |  |   saveLock.unlock(); | 
					
						
							|  |  |  |  |   BUSY_END; | 
					
						
							|  |  |  |  |   return true; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-06-01 19:50:30 -04:00
										 |  |  |  | void DivEngine::moveSubSongUp(size_t index) { | 
					
						
							|  |  |  |  |   if (index<1 || index>=song.subsong.size()) return; | 
					
						
							|  |  |  |  |   BUSY_BEGIN; | 
					
						
							|  |  |  |  |   saveLock.lock(); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   if (index==curSubSongIndex) { | 
					
						
							|  |  |  |  |     curSubSongIndex--; | 
					
						
							|  |  |  |  |   } else if (index-1==curSubSongIndex) { | 
					
						
							|  |  |  |  |     curSubSongIndex++; | 
					
						
							|  |  |  |  |   } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   DivSubSong* prev=song.subsong[index-1]; | 
					
						
							|  |  |  |  |   song.subsong[index-1]=song.subsong[index]; | 
					
						
							|  |  |  |  |   song.subsong[index]=prev; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   saveLock.unlock(); | 
					
						
							|  |  |  |  |   BUSY_END; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | void DivEngine::moveSubSongDown(size_t index) { | 
					
						
							|  |  |  |  |   if (index>=song.subsong.size()-1) return; | 
					
						
							|  |  |  |  |   BUSY_BEGIN; | 
					
						
							|  |  |  |  |   saveLock.lock(); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   if (index==curSubSongIndex) { | 
					
						
							|  |  |  |  |     curSubSongIndex++; | 
					
						
							|  |  |  |  |   } else if (index+1==curSubSongIndex) { | 
					
						
							|  |  |  |  |     curSubSongIndex--; | 
					
						
							|  |  |  |  |   } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   DivSubSong* prev=song.subsong[index+1]; | 
					
						
							|  |  |  |  |   song.subsong[index+1]=song.subsong[index]; | 
					
						
							|  |  |  |  |   song.subsong[index]=prev; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   saveLock.unlock(); | 
					
						
							|  |  |  |  |   BUSY_END; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-05-17 02:42:21 -04:00
										 |  |  |  | void DivEngine::clearSubSongs() { | 
					
						
							|  |  |  |  |   BUSY_BEGIN; | 
					
						
							|  |  |  |  |   saveLock.lock(); | 
					
						
							|  |  |  |  |   song.clearSongData(); | 
					
						
							|  |  |  |  |   changeSong(0); | 
					
						
							|  |  |  |  |   curOrder=0; | 
					
						
							| 
									
										
										
										
											2022-06-06 02:05:06 -04:00
										 |  |  |  |   prevOrder=0; | 
					
						
							| 
									
										
										
										
											2022-05-17 02:42:21 -04:00
										 |  |  |  |   saveLock.unlock(); | 
					
						
							|  |  |  |  |   BUSY_END; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-09-08 03:52:15 -04:00
										 |  |  |  | void DivEngine::delUnusedIns() { | 
					
						
							|  |  |  |  |   BUSY_BEGIN; | 
					
						
							|  |  |  |  |   saveLock.lock(); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   bool isUsed[256]; | 
					
						
							|  |  |  |  |   memset(isUsed,0,256*sizeof(bool)); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   // scan
 | 
					
						
							|  |  |  |  |   for (int i=0; i<chans; i++) { | 
					
						
							|  |  |  |  |     for (size_t j=0; j<song.subsong.size(); j++) { | 
					
						
							|  |  |  |  |       for (int k=0; k<DIV_MAX_PATTERNS; k++) { | 
					
						
							|  |  |  |  |         if (song.subsong[j]->pat[i].data[k]==NULL) continue; | 
					
						
							|  |  |  |  |         for (int l=0; l<song.subsong[j]->patLen; l++) { | 
					
						
							|  |  |  |  |           if (song.subsong[j]->pat[i].data[k]->data[l][2]>=0 && song.subsong[j]->pat[i].data[k]->data[l][2]<256) { | 
					
						
							|  |  |  |  |             isUsed[song.subsong[j]->pat[i].data[k]->data[l][2]]=true; | 
					
						
							|  |  |  |  |           } | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  |       } | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  |   } | 
					
						
							|  |  |  |  |    | 
					
						
							|  |  |  |  |   // delete
 | 
					
						
							|  |  |  |  |   for (int i=0; i<song.insLen; i++) { | 
					
						
							|  |  |  |  |     if (!isUsed[i]) { | 
					
						
							|  |  |  |  |       delInstrumentUnsafe(i); | 
					
						
							|  |  |  |  |       // rotate
 | 
					
						
							|  |  |  |  |       for (int j=i; j<255; j++) { | 
					
						
							|  |  |  |  |         isUsed[j]=isUsed[j+1]; | 
					
						
							|  |  |  |  |       } | 
					
						
							|  |  |  |  |       isUsed[255]=true; | 
					
						
							|  |  |  |  |       i--; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  |   } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   saveLock.unlock(); | 
					
						
							|  |  |  |  |   BUSY_END; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | void DivEngine::delUnusedWaves() { | 
					
						
							|  |  |  |  |   BUSY_BEGIN; | 
					
						
							|  |  |  |  |   saveLock.lock(); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   saveLock.unlock(); | 
					
						
							|  |  |  |  |   BUSY_END; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | void DivEngine::delUnusedSamples() { | 
					
						
							|  |  |  |  |   BUSY_BEGIN; | 
					
						
							|  |  |  |  |   saveLock.lock(); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   bool isUsed[256]; | 
					
						
							|  |  |  |  |   memset(isUsed,0,256*sizeof(bool)); | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-05-25 22:35:06 -04:00
										 |  |  |  |   // scan in instruments
 | 
					
						
							| 
									
										
										
										
											2023-09-08 03:52:15 -04:00
										 |  |  |  |   for (DivInstrument* i: song.ins) { | 
					
						
							|  |  |  |  |     if ((i->type==DIV_INS_PCE && i->amiga.useSample) || | 
					
						
							|  |  |  |  |         i->type==DIV_INS_MSM6258 || | 
					
						
							|  |  |  |  |         i->type==DIV_INS_MSM6295 || | 
					
						
							|  |  |  |  |         i->type==DIV_INS_ADPCMA || | 
					
						
							|  |  |  |  |         i->type==DIV_INS_ADPCMB || | 
					
						
							|  |  |  |  |         i->type==DIV_INS_SEGAPCM || | 
					
						
							|  |  |  |  |         i->type==DIV_INS_QSOUND || | 
					
						
							|  |  |  |  |         i->type==DIV_INS_YMZ280B || | 
					
						
							|  |  |  |  |         i->type==DIV_INS_RF5C68 || | 
					
						
							|  |  |  |  |         i->type==DIV_INS_AMIGA || | 
					
						
							|  |  |  |  |         i->type==DIV_INS_MULTIPCM || | 
					
						
							|  |  |  |  |         (i->type==DIV_INS_MIKEY && i->amiga.useSample) || | 
					
						
							|  |  |  |  |         (i->type==DIV_INS_X1_010 && i->amiga.useSample) || | 
					
						
							|  |  |  |  |         (i->type==DIV_INS_SWAN && i->amiga.useSample) || | 
					
						
							|  |  |  |  |         (i->type==DIV_INS_AY && i->amiga.useSample) || | 
					
						
							|  |  |  |  |         (i->type==DIV_INS_AY8930 && i->amiga.useSample) || | 
					
						
							|  |  |  |  |         (i->type==DIV_INS_VRC6 && i->amiga.useSample) || | 
					
						
							|  |  |  |  |         (i->type==DIV_INS_SU && i->amiga.useSample) || | 
					
						
							|  |  |  |  |         i->type==DIV_INS_SNES || | 
					
						
							|  |  |  |  |         i->type==DIV_INS_ES5506 || | 
					
						
							|  |  |  |  |         i->type==DIV_INS_K007232 || | 
					
						
							|  |  |  |  |         i->type==DIV_INS_GA20 || | 
					
						
							|  |  |  |  |         i->type==DIV_INS_K053260 || | 
					
						
							|  |  |  |  |         i->type==DIV_INS_C140 || | 
					
						
							| 
									
										
										
										
											2024-01-09 23:17:36 -05:00
										 |  |  |  |         i->type==DIV_INS_C219 || | 
					
						
							| 
									
										
										
										
											2024-03-17 16:44:38 -04:00
										 |  |  |  |         i->type==DIV_INS_NDS || | 
					
						
							| 
									
										
										
										
											2024-03-16 03:59:02 -04:00
										 |  |  |  |         i->type==DIV_INS_GBA_DMA || | 
					
						
							|  |  |  |  |         i->type==DIV_INS_GBA_MINMOD) { | 
					
						
							| 
									
										
										
										
											2023-09-08 03:52:15 -04:00
										 |  |  |  |       if (i->amiga.initSample>=0 && i->amiga.initSample<song.sampleLen) { | 
					
						
							|  |  |  |  |         isUsed[i->amiga.initSample]=true; | 
					
						
							|  |  |  |  |       } | 
					
						
							|  |  |  |  |       if (i->amiga.useNoteMap) { | 
					
						
							|  |  |  |  |         for (int j=0; j<120; j++) { | 
					
						
							|  |  |  |  |           if (i->amiga.noteMap[j].map>=0 && i->amiga.noteMap[j].map<song.sampleLen) { | 
					
						
							|  |  |  |  |             isUsed[i->amiga.noteMap[j].map]=true; | 
					
						
							|  |  |  |  |           } | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  |       } | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  |   } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-05-25 22:35:06 -04:00
										 |  |  |  |   // scan in pattern (legacy sample mode)
 | 
					
						
							| 
									
										
										
										
											2024-05-25 23:03:35 -04:00
										 |  |  |  |   // disabled because it is unreliable
 | 
					
						
							|  |  |  |  |   /*
 | 
					
						
							| 
									
										
										
										
											2024-05-25 22:35:06 -04:00
										 |  |  |  |   for (DivSubSong* i: song.subsong) { | 
					
						
							| 
									
										
										
										
											2024-05-25 23:03:35 -04:00
										 |  |  |  |     for (int j=0; j<getTotalChannelCount(); j++) { | 
					
						
							|  |  |  |  |       bool is17On=false; | 
					
						
							|  |  |  |  |       int bank=0; | 
					
						
							|  |  |  |  |       for (int k=0; k<i->ordersLen; k++) { | 
					
						
							|  |  |  |  |         DivPattern* p=i->pat[j].getPattern(i->orders.ord[j][k],false); | 
					
						
							|  |  |  |  |         for (int l=0; l<i->patLen; l++) { | 
					
						
							|  |  |  |  |           for (int m=0; m<i->pat[j].effectCols; m++) { | 
					
						
							|  |  |  |  |             if (p->data[l][4+(m<<1)]==0x17) { | 
					
						
							|  |  |  |  |               is17On=(p->data[l][5+(m<<1)]>0); | 
					
						
							|  |  |  |  |             } | 
					
						
							|  |  |  |  |             if (p->data[l][4+(m<<1)]==0xeb) { | 
					
						
							|  |  |  |  |               bank=p->data[l][5+(m<<1)]; | 
					
						
							|  |  |  |  |               if (bank==-1) bank=0; | 
					
						
							|  |  |  |  |             } | 
					
						
							|  |  |  |  |           } | 
					
						
							|  |  |  |  |           if (is17On) { | 
					
						
							|  |  |  |  |             if (p->data[l][1]!=0 || p->data[l][0]!=0) { | 
					
						
							|  |  |  |  |               if (p->data[l][0]<=12) { | 
					
						
							|  |  |  |  |                 int note=(12*bank)+(p->data[l][0]%12); | 
					
						
							|  |  |  |  |                 if (note<256) isUsed[note]=true; | 
					
						
							|  |  |  |  |               } | 
					
						
							|  |  |  |  |             } | 
					
						
							|  |  |  |  |           } | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  |       } | 
					
						
							| 
									
										
										
										
											2024-05-25 22:35:06 -04:00
										 |  |  |  |     } | 
					
						
							| 
									
										
										
										
											2024-05-25 23:03:35 -04:00
										 |  |  |  |   }*/ | 
					
						
							| 
									
										
										
										
											2024-05-25 22:35:06 -04:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-09-08 03:52:15 -04:00
										 |  |  |  |   // delete
 | 
					
						
							|  |  |  |  |   for (int i=0; i<song.sampleLen; i++) { | 
					
						
							|  |  |  |  |     if (!isUsed[i]) { | 
					
						
							|  |  |  |  |       delSampleUnsafe(i,false); | 
					
						
							|  |  |  |  |       // rotate
 | 
					
						
							|  |  |  |  |       for (int j=i; j<255; j++) { | 
					
						
							|  |  |  |  |         isUsed[j]=isUsed[j+1]; | 
					
						
							|  |  |  |  |       } | 
					
						
							|  |  |  |  |       isUsed[255]=true; | 
					
						
							|  |  |  |  |       i--; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  |   } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   // render
 | 
					
						
							|  |  |  |  |   renderSamples(); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   saveLock.unlock(); | 
					
						
							|  |  |  |  |   BUSY_END; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-02-21 15:55:15 -05:00
										 |  |  |  | bool DivEngine::changeSystem(int index, DivSystem which, bool preserveOrder) { | 
					
						
							|  |  |  |  |   if (index<0 || index>=song.systemLen) { | 
					
						
							| 
									
										
										
										
											2024-05-27 05:02:07 -04:00
										 |  |  |  |     lastError=_("invalid index"); | 
					
						
							| 
									
										
										
										
											2024-02-21 15:55:15 -05:00
										 |  |  |  |     return false; | 
					
						
							|  |  |  |  |   } | 
					
						
							|  |  |  |  |   if (chans-getChannelCount(song.system[index])+getChannelCount(which)>DIV_MAX_CHANS) { | 
					
						
							| 
									
										
										
										
											2024-05-27 05:02:07 -04:00
										 |  |  |  |     lastError=fmt::sprintf(_("max number of total channels is %d"),DIV_MAX_CHANS); | 
					
						
							| 
									
										
										
										
											2024-02-21 15:55:15 -05:00
										 |  |  |  |     return false; | 
					
						
							|  |  |  |  |   } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-28 04:36:15 -04:00
										 |  |  |  |   int chanCount=chans; | 
					
						
							| 
									
										
										
										
											2021-12-17 22:14:41 -05:00
										 |  |  |  |   quitDispatch(); | 
					
						
							| 
									
										
										
										
											2022-03-23 22:38:28 -04:00
										 |  |  |  |   BUSY_BEGIN; | 
					
						
							| 
									
										
										
										
											2022-03-21 18:56:48 -04:00
										 |  |  |  |   saveLock.lock(); | 
					
						
							| 
									
										
										
										
											2022-04-28 04:36:15 -04:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |   if (!preserveOrder) { | 
					
						
							|  |  |  |  |     int firstChan=0; | 
					
						
							|  |  |  |  |     int chanMovement=getChannelCount(which)-getChannelCount(song.system[index]); | 
					
						
							|  |  |  |  |     while (dispatchOfChan[firstChan]!=index) firstChan++; | 
					
						
							|  |  |  |  |     int lastChan=firstChan+getChannelCount(song.system[index]); | 
					
						
							|  |  |  |  |     if (chanMovement!=0) { | 
					
						
							|  |  |  |  |       if (chanMovement>0) { | 
					
						
							|  |  |  |  |         // add channels
 | 
					
						
							|  |  |  |  |         for (int i=chanCount+chanMovement-1; i>=lastChan+chanMovement; i--) { | 
					
						
							|  |  |  |  |           swapChannels(i,i-chanMovement); | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  |         for (int i=lastChan; i<lastChan+chanMovement; i++) { | 
					
						
							|  |  |  |  |           stompChannel(i); | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  |       } else { | 
					
						
							|  |  |  |  |         // remove channels
 | 
					
						
							|  |  |  |  |         for (int i=lastChan+chanMovement; i<lastChan; i++) { | 
					
						
							|  |  |  |  |           stompChannel(i); | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  |         for (int i=lastChan+chanMovement; i<chanCount+chanMovement; i++) { | 
					
						
							|  |  |  |  |           swapChannels(i,i-chanMovement); | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  |       } | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  |   } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-01-08 16:03:32 -05:00
										 |  |  |  |   song.system[index]=which; | 
					
						
							| 
									
										
										
										
											2022-09-30 01:26:54 -04:00
										 |  |  |  |   song.systemFlags[index].clear(); | 
					
						
							| 
									
										
										
										
											2022-01-08 16:03:32 -05:00
										 |  |  |  |   recalcChans(); | 
					
						
							| 
									
										
										
										
											2022-03-21 18:56:48 -04:00
										 |  |  |  |   saveLock.unlock(); | 
					
						
							| 
									
										
										
										
											2022-03-23 22:38:28 -04:00
										 |  |  |  |   BUSY_END; | 
					
						
							| 
									
										
										
										
											2021-12-17 22:14:41 -05:00
										 |  |  |  |   initDispatch(); | 
					
						
							| 
									
										
										
										
											2022-03-23 22:38:28 -04:00
										 |  |  |  |   BUSY_BEGIN; | 
					
						
							| 
									
										
										
										
											2021-12-17 22:14:41 -05:00
										 |  |  |  |   renderSamples(); | 
					
						
							|  |  |  |  |   reset(); | 
					
						
							| 
									
										
										
										
											2022-03-23 22:38:28 -04:00
										 |  |  |  |   BUSY_END; | 
					
						
							| 
									
										
										
										
											2024-02-21 15:55:15 -05:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |   return true; | 
					
						
							| 
									
										
										
										
											2021-12-17 22:14:41 -05:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-01-08 18:18:23 -05:00
										 |  |  |  | bool DivEngine::addSystem(DivSystem which) { | 
					
						
							| 
									
										
										
										
											2024-05-04 14:40:29 -04:00
										 |  |  |  |   if (song.systemLen>=DIV_MAX_CHIPS) { | 
					
						
							| 
									
										
										
										
											2024-05-27 05:02:07 -04:00
										 |  |  |  |     lastError=fmt::sprintf(_("max number of systems is %d"),DIV_MAX_CHIPS); | 
					
						
							| 
									
										
										
										
											2022-01-08 18:18:23 -05:00
										 |  |  |  |     return false; | 
					
						
							|  |  |  |  |   } | 
					
						
							| 
									
										
										
										
											2022-05-27 00:38:45 -04:00
										 |  |  |  |   if (chans+getChannelCount(which)>DIV_MAX_CHANS) { | 
					
						
							| 
									
										
										
										
											2024-05-27 05:02:07 -04:00
										 |  |  |  |     lastError=fmt::sprintf(_("max number of total channels is %d"),DIV_MAX_CHANS); | 
					
						
							| 
									
										
										
										
											2022-01-08 18:18:23 -05:00
										 |  |  |  |     return false; | 
					
						
							|  |  |  |  |   } | 
					
						
							|  |  |  |  |   quitDispatch(); | 
					
						
							| 
									
										
										
										
											2022-03-23 22:38:28 -04:00
										 |  |  |  |   BUSY_BEGIN; | 
					
						
							| 
									
										
										
										
											2022-03-21 18:56:48 -04:00
										 |  |  |  |   saveLock.lock(); | 
					
						
							| 
									
										
										
										
											2022-03-15 19:34:41 -04:00
										 |  |  |  |   song.system[song.systemLen]=which; | 
					
						
							| 
									
										
										
										
											2023-01-06 17:43:08 -05:00
										 |  |  |  |   song.systemVol[song.systemLen]=1.0; | 
					
						
							| 
									
										
										
										
											2022-03-15 19:34:41 -04:00
										 |  |  |  |   song.systemPan[song.systemLen]=0; | 
					
						
							| 
									
										
										
										
											2023-01-06 17:43:08 -05:00
										 |  |  |  |   song.systemPanFR[song.systemLen]=0; | 
					
						
							| 
									
										
										
										
											2022-09-30 01:26:54 -04:00
										 |  |  |  |   song.systemFlags[song.systemLen++].clear(); | 
					
						
							| 
									
										
										
										
											2022-01-08 18:18:23 -05:00
										 |  |  |  |   recalcChans(); | 
					
						
							| 
									
										
										
										
											2022-03-21 18:56:48 -04:00
										 |  |  |  |   saveLock.unlock(); | 
					
						
							| 
									
										
										
										
											2022-03-23 22:38:28 -04:00
										 |  |  |  |   BUSY_END; | 
					
						
							| 
									
										
										
										
											2022-01-08 18:18:23 -05:00
										 |  |  |  |   initDispatch(); | 
					
						
							| 
									
										
										
										
											2022-03-23 22:38:28 -04:00
										 |  |  |  |   BUSY_BEGIN; | 
					
						
							| 
									
										
										
										
											2023-01-10 15:58:15 -05:00
										 |  |  |  |   saveLock.lock(); | 
					
						
							|  |  |  |  |   if (song.patchbayAuto) { | 
					
						
							|  |  |  |  |     autoPatchbay(); | 
					
						
							|  |  |  |  |   } else { | 
					
						
							|  |  |  |  |     int i=song.systemLen-1; | 
					
						
							|  |  |  |  |     if (disCont[i].dispatch!=NULL) { | 
					
						
							|  |  |  |  |       unsigned int outs=disCont[i].dispatch->getOutputCount(); | 
					
						
							|  |  |  |  |       if (outs>16) outs=16; | 
					
						
							|  |  |  |  |       if (outs<2) { | 
					
						
							| 
									
										
										
										
											2023-08-24 12:08:06 -04:00
										 |  |  |  |         song.patchbay.reserve(DIV_MAX_OUTPUTS); | 
					
						
							| 
									
										
										
										
											2023-01-10 15:58:15 -05:00
										 |  |  |  |         for (unsigned int j=0; j<DIV_MAX_OUTPUTS; j++) { | 
					
						
							|  |  |  |  |           song.patchbay.push_back((i<<20)|j); | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  |       } else { | 
					
						
							| 
									
										
										
										
											2023-10-29 04:38:51 -04:00
										 |  |  |  |         if (outs>0) song.patchbay.reserve(outs); | 
					
						
							| 
									
										
										
										
											2023-01-10 15:58:15 -05:00
										 |  |  |  |         for (unsigned int j=0; j<outs; j++) { | 
					
						
							|  |  |  |  |           song.patchbay.push_back((i<<20)|(j<<16)|j); | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  |       } | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  |   } | 
					
						
							|  |  |  |  |   saveLock.unlock(); | 
					
						
							| 
									
										
										
										
											2022-01-08 18:18:23 -05:00
										 |  |  |  |   renderSamples(); | 
					
						
							| 
									
										
										
										
											2024-02-21 15:55:15 -05:00
										 |  |  |  |   reset(); | 
					
						
							|  |  |  |  |   BUSY_END; | 
					
						
							|  |  |  |  |   return true; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | bool DivEngine::duplicateSystem(int index, bool pat, bool end) { | 
					
						
							|  |  |  |  |   if (index<0 || index>=song.systemLen) { | 
					
						
							| 
									
										
										
										
											2024-05-27 05:02:07 -04:00
										 |  |  |  |     lastError=_("invalid index"); | 
					
						
							| 
									
										
										
										
											2024-02-21 15:55:15 -05:00
										 |  |  |  |     return false; | 
					
						
							|  |  |  |  |   } | 
					
						
							| 
									
										
										
										
											2024-05-04 14:40:29 -04:00
										 |  |  |  |   if (song.systemLen>=DIV_MAX_CHIPS) { | 
					
						
							| 
									
										
										
										
											2024-05-27 05:02:07 -04:00
										 |  |  |  |     lastError=fmt::sprintf(_("max number of systems is %d"),DIV_MAX_CHIPS); | 
					
						
							| 
									
										
										
										
											2024-02-21 15:55:15 -05:00
										 |  |  |  |     return false; | 
					
						
							|  |  |  |  |   } | 
					
						
							|  |  |  |  |   if (chans+getChannelCount(song.system[index])>DIV_MAX_CHANS) { | 
					
						
							| 
									
										
										
										
											2024-05-27 05:02:07 -04:00
										 |  |  |  |     lastError=fmt::sprintf(_("max number of total channels is %d"),DIV_MAX_CHANS); | 
					
						
							| 
									
										
										
										
											2024-02-21 15:55:15 -05:00
										 |  |  |  |     return false; | 
					
						
							|  |  |  |  |   } | 
					
						
							|  |  |  |  |   quitDispatch(); | 
					
						
							|  |  |  |  |   BUSY_BEGIN; | 
					
						
							|  |  |  |  |   saveLock.lock(); | 
					
						
							|  |  |  |  |   song.system[song.systemLen]=song.system[index]; | 
					
						
							|  |  |  |  |   song.systemVol[song.systemLen]=song.systemVol[index]; | 
					
						
							|  |  |  |  |   song.systemPan[song.systemLen]=song.systemPan[index]; | 
					
						
							|  |  |  |  |   song.systemPanFR[song.systemLen]=song.systemPanFR[index]; | 
					
						
							|  |  |  |  |   song.systemFlags[song.systemLen++]=song.systemFlags[index]; | 
					
						
							|  |  |  |  |   recalcChans(); | 
					
						
							|  |  |  |  |   saveLock.unlock(); | 
					
						
							|  |  |  |  |   BUSY_END; | 
					
						
							|  |  |  |  |   initDispatch(); | 
					
						
							|  |  |  |  |   BUSY_BEGIN; | 
					
						
							|  |  |  |  |   saveLock.lock(); | 
					
						
							|  |  |  |  |   if (song.patchbayAuto) { | 
					
						
							|  |  |  |  |     autoPatchbay(); | 
					
						
							|  |  |  |  |   } else { | 
					
						
							|  |  |  |  |     int i=song.systemLen-1; | 
					
						
							|  |  |  |  |     if (disCont[i].dispatch!=NULL) { | 
					
						
							|  |  |  |  |       unsigned int outs=disCont[i].dispatch->getOutputCount(); | 
					
						
							|  |  |  |  |       if (outs>16) outs=16; | 
					
						
							|  |  |  |  |       if (outs<2) { | 
					
						
							|  |  |  |  |         song.patchbay.reserve(DIV_MAX_OUTPUTS); | 
					
						
							|  |  |  |  |         for (unsigned int j=0; j<DIV_MAX_OUTPUTS; j++) { | 
					
						
							|  |  |  |  |           song.patchbay.push_back((i<<20)|j); | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  |       } else { | 
					
						
							|  |  |  |  |         if (outs>0) song.patchbay.reserve(outs); | 
					
						
							|  |  |  |  |         for (unsigned int j=0; j<outs; j++) { | 
					
						
							|  |  |  |  |           song.patchbay.push_back((i<<20)|(j<<16)|j); | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  |       } | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  |   } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   // duplicate patterns
 | 
					
						
							|  |  |  |  |   if (pat) { | 
					
						
							| 
									
										
										
										
											2024-02-21 16:27:52 -05:00
										 |  |  |  |     int srcChan=0; | 
					
						
							|  |  |  |  |     int destChan=0; | 
					
						
							|  |  |  |  |     for (int i=0; i<index; i++) { | 
					
						
							|  |  |  |  |       srcChan+=getChannelCount(song.system[i]); | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  |     for (int i=0; i<song.systemLen-1; i++) { | 
					
						
							|  |  |  |  |       destChan+=getChannelCount(song.system[i]); | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  |     for (DivSubSong* i: song.subsong) { | 
					
						
							|  |  |  |  |       for (int j=0; j<getChannelCount(song.system[index]); j++) { | 
					
						
							|  |  |  |  |         i->pat[destChan+j].effectCols=i->pat[srcChan+j].effectCols; | 
					
						
							|  |  |  |  |         i->chanShow[destChan+j]=i->chanShow[srcChan+j]; | 
					
						
							|  |  |  |  |         i->chanShowChanOsc[destChan+j]=i->chanShowChanOsc[srcChan+j]; | 
					
						
							|  |  |  |  |         i->chanCollapse[destChan+j]=i->chanCollapse[srcChan+j]; | 
					
						
							|  |  |  |  |         i->chanName[destChan+j]=i->chanName[srcChan+j]; | 
					
						
							|  |  |  |  |         i->chanShortName[destChan+j]=i->chanShortName[srcChan+j]; | 
					
						
							|  |  |  |  |         for (int k=0; k<DIV_MAX_PATTERNS; k++) { | 
					
						
							|  |  |  |  |           if (i->pat[srcChan+j].data[k]!=NULL) { | 
					
						
							|  |  |  |  |             i->pat[srcChan+j].data[k]->copyOn(i->pat[destChan+j].getPattern(k,true)); | 
					
						
							|  |  |  |  |           } | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         for (int k=0; k<DIV_MAX_PATTERNS; k++) { | 
					
						
							|  |  |  |  |           i->orders.ord[destChan+j][k]=i->orders.ord[srcChan+j][k]; | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  |       } | 
					
						
							|  |  |  |  |     } | 
					
						
							| 
									
										
										
										
											2024-02-21 15:55:15 -05:00
										 |  |  |  |   } | 
					
						
							|  |  |  |  |   saveLock.unlock(); | 
					
						
							|  |  |  |  |   renderSamples(); | 
					
						
							| 
									
										
										
										
											2022-01-08 18:18:23 -05:00
										 |  |  |  |   reset(); | 
					
						
							| 
									
										
										
										
											2022-03-23 22:38:28 -04:00
										 |  |  |  |   BUSY_END; | 
					
						
							| 
									
										
										
										
											2024-02-22 12:48:16 -05:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |   if (!end) { | 
					
						
							|  |  |  |  |     quitDispatch(); | 
					
						
							|  |  |  |  |     BUSY_BEGIN; | 
					
						
							|  |  |  |  |     saveLock.lock(); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     for (int i=song.systemLen-1; i>index; i--) { | 
					
						
							|  |  |  |  |       swapSystemUnsafe(i,i-1,false); | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     recalcChans(); | 
					
						
							|  |  |  |  |     saveLock.unlock(); | 
					
						
							|  |  |  |  |     BUSY_END; | 
					
						
							|  |  |  |  |     initDispatch(); | 
					
						
							|  |  |  |  |     BUSY_BEGIN; | 
					
						
							|  |  |  |  |     renderSamples(); | 
					
						
							|  |  |  |  |     reset(); | 
					
						
							|  |  |  |  |     BUSY_END; | 
					
						
							|  |  |  |  |   } | 
					
						
							| 
									
										
										
										
											2022-01-08 18:18:23 -05:00
										 |  |  |  |   return true; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-05-15 02:55:25 -04:00
										 |  |  |  | // TODO: maybe issue with subsongs?
 | 
					
						
							| 
									
										
										
										
											2022-04-28 04:36:15 -04:00
										 |  |  |  | bool DivEngine::removeSystem(int index, bool preserveOrder) { | 
					
						
							| 
									
										
										
										
											2022-01-09 16:36:47 -05:00
										 |  |  |  |   if (song.systemLen<=1) { | 
					
						
							| 
									
										
										
										
											2024-05-27 05:02:07 -04:00
										 |  |  |  |     lastError=_("cannot remove the last one"); | 
					
						
							| 
									
										
										
										
											2022-01-09 16:36:47 -05:00
										 |  |  |  |     return false; | 
					
						
							|  |  |  |  |   } | 
					
						
							|  |  |  |  |   if (index<0 || index>=song.systemLen) { | 
					
						
							| 
									
										
										
										
											2024-05-27 05:02:07 -04:00
										 |  |  |  |     lastError=_("invalid index"); | 
					
						
							| 
									
										
										
										
											2022-01-09 16:36:47 -05:00
										 |  |  |  |     return false; | 
					
						
							|  |  |  |  |   } | 
					
						
							| 
									
										
										
										
											2022-04-28 04:36:15 -04:00
										 |  |  |  |   int chanCount=chans; | 
					
						
							| 
									
										
										
										
											2022-01-09 16:36:47 -05:00
										 |  |  |  |   quitDispatch(); | 
					
						
							| 
									
										
										
										
											2022-03-23 22:38:28 -04:00
										 |  |  |  |   BUSY_BEGIN; | 
					
						
							| 
									
										
										
										
											2022-03-21 18:56:48 -04:00
										 |  |  |  |   saveLock.lock(); | 
					
						
							| 
									
										
										
										
											2022-04-28 04:36:15 -04:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |   if (!preserveOrder) { | 
					
						
							|  |  |  |  |     int firstChan=0; | 
					
						
							|  |  |  |  |     while (dispatchOfChan[firstChan]!=index) firstChan++; | 
					
						
							|  |  |  |  |     for (int i=0; i<getChannelCount(song.system[index]); i++) { | 
					
						
							|  |  |  |  |       stompChannel(i+firstChan); | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  |     for (int i=firstChan+getChannelCount(song.system[index]); i<chanCount; i++) { | 
					
						
							|  |  |  |  |       swapChannels(i,i-getChannelCount(song.system[index])); | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  |   } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-01-10 15:58:15 -05:00
										 |  |  |  |   // patchbay
 | 
					
						
							|  |  |  |  |   for (size_t i=0; i<song.patchbay.size(); i++) { | 
					
						
							| 
									
										
										
										
											2023-01-11 00:10:42 -05:00
										 |  |  |  |     if (((song.patchbay[i]>>20)&0xfff)==(unsigned int)index) { | 
					
						
							| 
									
										
										
										
											2023-01-10 15:58:15 -05:00
										 |  |  |  |       song.patchbay.erase(song.patchbay.begin()+i); | 
					
						
							|  |  |  |  |       i--; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  |   } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-01-09 16:36:47 -05:00
										 |  |  |  |   song.system[index]=DIV_SYSTEM_NULL; | 
					
						
							|  |  |  |  |   song.systemLen--; | 
					
						
							|  |  |  |  |   for (int i=index; i<song.systemLen; i++) { | 
					
						
							| 
									
										
										
										
											2022-01-18 22:02:04 -05:00
										 |  |  |  |     song.system[i]=song.system[i+1]; | 
					
						
							| 
									
										
										
										
											2022-04-28 04:36:15 -04:00
										 |  |  |  |     song.systemVol[i]=song.systemVol[i+1]; | 
					
						
							|  |  |  |  |     song.systemPan[i]=song.systemPan[i+1]; | 
					
						
							| 
									
										
										
										
											2023-01-06 17:43:08 -05:00
										 |  |  |  |     song.systemPanFR[i]=song.systemPanFR[i+1]; | 
					
						
							| 
									
										
										
										
											2022-09-30 01:26:54 -04:00
										 |  |  |  |     song.systemFlags[i]=song.systemFlags[i+1]; | 
					
						
							| 
									
										
										
										
											2022-01-09 16:36:47 -05:00
										 |  |  |  |   } | 
					
						
							|  |  |  |  |   recalcChans(); | 
					
						
							| 
									
										
										
										
											2022-03-21 18:56:48 -04:00
										 |  |  |  |   saveLock.unlock(); | 
					
						
							| 
									
										
										
										
											2022-03-23 22:38:28 -04:00
										 |  |  |  |   BUSY_END; | 
					
						
							| 
									
										
										
										
											2022-01-09 16:36:47 -05:00
										 |  |  |  |   initDispatch(); | 
					
						
							| 
									
										
										
										
											2022-03-23 22:38:28 -04:00
										 |  |  |  |   BUSY_BEGIN; | 
					
						
							| 
									
										
										
										
											2022-01-09 16:36:47 -05:00
										 |  |  |  |   renderSamples(); | 
					
						
							|  |  |  |  |   reset(); | 
					
						
							| 
									
										
										
										
											2022-03-23 22:38:28 -04:00
										 |  |  |  |   BUSY_END; | 
					
						
							| 
									
										
										
										
											2022-01-09 16:36:47 -05:00
										 |  |  |  |   return true; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-02-22 12:48:16 -05:00
										 |  |  |  | void DivEngine::swapSystemUnsafe(int src, int dest, bool preserveOrder) { | 
					
						
							| 
									
										
										
										
											2022-08-19 17:25:32 -04:00
										 |  |  |  |   if (!preserveOrder) { | 
					
						
							|  |  |  |  |     // move channels
 | 
					
						
							| 
									
										
										
										
											2022-08-26 04:03:36 -04:00
										 |  |  |  |     unsigned char unswappedChannels[DIV_MAX_CHANS]; | 
					
						
							|  |  |  |  |     unsigned char swappedChannels[DIV_MAX_CHANS]; | 
					
						
							|  |  |  |  |     std::vector<std::vector<int>> swapList; | 
					
						
							|  |  |  |  |     std::vector<int> chanList; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     int tchans=0; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     for (int i=0; i<song.systemLen; i++) { | 
					
						
							|  |  |  |  |       tchans+=getChannelCount(song.system[i]); | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     memset(unswappedChannels,0,DIV_MAX_CHANS); | 
					
						
							|  |  |  |  |     memset(swappedChannels,0,DIV_MAX_CHANS); | 
					
						
							|  |  |  |  |      | 
					
						
							|  |  |  |  |     for (int i=0; i<tchans; i++) { | 
					
						
							|  |  |  |  |       unswappedChannels[i]=i; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     // prepare swap list
 | 
					
						
							|  |  |  |  |     int index=0; | 
					
						
							| 
									
										
										
										
											2023-10-29 04:38:51 -04:00
										 |  |  |  |     if (song.systemLen>0) swapList.reserve(song.systemLen); | 
					
						
							| 
									
										
										
										
											2022-08-26 04:03:36 -04:00
										 |  |  |  |     for (int i=0; i<song.systemLen; i++) { | 
					
						
							|  |  |  |  |       chanList.clear(); | 
					
						
							| 
									
										
										
										
											2023-08-24 16:33:53 -04:00
										 |  |  |  |       const int channelCount=getChannelCount(song.system[i]); | 
					
						
							| 
									
										
										
										
											2023-10-29 04:38:51 -04:00
										 |  |  |  |       if (channelCount>0) chanList.reserve(channelCount); | 
					
						
							| 
									
										
										
										
											2023-08-24 12:08:06 -04:00
										 |  |  |  |       for (int j=0; j<channelCount; j++) { | 
					
						
							| 
									
										
										
										
											2022-08-26 04:03:36 -04:00
										 |  |  |  |         chanList.push_back(index); | 
					
						
							|  |  |  |  |         index++; | 
					
						
							|  |  |  |  |       } | 
					
						
							|  |  |  |  |       swapList.push_back(chanList); | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  |     swapList[src].swap(swapList[dest]); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     // unfold it
 | 
					
						
							|  |  |  |  |     index=0; | 
					
						
							|  |  |  |  |     for (std::vector<int>& i: swapList) { | 
					
						
							|  |  |  |  |       for (int& j: i) { | 
					
						
							|  |  |  |  |         swappedChannels[index++]=j; | 
					
						
							|  |  |  |  |       } | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-31 03:52:35 -04:00
										 |  |  |  |     // swap channels
 | 
					
						
							| 
									
										
										
										
											2022-08-26 04:03:36 -04:00
										 |  |  |  |     logV("swap list:"); | 
					
						
							|  |  |  |  |     for (int i=0; i<tchans; i++) { | 
					
						
							|  |  |  |  |       logV("- %d -> %d",unswappedChannels[i],swappedChannels[i]); | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-31 03:52:35 -04:00
										 |  |  |  |     for (size_t i=0; i<song.subsong.size(); i++) { | 
					
						
							|  |  |  |  |       DivOrders prevOrders=song.subsong[i]->orders; | 
					
						
							| 
									
										
										
										
											2022-12-09 16:11:46 -05:00
										 |  |  |  |       DivPattern* prevPat[DIV_MAX_CHANS][DIV_MAX_PATTERNS]; | 
					
						
							| 
									
										
										
										
											2022-08-31 03:52:35 -04:00
										 |  |  |  |       unsigned char prevEffectCols[DIV_MAX_CHANS]; | 
					
						
							|  |  |  |  |       String prevChanName[DIV_MAX_CHANS]; | 
					
						
							|  |  |  |  |       String prevChanShortName[DIV_MAX_CHANS]; | 
					
						
							|  |  |  |  |       bool prevChanShow[DIV_MAX_CHANS]; | 
					
						
							| 
									
										
										
										
											2024-02-24 18:12:02 -05:00
										 |  |  |  |       bool prevChanShowChanOsc[DIV_MAX_CHANS]; | 
					
						
							| 
									
										
										
										
											2022-08-31 03:52:35 -04:00
										 |  |  |  |       unsigned char prevChanCollapse[DIV_MAX_CHANS]; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |       for (int j=0; j<tchans; j++) { | 
					
						
							| 
									
										
										
										
											2022-12-09 00:30:27 -05:00
										 |  |  |  |         for (int k=0; k<DIV_MAX_PATTERNS; k++) { | 
					
						
							| 
									
										
										
										
											2022-08-31 03:52:35 -04:00
										 |  |  |  |           prevPat[j][k]=song.subsong[i]->pat[j].data[k]; | 
					
						
							| 
									
										
										
										
											2022-08-26 04:03:36 -04:00
										 |  |  |  |         } | 
					
						
							| 
									
										
										
										
											2022-08-31 03:52:35 -04:00
										 |  |  |  |         prevEffectCols[j]=song.subsong[i]->pat[j].effectCols; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         prevChanName[j]=song.subsong[i]->chanName[j]; | 
					
						
							|  |  |  |  |         prevChanShortName[j]=song.subsong[i]->chanShortName[j]; | 
					
						
							|  |  |  |  |         prevChanShow[j]=song.subsong[i]->chanShow[j]; | 
					
						
							| 
									
										
										
										
											2024-02-24 18:12:02 -05:00
										 |  |  |  |         prevChanShowChanOsc[j]=song.subsong[i]->chanShowChanOsc[j]; | 
					
						
							| 
									
										
										
										
											2022-08-31 03:52:35 -04:00
										 |  |  |  |         prevChanCollapse[j]=song.subsong[i]->chanCollapse[j]; | 
					
						
							|  |  |  |  |       } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |       for (int j=0; j<tchans; j++) { | 
					
						
							| 
									
										
										
										
											2022-12-09 00:30:27 -05:00
										 |  |  |  |         for (int k=0; k<DIV_MAX_PATTERNS; k++) { | 
					
						
							| 
									
										
										
										
											2022-08-31 03:52:35 -04:00
										 |  |  |  |           song.subsong[i]->orders.ord[j][k]=prevOrders.ord[swappedChannels[j]][k]; | 
					
						
							|  |  |  |  |           song.subsong[i]->pat[j].data[k]=prevPat[swappedChannels[j]][k]; | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         song.subsong[i]->pat[j].effectCols=prevEffectCols[swappedChannels[j]]; | 
					
						
							|  |  |  |  |         song.subsong[i]->chanName[j]=prevChanName[swappedChannels[j]]; | 
					
						
							|  |  |  |  |         song.subsong[i]->chanShortName[j]=prevChanShortName[swappedChannels[j]]; | 
					
						
							|  |  |  |  |         song.subsong[i]->chanShow[j]=prevChanShow[swappedChannels[j]]; | 
					
						
							| 
									
										
										
										
											2024-02-24 18:12:02 -05:00
										 |  |  |  |         song.subsong[i]->chanShowChanOsc[j]=prevChanShowChanOsc[swappedChannels[j]]; | 
					
						
							| 
									
										
										
										
											2022-08-31 03:52:35 -04:00
										 |  |  |  |         song.subsong[i]->chanCollapse[j]=prevChanCollapse[swappedChannels[j]]; | 
					
						
							| 
									
										
										
										
											2022-08-26 04:03:36 -04:00
										 |  |  |  |       } | 
					
						
							|  |  |  |  |     } | 
					
						
							| 
									
										
										
										
											2022-08-19 17:25:32 -04:00
										 |  |  |  |   } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   DivSystem srcSystem=song.system[src]; | 
					
						
							| 
									
										
										
										
											2023-01-06 17:43:08 -05:00
										 |  |  |  |   float srcVol=song.systemVol[src]; | 
					
						
							|  |  |  |  |   float srcPan=song.systemPan[src]; | 
					
						
							|  |  |  |  |   float srcPanFR=song.systemPanFR[src]; | 
					
						
							| 
									
										
										
										
											2022-08-19 17:25:32 -04:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |   song.system[src]=song.system[dest]; | 
					
						
							|  |  |  |  |   song.system[dest]=srcSystem; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-01-06 17:43:08 -05:00
										 |  |  |  |   song.systemVol[src]=song.systemVol[dest]; | 
					
						
							|  |  |  |  |   song.systemVol[dest]=srcVol; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   song.systemPan[src]=song.systemPan[dest]; | 
					
						
							|  |  |  |  |   song.systemPan[dest]=srcPan; | 
					
						
							| 
									
										
										
										
											2022-08-19 17:25:32 -04:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-01-06 17:43:08 -05:00
										 |  |  |  |   song.systemPanFR[src]=song.systemPanFR[dest]; | 
					
						
							|  |  |  |  |   song.systemPanFR[dest]=srcPanFR; | 
					
						
							| 
									
										
										
										
											2022-08-19 17:25:32 -04:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-30 01:26:54 -04:00
										 |  |  |  |   // I am kinda scared to use std::swap
 | 
					
						
							|  |  |  |  |   DivConfig oldFlags=song.systemFlags[src]; | 
					
						
							|  |  |  |  |   song.systemFlags[src]=song.systemFlags[dest]; | 
					
						
							|  |  |  |  |   song.systemFlags[dest]=oldFlags; | 
					
						
							| 
									
										
										
										
											2022-08-19 17:25:32 -04:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-01-10 15:58:15 -05:00
										 |  |  |  |   // patchbay
 | 
					
						
							|  |  |  |  |   for (unsigned int& i: song.patchbay) { | 
					
						
							| 
									
										
										
										
											2023-01-11 00:10:42 -05:00
										 |  |  |  |     if (((i>>20)&0xfff)==(unsigned int)src) { | 
					
						
							| 
									
										
										
										
											2023-01-10 15:58:15 -05:00
										 |  |  |  |       i=(i&(~0xfff00000))|((unsigned int)dest<<20); | 
					
						
							| 
									
										
										
										
											2023-01-11 00:10:42 -05:00
										 |  |  |  |     } else if (((i>>20)&0xfff)==(unsigned int)dest) { | 
					
						
							| 
									
										
										
										
											2023-01-10 15:58:15 -05:00
										 |  |  |  |       i=(i&(~0xfff00000))|((unsigned int)src<<20); | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  |   } | 
					
						
							| 
									
										
										
										
											2024-02-22 12:48:16 -05:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | bool DivEngine::swapSystem(int src, int dest, bool preserveOrder) { | 
					
						
							|  |  |  |  |   if (src==dest) { | 
					
						
							| 
									
										
										
										
											2024-05-27 05:02:07 -04:00
										 |  |  |  |     lastError=_("source and destination are equal"); | 
					
						
							| 
									
										
										
										
											2024-02-22 12:48:16 -05:00
										 |  |  |  |     return false; | 
					
						
							|  |  |  |  |   } | 
					
						
							|  |  |  |  |   if (src<0 || src>=song.systemLen) { | 
					
						
							| 
									
										
										
										
											2024-05-27 05:02:07 -04:00
										 |  |  |  |     lastError=_("invalid source index"); | 
					
						
							| 
									
										
										
										
											2024-02-22 12:48:16 -05:00
										 |  |  |  |     return false; | 
					
						
							|  |  |  |  |   } | 
					
						
							|  |  |  |  |   if (dest<0 || dest>=song.systemLen) { | 
					
						
							| 
									
										
										
										
											2024-05-27 05:02:07 -04:00
										 |  |  |  |     lastError=_("invalid destination index"); | 
					
						
							| 
									
										
										
										
											2024-02-22 12:48:16 -05:00
										 |  |  |  |     return false; | 
					
						
							|  |  |  |  |   } | 
					
						
							|  |  |  |  |   //int chanCount=chans;
 | 
					
						
							|  |  |  |  |   quitDispatch(); | 
					
						
							|  |  |  |  |   BUSY_BEGIN; | 
					
						
							|  |  |  |  |   saveLock.lock(); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   swapSystemUnsafe(src,dest,preserveOrder); | 
					
						
							| 
									
										
										
										
											2023-01-10 15:58:15 -05:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-19 17:25:32 -04:00
										 |  |  |  |   recalcChans(); | 
					
						
							|  |  |  |  |   saveLock.unlock(); | 
					
						
							|  |  |  |  |   BUSY_END; | 
					
						
							|  |  |  |  |   initDispatch(); | 
					
						
							|  |  |  |  |   BUSY_BEGIN; | 
					
						
							|  |  |  |  |   renderSamples(); | 
					
						
							|  |  |  |  |   reset(); | 
					
						
							|  |  |  |  |   BUSY_END; | 
					
						
							|  |  |  |  |   return true; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-02-01 18:08:19 -05:00
										 |  |  |  | void DivEngine::poke(int sys, unsigned int addr, unsigned short val) { | 
					
						
							|  |  |  |  |   if (sys<0 || sys>=song.systemLen) return; | 
					
						
							| 
									
										
										
										
											2022-03-23 22:38:28 -04:00
										 |  |  |  |   BUSY_BEGIN; | 
					
						
							| 
									
										
										
										
											2022-02-01 18:08:19 -05:00
										 |  |  |  |   disCont[sys].dispatch->poke(addr,val); | 
					
						
							| 
									
										
										
										
											2022-03-23 22:38:28 -04:00
										 |  |  |  |   BUSY_END; | 
					
						
							| 
									
										
										
										
											2022-02-01 18:08:19 -05:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | void DivEngine::poke(int sys, std::vector<DivRegWrite>& wlist) { | 
					
						
							|  |  |  |  |   if (sys<0 || sys>=song.systemLen) return; | 
					
						
							| 
									
										
										
										
											2022-03-23 22:38:28 -04:00
										 |  |  |  |   BUSY_BEGIN; | 
					
						
							| 
									
										
										
										
											2022-02-01 18:08:19 -05:00
										 |  |  |  |   disCont[sys].dispatch->poke(wlist); | 
					
						
							| 
									
										
										
										
											2022-03-23 22:38:28 -04:00
										 |  |  |  |   BUSY_END; | 
					
						
							| 
									
										
										
										
											2022-02-01 18:08:19 -05:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-12-20 23:20:30 -05:00
										 |  |  |  | String DivEngine::getLastError() { | 
					
						
							|  |  |  |  |   return lastError; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-01-29 01:22:32 -05:00
										 |  |  |  | String DivEngine::getWarnings() { | 
					
						
							|  |  |  |  |   return warnings; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-10-02 02:02:01 -04:00
										 |  |  |  | String DivEngine::getPlaybackDebugInfo() { | 
					
						
							|  |  |  |  |   return fmt::sprintf( | 
					
						
							|  |  |  |  |     "curOrder: %d\n" | 
					
						
							|  |  |  |  |     "prevOrder: %d\n" | 
					
						
							|  |  |  |  |     "curRow: %d\n" | 
					
						
							|  |  |  |  |     "prevRow: %d\n" | 
					
						
							|  |  |  |  |     "ticks: %d\n" | 
					
						
							|  |  |  |  |     "subticks: %d\n" | 
					
						
							|  |  |  |  |     "totalLoops: %d\n" | 
					
						
							|  |  |  |  |     "lastLoopPos: %d\n" | 
					
						
							|  |  |  |  |     "nextSpeed: %d\n" | 
					
						
							|  |  |  |  |     "divider: %f\n" | 
					
						
							|  |  |  |  |     "cycles: %d\n" | 
					
						
							|  |  |  |  |     "clockDrift: %f\n" | 
					
						
							| 
									
										
										
										
											2023-05-09 06:05:53 -04:00
										 |  |  |  |     "midiClockCycles: %d\n" | 
					
						
							|  |  |  |  |     "midiClockDrift: %f\n" | 
					
						
							|  |  |  |  |     "midiTimeCycles: %d\n" | 
					
						
							| 
									
										
										
										
											2023-05-10 03:57:59 -04:00
										 |  |  |  |     "midiTimeDrift: %f\n" | 
					
						
							| 
									
										
										
										
											2022-10-02 02:02:01 -04:00
										 |  |  |  |     "changeOrd: %d\n" | 
					
						
							|  |  |  |  |     "changePos: %d\n" | 
					
						
							|  |  |  |  |     "totalSeconds: %d\n" | 
					
						
							|  |  |  |  |     "totalTicks: %d\n" | 
					
						
							|  |  |  |  |     "totalTicksR: %d\n" | 
					
						
							| 
									
										
										
										
											2023-05-09 23:12:14 -04:00
										 |  |  |  |     "curMidiClock: %d\n" | 
					
						
							|  |  |  |  |     "curMidiTime: %d\n" | 
					
						
							| 
									
										
										
										
											2022-10-02 02:02:01 -04:00
										 |  |  |  |     "totalCmds: %d\n" | 
					
						
							|  |  |  |  |     "lastCmds: %d\n" | 
					
						
							|  |  |  |  |     "cmdsPerSecond: %d\n" | 
					
						
							|  |  |  |  |     "globalPitch: %d\n" | 
					
						
							|  |  |  |  |     "extValue: %d\n" | 
					
						
							|  |  |  |  |     "tempoAccum: %d\n" | 
					
						
							| 
									
										
										
										
											2023-01-03 01:09:46 -05:00
										 |  |  |  |     "totalProcessed: %d\n" | 
					
						
							|  |  |  |  |     "bufferPos: %d\n", | 
					
						
							| 
									
										
										
										
											2022-10-02 02:02:01 -04:00
										 |  |  |  |     curOrder,prevOrder,curRow,prevRow,ticks,subticks,totalLoops,lastLoopPos,nextSpeed,divider,cycles,clockDrift, | 
					
						
							| 
									
										
										
										
											2023-05-09 06:05:53 -04:00
										 |  |  |  |     midiClockCycles,midiClockDrift,midiTimeCycles,midiTimeDrift,changeOrd,changePos,totalSeconds,totalTicks, | 
					
						
							| 
									
										
										
										
											2023-05-09 23:12:14 -04:00
										 |  |  |  |     totalTicksR,curMidiClock,curMidiTime,totalCmds,lastCmds,cmdsPerSecond,globalPitch, | 
					
						
							|  |  |  |  |     (int)extValue,(int)tempoAccum,(int)totalProcessed,(int)bufferPos | 
					
						
							| 
									
										
										
										
											2022-10-02 02:02:01 -04:00
										 |  |  |  |   ); | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-21 03:24:06 -04:00
										 |  |  |  | DivInstrument* DivEngine::getIns(int index, DivInstrumentType fallbackType) { | 
					
						
							| 
									
										
										
										
											2022-04-25 19:58:17 -04:00
										 |  |  |  |   if (index==-2 && tempIns!=NULL) { | 
					
						
							|  |  |  |  |     return tempIns; | 
					
						
							|  |  |  |  |   } | 
					
						
							| 
									
										
										
										
											2022-04-21 03:24:06 -04:00
										 |  |  |  |   if (index<0 || index>=song.insLen) { | 
					
						
							|  |  |  |  |     switch (fallbackType) { | 
					
						
							|  |  |  |  |       case DIV_INS_OPLL: | 
					
						
							|  |  |  |  |         return &song.nullInsOPLL; | 
					
						
							|  |  |  |  |         break; | 
					
						
							|  |  |  |  |       case DIV_INS_OPL: | 
					
						
							|  |  |  |  |         return &song.nullInsOPL; | 
					
						
							|  |  |  |  |         break; | 
					
						
							| 
									
										
										
										
											2022-05-31 18:44:52 -04:00
										 |  |  |  |       case DIV_INS_OPL_DRUMS: | 
					
						
							|  |  |  |  |         return &song.nullInsOPLDrums; | 
					
						
							|  |  |  |  |         break; | 
					
						
							| 
									
										
										
										
											2023-10-14 07:10:37 -04:00
										 |  |  |  |       case DIV_INS_ESFM: | 
					
						
							|  |  |  |  |         return &song.nullInsESFM; | 
					
						
							|  |  |  |  |         break; | 
					
						
							| 
									
										
										
										
											2022-04-21 03:24:06 -04:00
										 |  |  |  |       default: | 
					
						
							|  |  |  |  |         break; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  |     return &song.nullIns; | 
					
						
							|  |  |  |  |   } | 
					
						
							| 
									
										
										
										
											2021-05-16 21:49:54 -04:00
										 |  |  |  |   return song.ins[index]; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-27 14:30:37 -04:00
										 |  |  |  | DivWavetable* DivEngine::getWave(int index) { | 
					
						
							| 
									
										
										
										
											2021-06-08 23:21:05 -04:00
										 |  |  |  |   if (index<0 || index>=song.waveLen) { | 
					
						
							|  |  |  |  |     if (song.waveLen>0) { | 
					
						
							|  |  |  |  |       return song.wave[0]; | 
					
						
							|  |  |  |  |     } else { | 
					
						
							|  |  |  |  |       return &song.nullWave; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  |   } | 
					
						
							| 
									
										
										
										
											2021-05-27 14:30:37 -04:00
										 |  |  |  |   return song.wave[index]; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-02-24 22:52:20 -05:00
										 |  |  |  | DivSample* DivEngine::getSample(int index) { | 
					
						
							|  |  |  |  |   if (index<0 || index>=song.sampleLen) return &song.nullSample; | 
					
						
							|  |  |  |  |   return song.sample[index]; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-05-01 13:57:44 -04:00
										 |  |  |  | DivDispatch* DivEngine::getDispatch(int index) { | 
					
						
							|  |  |  |  |   if (index<0 || index>=song.systemLen) return NULL; | 
					
						
							|  |  |  |  |   return disCont[index].dispatch; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-12-07 04:22:36 -05:00
										 |  |  |  | void DivEngine::setLoops(int loops) { | 
					
						
							|  |  |  |  |   remainingLoops=loops; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-01-27 00:29:16 -05:00
										 |  |  |  | DivChannelState* DivEngine::getChanState(int ch) { | 
					
						
							|  |  |  |  |   if (ch<0 || ch>=chans) return NULL; | 
					
						
							|  |  |  |  |   return &chan[ch]; | 
					
						
							|  |  |  |  | } | 
					
						
							| 
									
										
										
										
											2023-08-24 04:25:38 -04:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  | unsigned short DivEngine::getChanPan(int ch) { | 
					
						
							|  |  |  |  |   if (ch<0 || ch>=chans) return 0; | 
					
						
							|  |  |  |  |   return disCont[dispatchOfChan[ch]].dispatch->getPan(dispatchChanOfChan[ch]); | 
					
						
							|  |  |  |  | } | 
					
						
							| 
									
										
										
										
											2022-01-27 00:29:16 -05:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  | void* DivEngine::getDispatchChanState(int ch) { | 
					
						
							|  |  |  |  |   if (ch<0 || ch>=chans) return NULL; | 
					
						
							|  |  |  |  |   return disCont[dispatchOfChan[ch]].dispatch->getChanState(dispatchChanOfChan[ch]); | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-08-14 13:01:16 -04:00
										 |  |  |  | void DivEngine::getChanPaired(int ch, std::vector<DivChannelPair>& ret) { | 
					
						
							|  |  |  |  |   if (ch<0 || ch>=chans) return; | 
					
						
							| 
									
										
										
										
											2024-09-16 04:09:56 -04:00
										 |  |  |  |   disCont[dispatchOfChan[ch]].dispatch->getPaired(dispatchChanOfChan[ch],ret); | 
					
						
							| 
									
										
										
										
											2023-10-07 21:35:25 -04:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-10-26 20:03:45 -04:00
										 |  |  |  | DivChannelModeHints DivEngine::getChanModeHints(int ch) { | 
					
						
							|  |  |  |  |   if (ch<0 || ch>=chans) return DivChannelModeHints(); | 
					
						
							|  |  |  |  |   return disCont[dispatchOfChan[ch]].dispatch->getModeHints(dispatchChanOfChan[ch]); | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-02-22 04:01:57 -05:00
										 |  |  |  | unsigned char* DivEngine::getRegisterPool(int sys, int& size, int& depth) { | 
					
						
							| 
									
										
										
										
											2022-02-21 22:31:27 -05:00
										 |  |  |  |   if (sys<0 || sys>=song.systemLen) return NULL; | 
					
						
							|  |  |  |  |   if (disCont[sys].dispatch==NULL) return NULL; | 
					
						
							|  |  |  |  |   size=disCont[sys].dispatch->getRegisterPoolSize(); | 
					
						
							| 
									
										
										
										
											2022-02-22 04:01:57 -05:00
										 |  |  |  |   depth=disCont[sys].dispatch->getRegisterPoolDepth(); | 
					
						
							| 
									
										
										
										
											2022-02-21 22:31:27 -05:00
										 |  |  |  |   return disCont[sys].dispatch->getRegisterPool(); | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-30 04:58:30 -04:00
										 |  |  |  | DivMacroInt* DivEngine::getMacroInt(int chan) { | 
					
						
							|  |  |  |  |   if (chan<0 || chan>=chans) return NULL; | 
					
						
							|  |  |  |  |   return disCont[dispatchOfChan[chan]].dispatch->getChanMacroInt(dispatchChanOfChan[chan]); | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-03-19 04:12:08 -04:00
										 |  |  |  | DivSamplePos DivEngine::getSamplePos(int chan) { | 
					
						
							|  |  |  |  |   if (chan<0 || chan>=chans) return DivSamplePos(); | 
					
						
							|  |  |  |  |   return disCont[dispatchOfChan[chan]].dispatch->getSamplePos(dispatchChanOfChan[chan]); | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-30 04:58:30 -04:00
										 |  |  |  | DivDispatchOscBuffer* DivEngine::getOscBuffer(int chan) { | 
					
						
							|  |  |  |  |   if (chan<0 || chan>=chans) return NULL; | 
					
						
							|  |  |  |  |   return disCont[dispatchOfChan[chan]].dispatch->getOscBuffer(dispatchChanOfChan[chan]); | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-02-16 16:11:15 -05:00
										 |  |  |  | void DivEngine::enableCommandStream(bool enable) { | 
					
						
							|  |  |  |  |   cmdStreamEnabled=enable; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | void DivEngine::getCommandStream(std::vector<DivCommand>& where) { | 
					
						
							| 
									
										
										
										
											2022-03-23 22:38:28 -04:00
										 |  |  |  |   BUSY_BEGIN; | 
					
						
							| 
									
										
										
										
											2022-02-16 16:11:15 -05:00
										 |  |  |  |   where.clear(); | 
					
						
							| 
									
										
										
										
											2023-08-24 12:08:06 -04:00
										 |  |  |  |   where.reserve(cmdStream.size()); | 
					
						
							| 
									
										
										
										
											2022-02-16 16:11:15 -05:00
										 |  |  |  |   for (DivCommand& i: cmdStream) { | 
					
						
							|  |  |  |  |     where.push_back(i); | 
					
						
							|  |  |  |  |   } | 
					
						
							|  |  |  |  |   cmdStream.clear(); | 
					
						
							| 
									
										
										
										
											2022-03-23 22:38:28 -04:00
										 |  |  |  |   BUSY_END; | 
					
						
							| 
									
										
										
										
											2022-02-16 16:11:15 -05:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-02-06 00:07:35 -05:00
										 |  |  |  | void DivEngine::playSub(bool preserveDrift, int goalRow) { | 
					
						
							| 
									
										
										
										
											2022-10-22 04:31:03 -04:00
										 |  |  |  |   logV("playSub() called"); | 
					
						
							| 
									
										
										
										
											2022-04-10 18:24:41 -04:00
										 |  |  |  |   std::chrono::high_resolution_clock::time_point timeStart=std::chrono::high_resolution_clock::now(); | 
					
						
							| 
									
										
										
										
											2022-02-20 23:03:42 -05:00
										 |  |  |  |   for (int i=0; i<song.systemLen; i++) disCont[i].dispatch->setSkipRegisterWrites(false); | 
					
						
							| 
									
										
										
										
											2021-12-11 16:51:34 -05:00
										 |  |  |  |   reset(); | 
					
						
							| 
									
										
										
										
											2022-10-22 04:31:03 -04:00
										 |  |  |  |   if (preserveDrift && curOrder==0) { | 
					
						
							|  |  |  |  |     logV("preserveDrift && curOrder is true"); | 
					
						
							|  |  |  |  |     return; | 
					
						
							|  |  |  |  |   } | 
					
						
							| 
									
										
										
										
											2021-12-21 17:42:27 -05:00
										 |  |  |  |   bool oldRepeatPattern=repeatPattern; | 
					
						
							|  |  |  |  |   repeatPattern=false; | 
					
						
							| 
									
										
										
										
											2021-12-21 01:29:07 -05:00
										 |  |  |  |   int goal=curOrder; | 
					
						
							|  |  |  |  |   curOrder=0; | 
					
						
							| 
									
										
										
										
											2021-12-11 03:34:43 -05:00
										 |  |  |  |   curRow=0; | 
					
						
							| 
									
										
										
										
											2022-06-06 02:05:06 -04:00
										 |  |  |  |   prevOrder=0; | 
					
						
							|  |  |  |  |   prevRow=0; | 
					
						
							| 
									
										
										
										
											2022-02-06 00:42:07 -05:00
										 |  |  |  |   stepPlay=0; | 
					
						
							| 
									
										
										
										
											2023-05-24 05:08:28 -04:00
										 |  |  |  |   if (curSubSong!=NULL) curSubSong->arpLen=1; | 
					
						
							| 
									
										
										
										
											2023-05-09 06:05:53 -04:00
										 |  |  |  |   int prevDrift, prevMidiClockDrift, prevMidiTimeDrift; | 
					
						
							| 
									
										
										
										
											2022-01-12 17:45:07 -05:00
										 |  |  |  |   prevDrift=clockDrift; | 
					
						
							| 
									
										
										
										
											2023-05-09 06:05:53 -04:00
										 |  |  |  |   prevMidiClockDrift=midiClockDrift; | 
					
						
							|  |  |  |  |   prevMidiTimeDrift=midiTimeDrift; | 
					
						
							| 
									
										
										
										
											2022-01-12 17:45:07 -05:00
										 |  |  |  |   clockDrift=0; | 
					
						
							|  |  |  |  |   cycles=0; | 
					
						
							| 
									
										
										
										
											2023-05-09 06:05:53 -04:00
										 |  |  |  |   midiClockCycles=0; | 
					
						
							|  |  |  |  |   midiClockDrift=0; | 
					
						
							|  |  |  |  |   midiTimeCycles=0; | 
					
						
							|  |  |  |  |   midiTimeDrift=0; | 
					
						
							| 
									
										
										
										
											2022-10-22 04:31:03 -04:00
										 |  |  |  |   if (!preserveDrift) { | 
					
						
							| 
									
										
										
										
											2021-12-21 02:02:25 -05:00
										 |  |  |  |     ticks=1; | 
					
						
							| 
									
										
										
										
											2022-05-18 01:05:25 -04:00
										 |  |  |  |     tempoAccum=0; | 
					
						
							| 
									
										
										
										
											2021-12-21 02:30:09 -05:00
										 |  |  |  |     totalTicks=0; | 
					
						
							| 
									
										
										
										
											2024-12-09 13:56:20 -05:00
										 |  |  |  |     totalTicksOff=0; | 
					
						
							| 
									
										
										
										
											2022-01-12 02:45:26 -05:00
										 |  |  |  |     totalSeconds=0; | 
					
						
							|  |  |  |  |     totalTicksR=0; | 
					
						
							| 
									
										
										
										
											2023-05-09 23:12:14 -04:00
										 |  |  |  |     curMidiClock=0; | 
					
						
							|  |  |  |  |     curMidiTime=0; | 
					
						
							| 
									
										
										
										
											2023-05-10 03:57:59 -04:00
										 |  |  |  |     curMidiTimeCode=0; | 
					
						
							|  |  |  |  |     curMidiTimePiece=0; | 
					
						
							| 
									
										
										
										
											2022-06-06 04:05:55 -04:00
										 |  |  |  |     totalLoops=0; | 
					
						
							|  |  |  |  |     lastLoopPos=-1; | 
					
						
							| 
									
										
										
										
											2021-12-21 02:02:25 -05:00
										 |  |  |  |   } | 
					
						
							| 
									
										
										
										
											2022-10-22 04:31:03 -04:00
										 |  |  |  |   endOfSong=false; | 
					
						
							| 
									
										
										
										
											2023-02-05 02:56:39 -05:00
										 |  |  |  |   // whaaaaa?
 | 
					
						
							|  |  |  |  |   curSpeed=0; | 
					
						
							| 
									
										
										
										
											2021-12-11 16:51:34 -05:00
										 |  |  |  |   playing=true; | 
					
						
							| 
									
										
										
										
											2022-03-31 04:33:05 -04:00
										 |  |  |  |   skipping=true; | 
					
						
							| 
									
										
										
										
											2022-09-10 02:39:42 -04:00
										 |  |  |  |   memset(walked,0,8192); | 
					
						
							| 
									
										
										
										
											2022-01-08 16:03:32 -05:00
										 |  |  |  |   for (int i=0; i<song.systemLen; i++) disCont[i].dispatch->setSkipRegisterWrites(true); | 
					
						
							| 
									
										
										
										
											2022-10-02 02:02:01 -04:00
										 |  |  |  |   logV("goal: %d goalRow: %d",goal,goalRow); | 
					
						
							| 
									
										
										
										
											2022-02-17 03:20:08 -05:00
										 |  |  |  |   while (playing && curOrder<goal) { | 
					
						
							| 
									
										
										
										
											2022-03-31 04:33:05 -04:00
										 |  |  |  |     if (nextTick(preserveDrift)) { | 
					
						
							|  |  |  |  |       skipping=false; | 
					
						
							| 
									
										
										
										
											2023-12-10 17:52:32 -05:00
										 |  |  |  |       cmdStream.clear(); | 
					
						
							|  |  |  |  |       for (int i=0; i<song.systemLen; i++) disCont[i].dispatch->setSkipRegisterWrites(false); | 
					
						
							|  |  |  |  |       if (goal>0 || goalRow>0) { | 
					
						
							|  |  |  |  |         for (int i=0; i<song.systemLen; i++) disCont[i].dispatch->forceIns(); | 
					
						
							|  |  |  |  |       } | 
					
						
							| 
									
										
										
										
											2022-03-31 04:33:05 -04:00
										 |  |  |  |       return; | 
					
						
							|  |  |  |  |     } | 
					
						
							| 
									
										
										
										
											2023-05-10 04:30:05 -04:00
										 |  |  |  |     if (!preserveDrift) { | 
					
						
							|  |  |  |  |       runMidiClock(cycles); | 
					
						
							|  |  |  |  |       runMidiTime(cycles); | 
					
						
							|  |  |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-12-21 01:29:07 -05:00
										 |  |  |  |   } | 
					
						
							| 
									
										
										
										
											2022-02-06 00:07:35 -05:00
										 |  |  |  |   int oldOrder=curOrder; | 
					
						
							| 
									
										
										
										
											2022-08-31 04:05:06 -04:00
										 |  |  |  |   while (playing && (curRow<goalRow || ticks>1)) { | 
					
						
							| 
									
										
										
										
											2022-03-31 04:33:05 -04:00
										 |  |  |  |     if (nextTick(preserveDrift)) { | 
					
						
							|  |  |  |  |       skipping=false; | 
					
						
							| 
									
										
										
										
											2023-12-10 17:52:32 -05:00
										 |  |  |  |       cmdStream.clear(); | 
					
						
							|  |  |  |  |       for (int i=0; i<song.systemLen; i++) disCont[i].dispatch->setSkipRegisterWrites(false); | 
					
						
							|  |  |  |  |       if (goal>0 || goalRow>0) { | 
					
						
							|  |  |  |  |         for (int i=0; i<song.systemLen; i++) disCont[i].dispatch->forceIns(); | 
					
						
							|  |  |  |  |       } | 
					
						
							| 
									
										
										
										
											2022-03-31 04:33:05 -04:00
										 |  |  |  |       return; | 
					
						
							|  |  |  |  |     } | 
					
						
							| 
									
										
										
										
											2023-05-10 04:30:05 -04:00
										 |  |  |  |     if (!preserveDrift) { | 
					
						
							|  |  |  |  |       runMidiClock(cycles); | 
					
						
							|  |  |  |  |       runMidiTime(cycles); | 
					
						
							|  |  |  |  |     } | 
					
						
							| 
									
										
										
										
											2022-02-06 00:07:35 -05:00
										 |  |  |  |     if (oldOrder!=curOrder) break; | 
					
						
							| 
									
										
										
										
											2024-03-15 15:56:55 -04:00
										 |  |  |  |     if (ticks-((tempoAccum+virtualTempoN)/MAX(1,virtualTempoD))<1 && curRow>=goalRow) break; | 
					
						
							| 
									
										
										
										
											2022-02-06 00:07:35 -05:00
										 |  |  |  |   } | 
					
						
							| 
									
										
										
										
											2022-01-08 16:03:32 -05:00
										 |  |  |  |   for (int i=0; i<song.systemLen; i++) disCont[i].dispatch->setSkipRegisterWrites(false); | 
					
						
							| 
									
										
										
										
											2022-02-06 00:07:35 -05:00
										 |  |  |  |   if (goal>0 || goalRow>0) { | 
					
						
							| 
									
										
										
										
											2022-01-08 16:03:32 -05:00
										 |  |  |  |     for (int i=0; i<song.systemLen; i++) disCont[i].dispatch->forceIns(); | 
					
						
							| 
									
										
										
										
											2021-12-21 16:02:31 -05:00
										 |  |  |  |   } | 
					
						
							| 
									
										
										
										
											2022-02-14 02:43:56 -05:00
										 |  |  |  |   for (int i=0; i<chans; i++) { | 
					
						
							|  |  |  |  |     chan[i].cut=-1; | 
					
						
							| 
									
										
										
										
											2024-03-15 14:45:57 -04:00
										 |  |  |  |     chan[i].cutType=0; | 
					
						
							| 
									
										
										
										
											2022-02-14 02:43:56 -05:00
										 |  |  |  |   } | 
					
						
							| 
									
										
										
										
											2021-12-21 17:42:27 -05:00
										 |  |  |  |   repeatPattern=oldRepeatPattern; | 
					
						
							| 
									
										
										
										
											2022-01-12 17:45:07 -05:00
										 |  |  |  |   if (preserveDrift) { | 
					
						
							|  |  |  |  |     clockDrift=prevDrift; | 
					
						
							| 
									
										
										
										
											2023-05-09 06:05:53 -04:00
										 |  |  |  |     midiClockDrift=prevMidiClockDrift; | 
					
						
							|  |  |  |  |     midiTimeDrift=prevMidiTimeDrift; | 
					
						
							| 
									
										
										
										
											2022-01-12 17:45:07 -05:00
										 |  |  |  |   } else { | 
					
						
							|  |  |  |  |     clockDrift=0; | 
					
						
							|  |  |  |  |     cycles=0; | 
					
						
							| 
									
										
										
										
											2023-05-09 06:05:53 -04:00
										 |  |  |  |     midiClockCycles=0; | 
					
						
							|  |  |  |  |     midiClockDrift=0; | 
					
						
							|  |  |  |  |     midiTimeCycles=0; | 
					
						
							|  |  |  |  |     midiTimeDrift=0; | 
					
						
							| 
									
										
										
										
											2023-05-10 04:30:05 -04:00
										 |  |  |  |     if (curMidiTime>0) { | 
					
						
							|  |  |  |  |       curMidiTime--; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  |     if (curMidiClock>0) { | 
					
						
							|  |  |  |  |       curMidiClock--; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  |     curMidiTimePiece=0; | 
					
						
							| 
									
										
										
										
											2021-12-21 02:02:25 -05:00
										 |  |  |  |   } | 
					
						
							|  |  |  |  |   if (!preserveDrift) { | 
					
						
							|  |  |  |  |     ticks=1; | 
					
						
							| 
									
										
										
										
											2022-04-15 23:22:47 -04:00
										 |  |  |  |     subticks=1; | 
					
						
							| 
									
										
										
										
											2022-06-06 02:05:06 -04:00
										 |  |  |  |     prevOrder=curOrder; | 
					
						
							|  |  |  |  |     prevRow=curRow; | 
					
						
							| 
									
										
										
										
											2025-03-11 05:15:14 -04:00
										 |  |  |  |     prevSpeed=nextSpeed; | 
					
						
							| 
									
										
										
										
											2022-10-02 02:02:01 -04:00
										 |  |  |  |     tempoAccum=0; | 
					
						
							| 
									
										
										
										
											2021-12-21 02:02:25 -05:00
										 |  |  |  |   } | 
					
						
							| 
									
										
										
										
											2022-03-31 04:33:05 -04:00
										 |  |  |  |   skipping=false; | 
					
						
							| 
									
										
										
										
											2022-02-16 16:11:15 -05:00
										 |  |  |  |   cmdStream.clear(); | 
					
						
							| 
									
										
										
										
											2022-04-10 18:24:41 -04:00
										 |  |  |  |   std::chrono::high_resolution_clock::time_point timeEnd=std::chrono::high_resolution_clock::now(); | 
					
						
							| 
									
										
										
										
											2022-04-10 23:16:42 -04:00
										 |  |  |  |   logV("playSub() took %dµs",std::chrono::duration_cast<std::chrono::microseconds>(timeEnd-timeStart).count()); | 
					
						
							| 
									
										
										
										
											2021-12-21 01:29:07 -05:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-03-24 00:19:16 -04:00
										 |  |  |  | /*
 | 
					
						
							| 
									
										
										
										
											2022-01-28 00:55:51 -05:00
										 |  |  |  | int DivEngine::calcBaseFreq(double clock, double divider, int note, bool period) { | 
					
						
							|  |  |  |  |   double base=(period?(song.tuning*0.0625):song.tuning)*pow(2.0,(float)(note+3)/12.0); | 
					
						
							|  |  |  |  |   return period? | 
					
						
							|  |  |  |  |          round((clock/base)/divider): | 
					
						
							|  |  |  |  |          base*(divider/clock); | 
					
						
							| 
									
										
										
										
											2022-03-24 00:19:16 -04:00
										 |  |  |  | }*/ | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | double DivEngine::calcBaseFreq(double clock, double divider, int note, bool period) { | 
					
						
							| 
									
										
										
										
											2022-05-10 04:51:18 -04:00
										 |  |  |  |   if (song.linearPitch==2) { // full linear
 | 
					
						
							|  |  |  |  |     return (note<<7); | 
					
						
							|  |  |  |  |   } | 
					
						
							| 
									
										
										
										
											2022-03-24 00:19:16 -04:00
										 |  |  |  |   double base=(period?(song.tuning*0.0625):song.tuning)*pow(2.0,(float)(note+3)/12.0); | 
					
						
							|  |  |  |  |   return period? | 
					
						
							|  |  |  |  |          (clock/base)/divider: | 
					
						
							|  |  |  |  |          base*(divider/clock); | 
					
						
							| 
									
										
										
										
											2022-01-28 00:55:51 -05:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-05-10 17:22:40 -04:00
										 |  |  |  | #define CONVERT_FNUM_BLOCK(bf,bits,note) \
 | 
					
						
							|  |  |  |  |   double tuning=song.tuning; \ | 
					
						
							|  |  |  |  |   if (tuning<400.0) tuning=400.0; \ | 
					
						
							|  |  |  |  |   if (tuning>500.0) tuning=500.0; \ | 
					
						
							|  |  |  |  |   int boundaryBottom=tuning*pow(2.0,0.25)*(divider/clock); \ | 
					
						
							|  |  |  |  |   int boundaryTop=2.0*tuning*pow(2.0,0.25)*(divider/clock); \ | 
					
						
							| 
									
										
										
										
											2022-12-26 16:09:38 -05:00
										 |  |  |  |   while (boundaryTop>((1<<bits)-1)) { \ | 
					
						
							|  |  |  |  |     boundaryTop>>=1; \ | 
					
						
							|  |  |  |  |     boundaryBottom>>=1; \ | 
					
						
							|  |  |  |  |   } \ | 
					
						
							| 
									
										
										
										
											2022-05-10 17:22:40 -04:00
										 |  |  |  |   int block=(note)/12; \ | 
					
						
							|  |  |  |  |   if (block<0) block=0; \ | 
					
						
							|  |  |  |  |   if (block>7) block=7; \ | 
					
						
							|  |  |  |  |   bf>>=block; \ | 
					
						
							|  |  |  |  |   if (bf<0) bf=0; \ | 
					
						
							|  |  |  |  |   /* octave boundaries */ \ | 
					
						
							|  |  |  |  |   while (bf>0 && bf<boundaryBottom && block>0) { \ | 
					
						
							|  |  |  |  |     bf<<=1; \ | 
					
						
							|  |  |  |  |     block--; \ | 
					
						
							|  |  |  |  |   } \ | 
					
						
							|  |  |  |  |   if (bf>boundaryTop) { \ | 
					
						
							|  |  |  |  |     while (block<7 && bf>boundaryTop) { \ | 
					
						
							|  |  |  |  |       bf>>=1; \ | 
					
						
							|  |  |  |  |       block++; \ | 
					
						
							|  |  |  |  |     } \ | 
					
						
							|  |  |  |  |     if (bf>((1<<bits)-1)) { \ | 
					
						
							|  |  |  |  |       bf=(1<<bits)-1; \ | 
					
						
							|  |  |  |  |     } \ | 
					
						
							|  |  |  |  |   } \ | 
					
						
							|  |  |  |  |   /* logV("f-num: %d block: %d",bf,block); */ \ | 
					
						
							| 
									
										
										
										
											2022-04-24 23:45:59 -04:00
										 |  |  |  |   return bf|(block<<bits); | 
					
						
							| 
									
										
										
										
											2022-05-10 17:22:40 -04:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-18 03:48:10 -04:00
										 |  |  |  | #define CONVERT_FNUM_FIXEDBLOCK(bf,bits,block) \
 | 
					
						
							|  |  |  |  |   bf>>=(block); \ | 
					
						
							|  |  |  |  |   if (bf<0) bf=0; \ | 
					
						
							|  |  |  |  |   if (bf>((1<<(bits))-1)) { \ | 
					
						
							|  |  |  |  |     bf=(1<<(bits))-1; \ | 
					
						
							|  |  |  |  |   } \ | 
					
						
							|  |  |  |  |   return bf|((block)<<(bits)); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | int DivEngine::calcBaseFreqFNumBlock(double clock, double divider, int note, int bits, int fixedBlock) { | 
					
						
							| 
									
										
										
										
											2022-05-10 04:51:18 -04:00
										 |  |  |  |   if (song.linearPitch==2) { // full linear
 | 
					
						
							|  |  |  |  |     return (note<<7); | 
					
						
							|  |  |  |  |   } | 
					
						
							| 
									
										
										
										
											2022-04-24 23:45:59 -04:00
										 |  |  |  |   int bf=calcBaseFreq(clock,divider,note,false); | 
					
						
							| 
									
										
										
										
											2024-09-18 03:48:10 -04:00
										 |  |  |  |   if (fixedBlock>0) { | 
					
						
							|  |  |  |  |     CONVERT_FNUM_FIXEDBLOCK(bf,bits,fixedBlock-1); | 
					
						
							|  |  |  |  |   } else { | 
					
						
							|  |  |  |  |     CONVERT_FNUM_BLOCK(bf,bits,note); | 
					
						
							|  |  |  |  |   } | 
					
						
							| 
									
										
										
										
											2022-04-24 23:45:59 -04:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-18 03:48:10 -04:00
										 |  |  |  | int DivEngine::calcFreq(int base, int pitch, int arp, bool arpFixed, bool period, int octave, int pitch2, double clock, double divider, int blockBits, int fixedBlock) { | 
					
						
							| 
									
										
										
										
											2022-05-10 04:51:18 -04:00
										 |  |  |  |   if (song.linearPitch==2) { | 
					
						
							|  |  |  |  |     // do frequency calculation here
 | 
					
						
							| 
									
										
										
										
											2022-05-10 17:22:40 -04:00
										 |  |  |  |     int nbase=base+pitch+pitch2; | 
					
						
							| 
									
										
										
										
											2022-12-17 02:07:24 -05:00
										 |  |  |  |     if (!song.oldArpStrategy) { | 
					
						
							|  |  |  |  |       if (arpFixed) { | 
					
						
							|  |  |  |  |         nbase=(arp<<7)+pitch+pitch2; | 
					
						
							|  |  |  |  |       } else { | 
					
						
							|  |  |  |  |         nbase+=arp<<7; | 
					
						
							|  |  |  |  |       } | 
					
						
							|  |  |  |  |     } | 
					
						
							| 
									
										
										
										
											2022-05-10 17:22:40 -04:00
										 |  |  |  |     double fbase=(period?(song.tuning*0.0625):song.tuning)*pow(2.0,(float)(nbase+384)/(128.0*12.0)); | 
					
						
							|  |  |  |  |     int bf=period? | 
					
						
							| 
									
										
										
										
											2022-05-11 00:42:24 -04:00
										 |  |  |  |            round((clock/fbase)/divider): | 
					
						
							|  |  |  |  |            round(fbase*(divider/clock)); | 
					
						
							| 
									
										
										
										
											2022-05-10 17:22:40 -04:00
										 |  |  |  |     if (blockBits>0) { | 
					
						
							| 
									
										
										
										
											2024-09-18 03:48:10 -04:00
										 |  |  |  |       if (fixedBlock>0) { | 
					
						
							|  |  |  |  |         CONVERT_FNUM_FIXEDBLOCK(bf,blockBits,fixedBlock-1); | 
					
						
							|  |  |  |  |       } else { | 
					
						
							|  |  |  |  |         CONVERT_FNUM_BLOCK(bf,blockBits,nbase>>7); | 
					
						
							|  |  |  |  |       } | 
					
						
							| 
									
										
										
										
											2022-05-10 17:22:40 -04:00
										 |  |  |  |     } else { | 
					
						
							|  |  |  |  |       return bf; | 
					
						
							|  |  |  |  |     } | 
					
						
							| 
									
										
										
										
											2022-05-10 04:51:18 -04:00
										 |  |  |  |   } | 
					
						
							|  |  |  |  |   if (song.linearPitch==1) { | 
					
						
							| 
									
										
										
										
											2022-03-25 04:41:43 -04:00
										 |  |  |  |     // global pitch multiplier
 | 
					
						
							|  |  |  |  |     int whatTheFuck=(1024+(globalPitch<<6)-(globalPitch<0?globalPitch-6:0)); | 
					
						
							|  |  |  |  |     if (whatTheFuck<1) whatTheFuck=1; // avoids division by zero but please kill me
 | 
					
						
							| 
									
										
										
										
											2022-04-28 01:26:21 -04:00
										 |  |  |  |     if (song.pitchMacroIsLinear) { | 
					
						
							|  |  |  |  |       pitch+=pitch2; | 
					
						
							|  |  |  |  |     } | 
					
						
							| 
									
										
										
										
											2022-03-25 03:52:41 -04:00
										 |  |  |  |     pitch+=2048; | 
					
						
							|  |  |  |  |     if (pitch<0) pitch=0; | 
					
						
							|  |  |  |  |     if (pitch>4095) pitch=4095; | 
					
						
							| 
									
										
										
										
											2022-04-28 01:26:21 -04:00
										 |  |  |  |     int ret=period? | 
					
						
							|  |  |  |  |               ((base*(reversePitchTable[pitch]))/whatTheFuck): | 
					
						
							|  |  |  |  |               (((base*(pitchTable[pitch]))>>10)*whatTheFuck)/1024; | 
					
						
							|  |  |  |  |     if (!song.pitchMacroIsLinear) { | 
					
						
							|  |  |  |  |       ret+=period?(-pitch2):pitch2; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  |     return ret; | 
					
						
							| 
									
										
										
										
											2022-02-03 00:52:50 -05:00
										 |  |  |  |   } | 
					
						
							| 
									
										
										
										
											2021-12-28 00:51:38 -05:00
										 |  |  |  |   return period? | 
					
						
							| 
									
										
										
										
											2022-04-28 01:26:21 -04:00
										 |  |  |  |            base-pitch-pitch2: | 
					
						
							|  |  |  |  |            base+((pitch*octave)>>1)+pitch2; | 
					
						
							| 
									
										
										
										
											2021-12-28 00:51:38 -05:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-22 16:59:45 -04:00
										 |  |  |  | int DivEngine::calcArp(int note, int arp, int offset) { | 
					
						
							| 
									
										
										
										
											2022-08-22 20:09:08 -04:00
										 |  |  |  |   if (arp<0) { | 
					
						
							|  |  |  |  |     if (!(arp&0x40000000)) return (arp|0x40000000)+offset; | 
					
						
							|  |  |  |  |   } else { | 
					
						
							|  |  |  |  |     if (arp&0x40000000) return (arp&(~0x40000000))+offset; | 
					
						
							|  |  |  |  |   } | 
					
						
							| 
									
										
										
										
											2022-08-22 16:59:45 -04:00
										 |  |  |  |   return note+arp; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-29 00:58:11 -04:00
										 |  |  |  | int DivEngine::convertPanSplitToLinear(unsigned int val, unsigned char bits, int range) { | 
					
						
							|  |  |  |  |   int panL=val>>bits; | 
					
						
							|  |  |  |  |   int panR=val&((1<<bits)-1); | 
					
						
							|  |  |  |  |   int diff=panR-panL; | 
					
						
							|  |  |  |  |   float pan=0.5f; | 
					
						
							|  |  |  |  |   if (diff!=0) { | 
					
						
							|  |  |  |  |     pan=(1.0f+((float)diff/(float)MAX(panL,panR)))*0.5f; | 
					
						
							|  |  |  |  |   } | 
					
						
							|  |  |  |  |   return pan*range; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-30 00:41:14 -04:00
										 |  |  |  | int DivEngine::convertPanSplitToLinearLR(unsigned char left, unsigned char right, int range) { | 
					
						
							|  |  |  |  |   return convertPanSplitToLinear((left<<8)|right,8,range); | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-29 00:58:11 -04:00
										 |  |  |  | unsigned int DivEngine::convertPanLinearToSplit(int val, unsigned char bits, int range) { | 
					
						
							|  |  |  |  |   if (val<0) val=0; | 
					
						
							|  |  |  |  |   if (val>range) val=range; | 
					
						
							|  |  |  |  |   int maxV=(1<<bits)-1; | 
					
						
							|  |  |  |  |   int panL=(((range-val)*maxV*2))/range; | 
					
						
							|  |  |  |  |   int panR=((val)*maxV*2)/range; | 
					
						
							|  |  |  |  |   if (panL>maxV) panL=maxV; | 
					
						
							|  |  |  |  |   if (panR>maxV) panR=maxV; | 
					
						
							|  |  |  |  |   return (panL<<bits)|panR; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-09-09 18:12:49 -04:00
										 |  |  |  | bool DivEngine::play() { | 
					
						
							| 
									
										
										
										
											2022-03-23 22:38:28 -04:00
										 |  |  |  |   BUSY_BEGIN_SOFT; | 
					
						
							| 
									
										
										
										
											2022-06-06 02:05:06 -04:00
										 |  |  |  |   curOrder=prevOrder; | 
					
						
							| 
									
										
										
										
											2022-02-06 16:29:30 -05:00
										 |  |  |  |   sPreview.sample=-1; | 
					
						
							|  |  |  |  |   sPreview.wave=-1; | 
					
						
							|  |  |  |  |   sPreview.pos=0; | 
					
						
							| 
									
										
										
										
											2022-08-11 09:21:54 -04:00
										 |  |  |  |   sPreview.dir=false; | 
					
						
							| 
									
										
										
										
											2022-09-29 01:27:40 -04:00
										 |  |  |  |   shallStop=false; | 
					
						
							| 
									
										
										
										
											2022-02-06 00:42:07 -05:00
										 |  |  |  |   if (stepPlay==0) { | 
					
						
							|  |  |  |  |     freelance=false; | 
					
						
							|  |  |  |  |     playSub(false); | 
					
						
							|  |  |  |  |   } else { | 
					
						
							|  |  |  |  |     stepPlay=0; | 
					
						
							|  |  |  |  |   } | 
					
						
							| 
									
										
										
										
											2022-02-10 03:15:39 -05:00
										 |  |  |  |   for (int i=0; i<DIV_MAX_CHANS; i++) { | 
					
						
							|  |  |  |  |     keyHit[i]=false; | 
					
						
							|  |  |  |  |   } | 
					
						
							| 
									
										
										
										
											2023-05-10 03:57:59 -04:00
										 |  |  |  |   curMidiTimePiece=0; | 
					
						
							| 
									
										
										
										
											2022-04-09 02:50:44 -04:00
										 |  |  |  |   if (output) if (!skipping && output->midiOut!=NULL) { | 
					
						
							| 
									
										
										
										
											2023-05-10 03:57:59 -04:00
										 |  |  |  |     if (midiOutClock) { | 
					
						
							|  |  |  |  |       output->midiOut->send(TAMidiMessage(TA_MIDI_POSITION,(curMidiClock>>7)&0x7f,curMidiClock&0x7f)); | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  |     if (midiOutTime) { | 
					
						
							|  |  |  |  |       TAMidiMessage msg; | 
					
						
							|  |  |  |  |       msg.type=TA_MIDI_SYSEX; | 
					
						
							|  |  |  |  |       msg.sysExData.reset(new unsigned char[10],std::default_delete<unsigned char[]>()); | 
					
						
							|  |  |  |  |       msg.sysExLen=10; | 
					
						
							|  |  |  |  |       unsigned char* msgData=msg.sysExData.get(); | 
					
						
							|  |  |  |  |       int actualTime=curMidiTime; | 
					
						
							|  |  |  |  |       int timeRate=midiOutTimeRate; | 
					
						
							|  |  |  |  |       int drop=0; | 
					
						
							|  |  |  |  |       if (timeRate<1 || timeRate>4) { | 
					
						
							|  |  |  |  |         if (curSubSong->hz>=47.98 && curSubSong->hz<=48.02) { | 
					
						
							|  |  |  |  |           timeRate=1; | 
					
						
							|  |  |  |  |         } else if (curSubSong->hz>=49.98 && curSubSong->hz<=50.02) { | 
					
						
							|  |  |  |  |           timeRate=2; | 
					
						
							|  |  |  |  |         } else if (curSubSong->hz>=59.9 && curSubSong->hz<=60.11) { | 
					
						
							|  |  |  |  |           timeRate=4; | 
					
						
							|  |  |  |  |         } else { | 
					
						
							|  |  |  |  |           timeRate=4; | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  |       } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |       switch (timeRate) { | 
					
						
							|  |  |  |  |         case 1: // 24
 | 
					
						
							|  |  |  |  |           msgData[5]=(actualTime/(60*60*24))%24; | 
					
						
							|  |  |  |  |           msgData[6]=(actualTime/(60*24))%60; | 
					
						
							|  |  |  |  |           msgData[7]=(actualTime/24)%60; | 
					
						
							|  |  |  |  |           msgData[8]=actualTime%24; | 
					
						
							|  |  |  |  |           break; | 
					
						
							|  |  |  |  |         case 2: // 25
 | 
					
						
							|  |  |  |  |           msgData[5]=(actualTime/(60*60*25))%24; | 
					
						
							|  |  |  |  |           msgData[6]=(actualTime/(60*25))%60; | 
					
						
							|  |  |  |  |           msgData[7]=(actualTime/25)%60; | 
					
						
							|  |  |  |  |           msgData[8]=actualTime%25; | 
					
						
							|  |  |  |  |           break; | 
					
						
							|  |  |  |  |         case 3: // 29.97 (NTSC drop)
 | 
					
						
							|  |  |  |  |           // drop
 | 
					
						
							|  |  |  |  |           drop=((actualTime/(30*60))-(actualTime/(30*600)))*2; | 
					
						
							|  |  |  |  |           actualTime+=drop; | 
					
						
							| 
									
										
										
										
											2023-05-10 04:30:05 -04:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-05-10 03:57:59 -04:00
										 |  |  |  |           msgData[5]=(actualTime/(60*60*30))%24; | 
					
						
							|  |  |  |  |           msgData[6]=(actualTime/(60*30))%60; | 
					
						
							|  |  |  |  |           msgData[7]=(actualTime/30)%60; | 
					
						
							|  |  |  |  |           msgData[8]=actualTime%30; | 
					
						
							|  |  |  |  |           break; | 
					
						
							|  |  |  |  |         case 4: // 30 (NTSC non-drop)
 | 
					
						
							|  |  |  |  |         default: | 
					
						
							|  |  |  |  |           msgData[5]=(actualTime/(60*60*30))%24; | 
					
						
							|  |  |  |  |           msgData[6]=(actualTime/(60*30))%60; | 
					
						
							|  |  |  |  |           msgData[7]=(actualTime/30)%60; | 
					
						
							|  |  |  |  |           msgData[8]=actualTime%30; | 
					
						
							|  |  |  |  |           break; | 
					
						
							|  |  |  |  |       } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |       msgData[5]|=(timeRate-1)<<5; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |       msgData[0]=0xf0; | 
					
						
							|  |  |  |  |       msgData[1]=0x7f; | 
					
						
							|  |  |  |  |       msgData[2]=0x7f; | 
					
						
							|  |  |  |  |       msgData[3]=0x01; | 
					
						
							|  |  |  |  |       msgData[4]=0x01; | 
					
						
							|  |  |  |  |       msgData[9]=0xf7; | 
					
						
							|  |  |  |  |       output->midiOut->send(msg); | 
					
						
							|  |  |  |  |     } | 
					
						
							| 
									
										
										
										
											2022-04-09 02:50:44 -04:00
										 |  |  |  |     output->midiOut->send(TAMidiMessage(TA_MIDI_MACHINE_PLAY,0,0)); | 
					
						
							|  |  |  |  |   } | 
					
						
							| 
									
										
										
										
											2023-09-09 18:12:49 -04:00
										 |  |  |  |   bool didItPlay=playing; | 
					
						
							| 
									
										
										
										
											2022-03-23 22:38:28 -04:00
										 |  |  |  |   BUSY_END; | 
					
						
							| 
									
										
										
										
											2023-09-09 18:12:49 -04:00
										 |  |  |  |   return didItPlay; | 
					
						
							| 
									
										
										
										
											2021-12-11 03:34:43 -05:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-09-09 18:12:49 -04:00
										 |  |  |  | bool DivEngine::playToRow(int row) { | 
					
						
							| 
									
										
										
										
											2022-03-23 22:38:28 -04:00
										 |  |  |  |   BUSY_BEGIN_SOFT; | 
					
						
							| 
									
										
										
										
											2022-02-06 16:29:30 -05:00
										 |  |  |  |   sPreview.sample=-1; | 
					
						
							|  |  |  |  |   sPreview.wave=-1; | 
					
						
							|  |  |  |  |   sPreview.pos=0; | 
					
						
							| 
									
										
										
										
											2022-08-11 09:21:54 -04:00
										 |  |  |  |   sPreview.dir=false; | 
					
						
							| 
									
										
										
										
											2022-02-06 00:07:35 -05:00
										 |  |  |  |   freelance=false; | 
					
						
							|  |  |  |  |   playSub(false,row); | 
					
						
							| 
									
										
										
										
											2022-02-10 03:15:39 -05:00
										 |  |  |  |   for (int i=0; i<DIV_MAX_CHANS; i++) { | 
					
						
							|  |  |  |  |     keyHit[i]=false; | 
					
						
							|  |  |  |  |   } | 
					
						
							| 
									
										
										
										
											2023-09-09 18:12:49 -04:00
										 |  |  |  |   bool didItPlay=playing; | 
					
						
							| 
									
										
										
										
											2022-03-23 22:38:28 -04:00
										 |  |  |  |   BUSY_END; | 
					
						
							| 
									
										
										
										
											2023-09-09 18:12:49 -04:00
										 |  |  |  |   return didItPlay; | 
					
						
							| 
									
										
										
										
											2022-02-06 00:07:35 -05:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-02-06 00:42:07 -05:00
										 |  |  |  | void DivEngine::stepOne(int row) { | 
					
						
							|  |  |  |  |   if (!isPlaying()) { | 
					
						
							| 
									
										
										
										
											2022-03-23 22:38:28 -04:00
										 |  |  |  |     BUSY_BEGIN_SOFT; | 
					
						
							| 
									
										
										
										
											2022-02-06 00:42:07 -05:00
										 |  |  |  |     freelance=false; | 
					
						
							|  |  |  |  |     playSub(false,row); | 
					
						
							| 
									
										
										
										
											2022-02-10 03:15:39 -05:00
										 |  |  |  |     for (int i=0; i<DIV_MAX_CHANS; i++) { | 
					
						
							|  |  |  |  |       keyHit[i]=false; | 
					
						
							|  |  |  |  |     } | 
					
						
							| 
									
										
										
										
											2022-03-23 22:38:28 -04:00
										 |  |  |  |   } else { | 
					
						
							|  |  |  |  |     BUSY_BEGIN; | 
					
						
							| 
									
										
										
										
											2022-02-06 00:42:07 -05:00
										 |  |  |  |   } | 
					
						
							|  |  |  |  |   stepPlay=2; | 
					
						
							|  |  |  |  |   ticks=1; | 
					
						
							| 
									
										
										
										
											2023-07-14 20:24:57 -04:00
										 |  |  |  |   prevOrder=curOrder; | 
					
						
							|  |  |  |  |   prevRow=curRow; | 
					
						
							| 
									
										
										
										
											2022-03-23 22:38:28 -04:00
										 |  |  |  |   BUSY_END; | 
					
						
							| 
									
										
										
										
											2022-02-06 00:42:07 -05:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-12-11 03:34:43 -05:00
										 |  |  |  | void DivEngine::stop() { | 
					
						
							| 
									
										
										
										
											2022-03-23 22:38:28 -04:00
										 |  |  |  |   BUSY_BEGIN; | 
					
						
							| 
									
										
										
										
											2021-12-28 18:23:57 -05:00
										 |  |  |  |   freelance=false; | 
					
						
							| 
									
										
										
										
											2023-04-30 18:22:35 -04:00
										 |  |  |  |   if (!playing) { | 
					
						
							|  |  |  |  |     //Send midi panic
 | 
					
						
							|  |  |  |  |     if (output) if (output->midiOut!=NULL) { | 
					
						
							|  |  |  |  |       output->midiOut->send(TAMidiMessage(TA_MIDI_CONTROL,0x7B,0)); | 
					
						
							|  |  |  |  |       logV("Midi panic sent"); | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  |   } | 
					
						
							| 
									
										
										
										
											2021-12-11 03:34:43 -05:00
										 |  |  |  |   playing=false; | 
					
						
							| 
									
										
										
										
											2021-12-20 19:46:49 -05:00
										 |  |  |  |   extValuePresent=false; | 
					
						
							| 
									
										
										
										
											2022-06-20 23:14:16 -04:00
										 |  |  |  |   endOfSong=false; // what?
 | 
					
						
							| 
									
										
										
										
											2022-02-06 00:42:07 -05:00
										 |  |  |  |   stepPlay=0; | 
					
						
							| 
									
										
										
										
											2022-06-20 23:14:16 -04:00
										 |  |  |  |   curOrder=prevOrder; | 
					
						
							|  |  |  |  |   curRow=prevRow; | 
					
						
							| 
									
										
										
										
											2022-01-17 23:34:29 -05:00
										 |  |  |  |   remainingLoops=-1; | 
					
						
							| 
									
										
										
										
											2022-02-06 16:29:30 -05:00
										 |  |  |  |   sPreview.sample=-1; | 
					
						
							|  |  |  |  |   sPreview.wave=-1; | 
					
						
							|  |  |  |  |   sPreview.pos=0; | 
					
						
							| 
									
										
										
										
											2022-08-11 09:21:54 -04:00
										 |  |  |  |   sPreview.dir=false; | 
					
						
							| 
									
										
										
										
											2022-03-07 01:48:48 -05:00
										 |  |  |  |   for (int i=0; i<song.systemLen; i++) { | 
					
						
							|  |  |  |  |     disCont[i].dispatch->notifyPlaybackStop(); | 
					
						
							|  |  |  |  |   } | 
					
						
							| 
									
										
										
										
											2022-04-09 02:50:44 -04:00
										 |  |  |  |   if (output) if (output->midiOut!=NULL) { | 
					
						
							|  |  |  |  |     output->midiOut->send(TAMidiMessage(TA_MIDI_MACHINE_STOP,0,0)); | 
					
						
							|  |  |  |  |     for (int i=0; i<chans; i++) { | 
					
						
							|  |  |  |  |       if (chan[i].curMidiNote>=0) { | 
					
						
							|  |  |  |  |         output->midiOut->send(TAMidiMessage(0x80|(i&15),chan[i].curMidiNote,0)); | 
					
						
							|  |  |  |  |       } | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  |   } | 
					
						
							| 
									
										
										
										
											2023-04-06 02:42:52 -04:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |   // reset all chan oscs
 | 
					
						
							|  |  |  |  |   for (int i=0; i<chans; i++) { | 
					
						
							|  |  |  |  |     DivDispatchOscBuffer* buf=disCont[dispatchOfChan[i]].dispatch->getOscBuffer(dispatchChanOfChan[i]); | 
					
						
							|  |  |  |  |     if (buf!=NULL) { | 
					
						
							| 
									
										
										
										
											2025-03-01 05:05:50 -05:00
										 |  |  |  |       buf->reset(); | 
					
						
							| 
									
										
										
										
											2023-04-06 02:42:52 -04:00
										 |  |  |  |     } | 
					
						
							|  |  |  |  |   } | 
					
						
							| 
									
										
										
										
											2022-03-23 22:38:28 -04:00
										 |  |  |  |   BUSY_END; | 
					
						
							| 
									
										
										
										
											2021-12-11 03:34:43 -05:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-02-03 18:38:57 -05:00
										 |  |  |  | void DivEngine::halt() { | 
					
						
							| 
									
										
										
										
											2022-03-23 22:38:28 -04:00
										 |  |  |  |   BUSY_BEGIN; | 
					
						
							| 
									
										
										
										
											2022-02-03 18:38:57 -05:00
										 |  |  |  |   halted=true; | 
					
						
							| 
									
										
										
										
											2022-03-23 22:38:28 -04:00
										 |  |  |  |   BUSY_END; | 
					
						
							| 
									
										
										
										
											2022-02-03 18:38:57 -05:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | void DivEngine::resume() { | 
					
						
							| 
									
										
										
										
											2022-03-23 22:38:28 -04:00
										 |  |  |  |   BUSY_BEGIN; | 
					
						
							| 
									
										
										
										
											2022-02-03 18:38:57 -05:00
										 |  |  |  |   halted=false; | 
					
						
							|  |  |  |  |   haltOn=DIV_HALT_NONE; | 
					
						
							| 
									
										
										
										
											2022-03-23 22:38:28 -04:00
										 |  |  |  |   BUSY_END; | 
					
						
							| 
									
										
										
										
											2022-02-03 18:38:57 -05:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | void DivEngine::haltWhen(DivHaltPositions when) { | 
					
						
							| 
									
										
										
										
											2022-03-23 22:38:28 -04:00
										 |  |  |  |   BUSY_BEGIN; | 
					
						
							| 
									
										
										
										
											2022-02-03 18:38:57 -05:00
										 |  |  |  |   halted=false; | 
					
						
							|  |  |  |  |   haltOn=when; | 
					
						
							| 
									
										
										
										
											2022-03-23 22:38:28 -04:00
										 |  |  |  |   BUSY_END; | 
					
						
							| 
									
										
										
										
											2022-02-03 18:38:57 -05:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | bool DivEngine::isHalted() { | 
					
						
							|  |  |  |  |   return halted; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | const char** DivEngine::getRegisterSheet(int sys) { | 
					
						
							|  |  |  |  |   if (sys<0 || sys>=song.systemLen) return NULL; | 
					
						
							|  |  |  |  |   return disCont[sys].dispatch->getRegisterSheet(); | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-01-08 16:03:32 -05:00
										 |  |  |  | void DivEngine::recalcChans() { | 
					
						
							| 
									
										
										
										
											2022-04-27 05:48:56 -04:00
										 |  |  |  |   bool isInsTypePossible[DIV_INS_MAX]; | 
					
						
							| 
									
										
										
										
											2022-01-08 16:03:32 -05:00
										 |  |  |  |   chans=0; | 
					
						
							|  |  |  |  |   int chanIndex=0; | 
					
						
							| 
									
										
										
										
											2022-04-27 05:48:56 -04:00
										 |  |  |  |   memset(isInsTypePossible,0,DIV_INS_MAX*sizeof(bool)); | 
					
						
							| 
									
										
										
										
											2022-01-08 16:03:32 -05:00
										 |  |  |  |   for (int i=0; i<song.systemLen; i++) { | 
					
						
							|  |  |  |  |     int chanCount=getChannelCount(song.system[i]); | 
					
						
							| 
									
										
										
										
											2023-10-07 21:35:25 -04:00
										 |  |  |  |     int firstChan=chans; | 
					
						
							| 
									
										
										
										
											2022-01-08 16:03:32 -05:00
										 |  |  |  |     chans+=chanCount; | 
					
						
							|  |  |  |  |     for (int j=0; j<chanCount; j++) { | 
					
						
							|  |  |  |  |       sysOfChan[chanIndex]=song.system[i]; | 
					
						
							|  |  |  |  |       dispatchOfChan[chanIndex]=i; | 
					
						
							|  |  |  |  |       dispatchChanOfChan[chanIndex]=j; | 
					
						
							| 
									
										
										
										
											2023-10-07 21:35:25 -04:00
										 |  |  |  |       dispatchFirstChan[chanIndex]=firstChan; | 
					
						
							| 
									
										
										
										
											2022-01-08 16:03:32 -05:00
										 |  |  |  |       chanIndex++; | 
					
						
							| 
									
										
										
										
											2022-04-27 05:48:56 -04:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |       if (sysDefs[song.system[i]]!=NULL) { | 
					
						
							|  |  |  |  |         if (sysDefs[song.system[i]]->chanInsType[j][0]!=DIV_INS_NULL) { | 
					
						
							|  |  |  |  |           isInsTypePossible[sysDefs[song.system[i]]->chanInsType[j][0]]=true; | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         if (sysDefs[song.system[i]]->chanInsType[j][1]!=DIV_INS_NULL) { | 
					
						
							|  |  |  |  |           isInsTypePossible[sysDefs[song.system[i]]->chanInsType[j][1]]=true; | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  |       } | 
					
						
							| 
									
										
										
										
											2022-01-08 16:03:32 -05:00
										 |  |  |  |     } | 
					
						
							|  |  |  |  |   } | 
					
						
							| 
									
										
										
										
											2022-04-27 05:48:56 -04:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |   possibleInsTypes.clear(); | 
					
						
							|  |  |  |  |   for (int i=0; i<DIV_INS_MAX; i++) { | 
					
						
							|  |  |  |  |     if (isInsTypePossible[i]) possibleInsTypes.push_back((DivInstrumentType)i); | 
					
						
							|  |  |  |  |   } | 
					
						
							| 
									
										
										
										
											2022-04-30 02:37:37 -04:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-03-12 17:13:00 -04:00
										 |  |  |  |   checkAssetDir(song.insDir,song.ins.size()); | 
					
						
							|  |  |  |  |   checkAssetDir(song.waveDir,song.wave.size()); | 
					
						
							|  |  |  |  |   checkAssetDir(song.sampleDir,song.sample.size()); | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-30 02:37:37 -04:00
										 |  |  |  |   hasLoadedSomething=true; | 
					
						
							| 
									
										
										
										
											2022-01-08 16:03:32 -05:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-12-11 16:51:34 -05:00
										 |  |  |  | void DivEngine::reset() { | 
					
						
							| 
									
										
										
										
											2022-09-22 03:46:55 -04:00
										 |  |  |  |   if (output) if (output->midiOut!=NULL) { | 
					
						
							|  |  |  |  |     output->midiOut->send(TAMidiMessage(TA_MIDI_MACHINE_STOP,0,0)); | 
					
						
							|  |  |  |  |     for (int i=0; i<chans; i++) { | 
					
						
							|  |  |  |  |       if (chan[i].curMidiNote>=0) { | 
					
						
							|  |  |  |  |         output->midiOut->send(TAMidiMessage(0x80|(i&15),chan[i].curMidiNote,0)); | 
					
						
							|  |  |  |  |       } | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  |   } | 
					
						
							| 
									
										
										
										
											2022-01-08 01:57:37 -05:00
										 |  |  |  |   for (int i=0; i<DIV_MAX_CHANS; i++) { | 
					
						
							| 
									
										
										
										
											2021-12-11 16:51:34 -05:00
										 |  |  |  |     chan[i]=DivChannelState(); | 
					
						
							| 
									
										
										
										
											2022-01-08 16:03:32 -05:00
										 |  |  |  |     if (i<chans) chan[i].volMax=(disCont[dispatchOfChan[i]].dispatch->dispatch(DivCommand(DIV_CMD_GET_VOLMAX,dispatchChanOfChan[i]))<<8)|0xff; | 
					
						
							| 
									
										
										
										
											2021-12-11 16:51:34 -05:00
										 |  |  |  |     chan[i].volume=chan[i].volMax; | 
					
						
							| 
									
										
										
										
											2022-05-10 04:51:18 -04:00
										 |  |  |  |     if (song.linearPitch==0) chan[i].vibratoFine=4; | 
					
						
							| 
									
										
										
										
											2021-12-11 16:51:34 -05:00
										 |  |  |  |   } | 
					
						
							| 
									
										
										
										
											2021-12-20 19:46:49 -05:00
										 |  |  |  |   extValue=0; | 
					
						
							|  |  |  |  |   extValuePresent=0; | 
					
						
							| 
									
										
										
										
											2023-02-05 02:56:39 -05:00
										 |  |  |  |   speeds=curSubSong->speeds; | 
					
						
							| 
									
										
										
										
											2024-03-15 15:56:55 -04:00
										 |  |  |  |   virtualTempoN=curSubSong->virtualTempoN; | 
					
						
							|  |  |  |  |   virtualTempoD=curSubSong->virtualTempoD; | 
					
						
							| 
									
										
										
										
											2022-03-24 00:56:59 -04:00
										 |  |  |  |   firstTick=false; | 
					
						
							| 
									
										
										
										
											2022-09-29 01:27:40 -04:00
										 |  |  |  |   shallStop=false; | 
					
						
							| 
									
										
										
										
											2022-10-22 04:46:39 -04:00
										 |  |  |  |   shallStopSched=false; | 
					
						
							| 
									
										
										
										
											2022-11-09 23:52:10 -05:00
										 |  |  |  |   pendingMetroTick=0; | 
					
						
							| 
									
										
										
										
											2022-11-10 01:26:59 -05:00
										 |  |  |  |   elapsedBars=0; | 
					
						
							|  |  |  |  |   elapsedBeats=0; | 
					
						
							| 
									
										
										
										
											2023-02-05 02:56:39 -05:00
										 |  |  |  |   nextSpeed=speeds.val[0]; | 
					
						
							| 
									
										
										
										
											2023-06-11 19:57:32 -04:00
										 |  |  |  |   divider=curSubSong->hz; | 
					
						
							| 
									
										
										
										
											2021-12-27 15:22:57 -05:00
										 |  |  |  |   globalPitch=0; | 
					
						
							| 
									
										
										
										
											2022-01-08 16:03:32 -05:00
										 |  |  |  |   for (int i=0; i<song.systemLen; i++) { | 
					
						
							|  |  |  |  |     disCont[i].dispatch->reset(); | 
					
						
							| 
									
										
										
										
											2022-01-19 21:04:51 -05:00
										 |  |  |  |     disCont[i].clear(); | 
					
						
							| 
									
										
										
										
											2022-01-08 16:03:32 -05:00
										 |  |  |  |   } | 
					
						
							| 
									
										
										
										
											2021-12-11 16:51:34 -05:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-12-19 21:11:23 -05:00
										 |  |  |  | void DivEngine::syncReset() { | 
					
						
							| 
									
										
										
										
											2022-03-23 22:38:28 -04:00
										 |  |  |  |   BUSY_BEGIN; | 
					
						
							| 
									
										
										
										
											2021-12-19 21:11:23 -05:00
										 |  |  |  |   reset(); | 
					
						
							| 
									
										
										
										
											2022-03-23 22:38:28 -04:00
										 |  |  |  |   BUSY_END; | 
					
						
							| 
									
										
										
										
											2021-12-19 21:11:23 -05:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-12-21 13:06:14 -05:00
										 |  |  |  | const int sampleRates[6]={ | 
					
						
							|  |  |  |  |   4000, 8000, 11025, 16000, 22050, 32000 | 
					
						
							|  |  |  |  | }; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-12-23 18:04:44 -05:00
										 |  |  |  | int DivEngine::fileToDivRate(int frate) { | 
					
						
							|  |  |  |  |   if (frate<0) frate=0; | 
					
						
							|  |  |  |  |   if (frate>5) frate=5; | 
					
						
							|  |  |  |  |   return sampleRates[frate]; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | int DivEngine::divToFileRate(int drate) { | 
					
						
							|  |  |  |  |   if (drate>26000) { | 
					
						
							|  |  |  |  |     return 5; | 
					
						
							|  |  |  |  |   } else if (drate>18000) { | 
					
						
							|  |  |  |  |     return 4; | 
					
						
							|  |  |  |  |   } else if (drate>14000) { | 
					
						
							|  |  |  |  |     return 3; | 
					
						
							|  |  |  |  |   } else if (drate>9500) { | 
					
						
							|  |  |  |  |     return 2; | 
					
						
							|  |  |  |  |   } else if (drate>6000) { | 
					
						
							|  |  |  |  |     return 1; | 
					
						
							|  |  |  |  |   } else { | 
					
						
							|  |  |  |  |     return 0; | 
					
						
							|  |  |  |  |   } | 
					
						
							|  |  |  |  |   return 4; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-03-13 15:17:05 -04:00
										 |  |  |  | void DivEngine::testFunction() { | 
					
						
							|  |  |  |  |   logI("it works!"); | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-12-23 18:04:44 -05:00
										 |  |  |  | int DivEngine::getEffectiveSampleRate(int rate) { | 
					
						
							|  |  |  |  |   if (rate<1) return 0; | 
					
						
							| 
									
										
										
										
											2022-01-08 03:02:04 -05:00
										 |  |  |  |   switch (song.system[0]) { | 
					
						
							| 
									
										
										
										
											2021-12-23 18:04:44 -05:00
										 |  |  |  |     case DIV_SYSTEM_YMU759: | 
					
						
							|  |  |  |  |       return 8000; | 
					
						
							| 
									
										
										
										
											2022-02-23 02:52:30 -05:00
										 |  |  |  |     case DIV_SYSTEM_YM2612: case DIV_SYSTEM_YM2612_EXT: | 
					
						
							| 
									
										
										
										
											2021-12-23 18:04:44 -05:00
										 |  |  |  |       return 1278409/(1280000/rate); | 
					
						
							|  |  |  |  |     case DIV_SYSTEM_PCE: | 
					
						
							|  |  |  |  |       return 1789773/(1789773/rate); | 
					
						
							| 
									
										
										
										
											2022-02-23 02:52:30 -05:00
										 |  |  |  |     case DIV_SYSTEM_SEGAPCM: case DIV_SYSTEM_SEGAPCM_COMPAT: | 
					
						
							| 
									
										
										
										
											2021-12-23 18:04:44 -05:00
										 |  |  |  |       return (31250*MIN(255,(rate*255/31250)))/255; | 
					
						
							| 
									
										
										
										
											2022-03-07 10:15:21 -05:00
										 |  |  |  |     case DIV_SYSTEM_QSOUND: | 
					
						
							| 
									
										
										
										
											2022-02-24 11:02:35 -05:00
										 |  |  |  |       return (24038*MIN(65535,(rate*4096/24038)))/4096; | 
					
						
							|  |  |  |  |     case DIV_SYSTEM_YM2610: case DIV_SYSTEM_YM2610_EXT: case DIV_SYSTEM_YM2610_FULL: case DIV_SYSTEM_YM2610_FULL_EXT: case DIV_SYSTEM_YM2610B: case DIV_SYSTEM_YM2610B_EXT: | 
					
						
							| 
									
										
										
										
											2021-12-23 18:04:44 -05:00
										 |  |  |  |       return 18518; | 
					
						
							| 
									
										
										
										
											2022-03-04 06:13:49 -05:00
										 |  |  |  |     case DIV_SYSTEM_VERA: | 
					
						
							|  |  |  |  |       return (48828*MIN(128,(rate*128/48828)))/128; | 
					
						
							| 
									
										
										
										
											2022-03-06 12:31:03 -05:00
										 |  |  |  |     case DIV_SYSTEM_X1_010: | 
					
						
							|  |  |  |  |       return (31250*MIN(255,(rate*16/31250)))/16; // TODO: support variable clock case
 | 
					
						
							| 
									
										
										
										
											2022-04-20 12:52:37 -04:00
										 |  |  |  |     case DIV_SYSTEM_ES5506: | 
					
						
							|  |  |  |  |       return (31250*MIN(131071,(rate*2048/31250)))/2048; // TODO: support variable clock, channel limit case
 | 
					
						
							| 
									
										
										
										
											2021-12-23 18:04:44 -05:00
										 |  |  |  |     default: | 
					
						
							|  |  |  |  |       break; | 
					
						
							|  |  |  |  |   } | 
					
						
							|  |  |  |  |   return rate; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-05-29 03:13:08 -04:00
										 |  |  |  | void DivEngine::previewSample(int sample, int note, int pStart, int pEnd) { | 
					
						
							| 
									
										
										
										
											2022-03-23 22:38:28 -04:00
										 |  |  |  |   BUSY_BEGIN; | 
					
						
							| 
									
										
										
										
											2022-11-06 02:06:51 -05:00
										 |  |  |  |   previewSampleNoLock(sample,note,pStart,pEnd); | 
					
						
							|  |  |  |  |   BUSY_END; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | void DivEngine::stopSamplePreview() { | 
					
						
							|  |  |  |  |   BUSY_BEGIN; | 
					
						
							|  |  |  |  |   stopSamplePreviewNoLock(); | 
					
						
							|  |  |  |  |   BUSY_END; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | void DivEngine::previewWave(int wave, int note) { | 
					
						
							|  |  |  |  |   BUSY_BEGIN; | 
					
						
							|  |  |  |  |   previewWaveNoLock(wave,note); | 
					
						
							|  |  |  |  |   BUSY_END; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | void DivEngine::stopWavePreview() { | 
					
						
							|  |  |  |  |   BUSY_BEGIN; | 
					
						
							|  |  |  |  |   stopWavePreviewNoLock(); | 
					
						
							|  |  |  |  |   BUSY_END; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | void DivEngine::previewSampleNoLock(int sample, int note, int pStart, int pEnd) { | 
					
						
							| 
									
										
										
										
											2022-05-29 03:13:08 -04:00
										 |  |  |  |   sPreview.pBegin=pStart; | 
					
						
							|  |  |  |  |   sPreview.pEnd=pEnd; | 
					
						
							| 
									
										
										
										
											2022-08-11 09:21:54 -04:00
										 |  |  |  |   sPreview.dir=false; | 
					
						
							| 
									
										
										
										
											2021-12-28 23:10:13 -05:00
										 |  |  |  |   if (sample<0 || sample>=(int)song.sample.size()) { | 
					
						
							| 
									
										
										
										
											2021-12-21 13:06:14 -05:00
										 |  |  |  |     sPreview.sample=-1; | 
					
						
							|  |  |  |  |     sPreview.pos=0; | 
					
						
							| 
									
										
										
										
											2022-08-11 09:21:54 -04:00
										 |  |  |  |     sPreview.dir=false; | 
					
						
							| 
									
										
										
										
											2021-12-21 13:06:14 -05:00
										 |  |  |  |     return; | 
					
						
							|  |  |  |  |   } | 
					
						
							| 
									
										
										
										
											2022-01-08 16:03:32 -05:00
										 |  |  |  |   blip_clear(samp_bb); | 
					
						
							| 
									
										
										
										
											2022-11-29 00:17:38 -05:00
										 |  |  |  |   double rate=song.sample[sample]->centerRate; | 
					
						
							| 
									
										
										
										
											2022-01-20 16:51:31 -05:00
										 |  |  |  |   if (note>=0) { | 
					
						
							| 
									
										
										
										
											2022-09-30 19:24:20 -04:00
										 |  |  |  |     rate=(pow(2.0,(double)(note)/12.0)*((double)song.sample[sample]->centerRate)*0.0625); | 
					
						
							| 
									
										
										
										
											2022-11-29 00:17:38 -05:00
										 |  |  |  |     if (rate<=0) rate=song.sample[sample]->centerRate; | 
					
						
							| 
									
										
										
										
											2022-01-20 16:51:31 -05:00
										 |  |  |  |   } | 
					
						
							| 
									
										
										
										
											2022-03-20 14:36:48 -04:00
										 |  |  |  |   if (rate<100) rate=100; | 
					
						
							| 
									
										
										
										
											2023-04-13 19:11:10 -04:00
										 |  |  |  |   double rateOrig=rate; | 
					
						
							|  |  |  |  |   sPreview.rateMul=1; | 
					
						
							|  |  |  |  |   while (sPreview.rateMul<0x40000000 && rate<got.rate) { | 
					
						
							|  |  |  |  |     sPreview.rateMul<<=1; | 
					
						
							|  |  |  |  |     rate*=2.0; | 
					
						
							|  |  |  |  |   } | 
					
						
							| 
									
										
										
										
											2022-01-20 16:51:31 -05:00
										 |  |  |  |   blip_set_rates(samp_bb,rate,got.rate); | 
					
						
							| 
									
										
										
										
											2022-01-08 16:03:32 -05:00
										 |  |  |  |   samp_prevSample=0; | 
					
						
							| 
									
										
										
										
											2023-04-13 19:11:10 -04:00
										 |  |  |  |   sPreview.rate=rateOrig; | 
					
						
							| 
									
										
										
										
											2022-05-29 03:13:08 -04:00
										 |  |  |  |   sPreview.pos=(sPreview.pBegin>=0)?sPreview.pBegin:0; | 
					
						
							| 
									
										
										
										
											2023-04-13 19:11:10 -04:00
										 |  |  |  |   sPreview.posSub=0; | 
					
						
							| 
									
										
										
										
											2021-12-21 13:06:14 -05:00
										 |  |  |  |   sPreview.sample=sample; | 
					
						
							| 
									
										
										
										
											2022-01-20 00:07:53 -05:00
										 |  |  |  |   sPreview.wave=-1; | 
					
						
							| 
									
										
										
										
											2022-08-11 09:21:54 -04:00
										 |  |  |  |   sPreview.dir=false; | 
					
						
							| 
									
										
										
										
											2021-12-21 13:06:14 -05:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-06 02:06:51 -05:00
										 |  |  |  | void DivEngine::stopSamplePreviewNoLock() { | 
					
						
							| 
									
										
										
										
											2022-01-20 16:51:31 -05:00
										 |  |  |  |   sPreview.sample=-1; | 
					
						
							|  |  |  |  |   sPreview.pos=0; | 
					
						
							| 
									
										
										
										
											2022-08-11 09:21:54 -04:00
										 |  |  |  |   sPreview.dir=false; | 
					
						
							| 
									
										
										
										
											2022-01-20 16:51:31 -05:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-06 02:06:51 -05:00
										 |  |  |  | void DivEngine::previewWaveNoLock(int wave, int note) { | 
					
						
							| 
									
										
										
										
											2022-01-20 00:07:53 -05:00
										 |  |  |  |   if (wave<0 || wave>=(int)song.wave.size()) { | 
					
						
							|  |  |  |  |     sPreview.wave=-1; | 
					
						
							|  |  |  |  |     sPreview.pos=0; | 
					
						
							| 
									
										
										
										
											2022-08-11 09:21:54 -04:00
										 |  |  |  |     sPreview.dir=false; | 
					
						
							| 
									
										
										
										
											2022-01-20 00:07:53 -05:00
										 |  |  |  |     return; | 
					
						
							|  |  |  |  |   } | 
					
						
							| 
									
										
										
										
											2022-01-20 00:43:08 -05:00
										 |  |  |  |   if (song.wave[wave]->len<=0) { | 
					
						
							|  |  |  |  |     return; | 
					
						
							|  |  |  |  |   } | 
					
						
							| 
									
										
										
										
											2022-01-20 00:07:53 -05:00
										 |  |  |  |   blip_clear(samp_bb); | 
					
						
							| 
									
										
										
										
											2022-03-20 14:36:48 -04:00
										 |  |  |  |   double rate=song.wave[wave]->len*((song.tuning*0.0625)*pow(2.0,(double)(note+3)/12.0)); | 
					
						
							|  |  |  |  |   if (rate<100) rate=100; | 
					
						
							| 
									
										
										
										
											2023-04-13 19:11:10 -04:00
										 |  |  |  |   double rateOrig=rate; | 
					
						
							|  |  |  |  |   sPreview.rateMul=1; | 
					
						
							|  |  |  |  |   while (sPreview.rateMul<0x40000000 && rate<got.rate) { | 
					
						
							|  |  |  |  |     sPreview.rateMul<<=1; | 
					
						
							|  |  |  |  |     rate*=2.0; | 
					
						
							|  |  |  |  |   } | 
					
						
							| 
									
										
										
										
											2022-03-20 14:36:48 -04:00
										 |  |  |  |   blip_set_rates(samp_bb,rate,got.rate); | 
					
						
							| 
									
										
										
										
											2022-01-20 00:07:53 -05:00
										 |  |  |  |   samp_prevSample=0; | 
					
						
							| 
									
										
										
										
											2023-04-13 19:11:10 -04:00
										 |  |  |  |   sPreview.rate=rateOrig; | 
					
						
							| 
									
										
										
										
											2022-01-20 00:07:53 -05:00
										 |  |  |  |   sPreview.pos=0; | 
					
						
							| 
									
										
										
										
											2023-04-13 19:11:10 -04:00
										 |  |  |  |   sPreview.posSub=0; | 
					
						
							| 
									
										
										
										
											2022-01-20 00:07:53 -05:00
										 |  |  |  |   sPreview.sample=-1; | 
					
						
							|  |  |  |  |   sPreview.wave=wave; | 
					
						
							| 
									
										
										
										
											2022-08-11 09:21:54 -04:00
										 |  |  |  |   sPreview.dir=false; | 
					
						
							| 
									
										
										
										
											2022-01-20 00:07:53 -05:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-06 02:06:51 -05:00
										 |  |  |  | void DivEngine::stopWavePreviewNoLock() { | 
					
						
							| 
									
										
										
										
											2022-01-20 00:07:53 -05:00
										 |  |  |  |   sPreview.wave=-1; | 
					
						
							|  |  |  |  |   sPreview.pos=0; | 
					
						
							| 
									
										
										
										
											2022-08-11 09:21:54 -04:00
										 |  |  |  |   sPreview.dir=false; | 
					
						
							| 
									
										
										
										
											2022-01-20 00:07:53 -05:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-30 18:47:17 -04:00
										 |  |  |  | bool DivEngine::isPreviewingSample() { | 
					
						
							| 
									
										
										
										
											2025-02-10 07:35:11 -05:00
										 |  |  |  |   return (sPreview.sample>=0 && sPreview.sample<(int)song.sample.size() && sPreview.pos!=sPreview.pEnd); | 
					
						
							| 
									
										
										
										
											2022-09-30 18:47:17 -04:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-09-05 16:38:02 -04:00
										 |  |  |  | int DivEngine::getSamplePreviewSample() { | 
					
						
							|  |  |  |  |   return sPreview.sample; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-30 18:47:17 -04:00
										 |  |  |  | int DivEngine::getSamplePreviewPos() { | 
					
						
							|  |  |  |  |   return sPreview.pos; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | double DivEngine::getSamplePreviewRate() { | 
					
						
							|  |  |  |  |   return sPreview.rate; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-02-06 02:14:21 -05:00
										 |  |  |  | double DivEngine::getCenterRate() { | 
					
						
							|  |  |  |  |   return song.oldCenterRate?8363.0:8372.0; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-12-19 03:16:24 -05:00
										 |  |  |  | String DivEngine::getConfigPath() { | 
					
						
							|  |  |  |  |   return configPath; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-12-13 17:09:46 -05:00
										 |  |  |  | int DivEngine::getMaxVolumeChan(int ch) { | 
					
						
							|  |  |  |  |   return chan[ch].volMax>>8; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-12-17 15:08:52 -05:00
										 |  |  |  | int DivEngine::mapVelocity(int ch, float vel) { | 
					
						
							|  |  |  |  |   if (ch<0) return 0; | 
					
						
							|  |  |  |  |   if (ch>=chans) return 0; | 
					
						
							|  |  |  |  |   if (disCont[dispatchOfChan[ch]].dispatch==NULL) return 0; | 
					
						
							|  |  |  |  |   return disCont[dispatchOfChan[ch]].dispatch->mapVelocity(dispatchChanOfChan[ch],vel); | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-07-22 20:08:39 -04:00
										 |  |  |  | float DivEngine::getGain(int ch, int vol) { | 
					
						
							|  |  |  |  |   if (ch<0) return 0; | 
					
						
							|  |  |  |  |   if (ch>=chans) return 0; | 
					
						
							|  |  |  |  |   if (disCont[dispatchOfChan[ch]].dispatch==NULL) return 0; | 
					
						
							|  |  |  |  |   return disCont[dispatchOfChan[ch]].dispatch->getGain(dispatchChanOfChan[ch],vol); | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-12-11 03:34:43 -05:00
										 |  |  |  | unsigned char DivEngine::getOrder() { | 
					
						
							| 
									
										
										
										
											2022-06-06 02:05:06 -04:00
										 |  |  |  |   return prevOrder; | 
					
						
							| 
									
										
										
										
											2021-12-11 03:34:43 -05:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-12-13 17:09:46 -05:00
										 |  |  |  | int DivEngine::getRow() { | 
					
						
							| 
									
										
										
										
											2022-06-06 02:05:06 -04:00
										 |  |  |  |   return prevRow; | 
					
						
							| 
									
										
										
										
											2021-12-13 17:09:46 -05:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-09-16 16:04:11 -04:00
										 |  |  |  | void DivEngine::getPlayPos(int& order, int& row) { | 
					
						
							|  |  |  |  |   playPosLock.lock(); | 
					
						
							|  |  |  |  |   order=prevOrder; | 
					
						
							|  |  |  |  |   row=prevRow; | 
					
						
							|  |  |  |  |   playPosLock.unlock(); | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-03-11 05:03:05 -04:00
										 |  |  |  | void DivEngine::getPlayPosTick(int& order, int& row, int& tick, int& speed) { | 
					
						
							|  |  |  |  |   playPosLock.lock(); | 
					
						
							|  |  |  |  |   order=prevOrder; | 
					
						
							|  |  |  |  |   row=prevRow; | 
					
						
							|  |  |  |  |   tick=ticks; | 
					
						
							| 
									
										
										
										
											2025-03-11 05:15:14 -04:00
										 |  |  |  |   speed=prevSpeed; | 
					
						
							| 
									
										
										
										
											2025-03-11 05:03:05 -04:00
										 |  |  |  |   playPosLock.unlock(); | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-10 01:26:59 -05:00
										 |  |  |  | int DivEngine::getElapsedBars() { | 
					
						
							|  |  |  |  |   return elapsedBars; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | int DivEngine::getElapsedBeats() { | 
					
						
							|  |  |  |  |   return elapsedBeats; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-05-15 02:42:49 -04:00
										 |  |  |  | size_t DivEngine::getCurrentSubSong() { | 
					
						
							|  |  |  |  |   return curSubSongIndex; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-05 02:56:39 -05:00
										 |  |  |  | const DivGroovePattern& DivEngine::getSpeeds() { | 
					
						
							|  |  |  |  |   return speeds; | 
					
						
							| 
									
										
										
										
											2021-12-21 02:30:09 -05:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-03-16 00:30:15 -04:00
										 |  |  |  | float DivEngine::getHz() { | 
					
						
							| 
									
										
										
										
											2023-06-11 19:57:32 -04:00
										 |  |  |  |   return curSubSong->hz; | 
					
						
							| 
									
										
										
										
											2021-12-21 02:30:09 -05:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-03-16 00:30:15 -04:00
										 |  |  |  | float DivEngine::getCurHz() { | 
					
						
							| 
									
										
										
										
											2022-01-12 02:45:26 -05:00
										 |  |  |  |   return divider; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-03-15 15:56:55 -04:00
										 |  |  |  | short DivEngine::getVirtualTempoN() { | 
					
						
							|  |  |  |  |   return virtualTempoN; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | short DivEngine::getVirtualTempoD() { | 
					
						
							|  |  |  |  |   return virtualTempoD; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | void DivEngine::virtualTempoChanged() { | 
					
						
							|  |  |  |  |   BUSY_BEGIN; | 
					
						
							|  |  |  |  |   virtualTempoN=curSubSong->virtualTempoN; | 
					
						
							|  |  |  |  |   virtualTempoD=curSubSong->virtualTempoD; | 
					
						
							|  |  |  |  |   BUSY_END; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-01-12 02:45:26 -05:00
										 |  |  |  | int DivEngine::getTotalSeconds() { | 
					
						
							|  |  |  |  |   return totalSeconds; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-12-21 02:30:09 -05:00
										 |  |  |  | int DivEngine::getTotalTicks() { | 
					
						
							|  |  |  |  |   return totalTicks; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-12-21 17:42:27 -05:00
										 |  |  |  | bool DivEngine::getRepeatPattern() { | 
					
						
							|  |  |  |  |   return repeatPattern; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | void DivEngine::setRepeatPattern(bool value) { | 
					
						
							| 
									
										
										
										
											2022-03-23 22:38:28 -04:00
										 |  |  |  |   BUSY_BEGIN; | 
					
						
							| 
									
										
										
										
											2021-12-21 17:42:27 -05:00
										 |  |  |  |   repeatPattern=value; | 
					
						
							| 
									
										
										
										
											2022-03-23 22:38:28 -04:00
										 |  |  |  |   BUSY_END; | 
					
						
							| 
									
										
										
										
											2021-12-21 17:42:27 -05:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-12-20 19:46:49 -05:00
										 |  |  |  | bool DivEngine::hasExtValue() { | 
					
						
							|  |  |  |  |   return extValuePresent; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | unsigned char DivEngine::getExtValue() { | 
					
						
							|  |  |  |  |   return extValue; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-12-13 17:09:46 -05:00
										 |  |  |  | bool DivEngine::isPlaying() { | 
					
						
							| 
									
										
										
										
											2021-12-28 18:23:57 -05:00
										 |  |  |  |   return (playing && !freelance); | 
					
						
							| 
									
										
										
										
											2021-12-13 17:09:46 -05:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-06-29 02:37:12 -04:00
										 |  |  |  | bool DivEngine::isRunning() { | 
					
						
							|  |  |  |  |   return playing; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-02-06 00:42:07 -05:00
										 |  |  |  | bool DivEngine::isStepping() { | 
					
						
							|  |  |  |  |   return !(stepPlay==0); | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-12-18 03:25:42 -05:00
										 |  |  |  | bool DivEngine::isChannelMuted(int chan) { | 
					
						
							|  |  |  |  |   return isMuted[chan]; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | void DivEngine::toggleMute(int chan) { | 
					
						
							|  |  |  |  |   muteChannel(chan,!isMuted[chan]); | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | void DivEngine::toggleSolo(int chan) { | 
					
						
							|  |  |  |  |   bool solo=false; | 
					
						
							|  |  |  |  |   for (int i=0; i<chans; i++) { | 
					
						
							|  |  |  |  |     if (i==chan) { | 
					
						
							|  |  |  |  |       solo=true; | 
					
						
							|  |  |  |  |       continue; | 
					
						
							|  |  |  |  |     } else { | 
					
						
							|  |  |  |  |       if (!isMuted[i]) { | 
					
						
							|  |  |  |  |         solo=false; | 
					
						
							|  |  |  |  |         break; | 
					
						
							|  |  |  |  |       } | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  |   } | 
					
						
							| 
									
										
										
										
											2022-03-23 22:38:28 -04:00
										 |  |  |  |   BUSY_BEGIN; | 
					
						
							| 
									
										
										
										
											2021-12-18 03:25:42 -05:00
										 |  |  |  |   if (!solo) { | 
					
						
							|  |  |  |  |     for (int i=0; i<chans; i++) { | 
					
						
							|  |  |  |  |       isMuted[i]=(i!=chan); | 
					
						
							| 
									
										
										
										
											2022-01-08 16:03:32 -05:00
										 |  |  |  |       if (disCont[dispatchOfChan[i]].dispatch!=NULL) { | 
					
						
							|  |  |  |  |         disCont[dispatchOfChan[i]].dispatch->muteChannel(dispatchChanOfChan[i],isMuted[i]); | 
					
						
							| 
									
										
										
										
											2021-12-18 03:25:42 -05:00
										 |  |  |  |       } | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  |   } else { | 
					
						
							|  |  |  |  |     for (int i=0; i<chans; i++) { | 
					
						
							|  |  |  |  |       isMuted[i]=false; | 
					
						
							| 
									
										
										
										
											2022-01-08 16:03:32 -05:00
										 |  |  |  |       if (disCont[dispatchOfChan[i]].dispatch!=NULL) { | 
					
						
							|  |  |  |  |         disCont[dispatchOfChan[i]].dispatch->muteChannel(dispatchChanOfChan[i],isMuted[i]); | 
					
						
							| 
									
										
										
										
											2021-12-18 03:25:42 -05:00
										 |  |  |  |       } | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  |   } | 
					
						
							| 
									
										
										
										
											2022-03-23 22:38:28 -04:00
										 |  |  |  |   BUSY_END; | 
					
						
							| 
									
										
										
										
											2021-12-18 03:25:42 -05:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | void DivEngine::muteChannel(int chan, bool mute) { | 
					
						
							| 
									
										
										
										
											2022-03-23 22:38:28 -04:00
										 |  |  |  |   BUSY_BEGIN; | 
					
						
							| 
									
										
										
										
											2021-12-18 03:25:42 -05:00
										 |  |  |  |   isMuted[chan]=mute; | 
					
						
							| 
									
										
										
										
											2022-01-08 16:03:32 -05:00
										 |  |  |  |   if (disCont[dispatchOfChan[chan]].dispatch!=NULL) { | 
					
						
							|  |  |  |  |     disCont[dispatchOfChan[chan]].dispatch->muteChannel(dispatchChanOfChan[chan],isMuted[chan]); | 
					
						
							| 
									
										
										
										
											2021-12-18 03:25:42 -05:00
										 |  |  |  |   } | 
					
						
							| 
									
										
										
										
											2022-03-23 22:38:28 -04:00
										 |  |  |  |   BUSY_END; | 
					
						
							| 
									
										
										
										
											2021-12-18 03:25:42 -05:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-02-11 18:20:39 -05:00
										 |  |  |  | void DivEngine::unmuteAll() { | 
					
						
							| 
									
										
										
										
											2022-03-23 22:38:28 -04:00
										 |  |  |  |   BUSY_BEGIN; | 
					
						
							| 
									
										
										
										
											2022-02-11 18:20:39 -05:00
										 |  |  |  |   for (int i=0; i<chans; i++) { | 
					
						
							|  |  |  |  |     isMuted[i]=false; | 
					
						
							|  |  |  |  |     if (disCont[dispatchOfChan[i]].dispatch!=NULL) { | 
					
						
							|  |  |  |  |       disCont[dispatchOfChan[i]].dispatch->muteChannel(dispatchChanOfChan[i],isMuted[i]); | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  |   } | 
					
						
							| 
									
										
										
										
											2022-03-23 22:38:28 -04:00
										 |  |  |  |   BUSY_END; | 
					
						
							| 
									
										
										
										
											2022-02-11 18:20:39 -05:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-10-17 15:11:35 -04:00
										 |  |  |  | void DivEngine::dumpSongInfo() { | 
					
						
							|  |  |  |  |   printf( | 
					
						
							|  |  |  |  |     "SONG INFORMATION\n" | 
					
						
							|  |  |  |  |     "- name: %s\n" | 
					
						
							|  |  |  |  |     "- author: %s\n" | 
					
						
							|  |  |  |  |     "- album: %s\n" | 
					
						
							|  |  |  |  |     "- system: %s\n" | 
					
						
							|  |  |  |  |     "- %d ins, %d waves, %d samples\n" | 
					
						
							|  |  |  |  |     "<<<\n%s\n>>>\n\n", | 
					
						
							|  |  |  |  |     song.name.c_str(), | 
					
						
							|  |  |  |  |     song.author.c_str(), | 
					
						
							|  |  |  |  |     song.category.c_str(), | 
					
						
							|  |  |  |  |     song.systemName.c_str(), | 
					
						
							|  |  |  |  |     song.insLen, | 
					
						
							|  |  |  |  |     song.waveLen, | 
					
						
							|  |  |  |  |     song.sampleLen, | 
					
						
							|  |  |  |  |     song.notes.c_str() | 
					
						
							|  |  |  |  |   ); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   printf("SUB-SONGS\n"); | 
					
						
							|  |  |  |  |   int index=0; | 
					
						
							|  |  |  |  |   for (DivSubSong* i: song.subsong) { | 
					
						
							|  |  |  |  |     printf( | 
					
						
							|  |  |  |  |       "=== %d: %s\n" | 
					
						
							|  |  |  |  |       "<<<\n%s\n>>>\n", | 
					
						
							|  |  |  |  |       index, | 
					
						
							|  |  |  |  |       i->name.c_str(), | 
					
						
							|  |  |  |  |       i->notes.c_str() | 
					
						
							|  |  |  |  |     ); | 
					
						
							|  |  |  |  |     index++; | 
					
						
							|  |  |  |  |   } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   if (!song.ins.empty()) { | 
					
						
							|  |  |  |  |     printf("\nINSTRUMENTS\n"); | 
					
						
							|  |  |  |  |     index=0; | 
					
						
							|  |  |  |  |     for (DivInstrument* i: song.ins) { | 
					
						
							|  |  |  |  |       printf( | 
					
						
							|  |  |  |  |         "- %d: %s\n", | 
					
						
							|  |  |  |  |         index, | 
					
						
							|  |  |  |  |         i->name.c_str() | 
					
						
							|  |  |  |  |       ); | 
					
						
							|  |  |  |  |       index++; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  |   } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   if (!song.sample.empty()) { | 
					
						
							|  |  |  |  |     printf("\nSAMPLES\n"); | 
					
						
							|  |  |  |  |     index=0; | 
					
						
							|  |  |  |  |     for (DivSample* i: song.sample) { | 
					
						
							|  |  |  |  |       printf( | 
					
						
							|  |  |  |  |         "- %d: %s\n", | 
					
						
							|  |  |  |  |         index, | 
					
						
							|  |  |  |  |         i->name.c_str() | 
					
						
							|  |  |  |  |       ); | 
					
						
							|  |  |  |  |       index++; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  |   } | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-13 18:29:37 -05:00
										 |  |  |  | int DivEngine::addInstrument(int refChan, DivInstrumentType fallbackType) { | 
					
						
							| 
									
										
										
										
											2022-05-14 18:51:05 -04:00
										 |  |  |  |   if (song.ins.size()>=256) return -1; | 
					
						
							| 
									
										
										
										
											2022-03-23 22:38:28 -04:00
										 |  |  |  |   BUSY_BEGIN; | 
					
						
							| 
									
										
										
										
											2021-12-17 03:33:12 -05:00
										 |  |  |  |   DivInstrument* ins=new DivInstrument; | 
					
						
							|  |  |  |  |   int insCount=(int)song.ins.size(); | 
					
						
							| 
									
										
										
										
											2022-11-13 18:29:37 -05:00
										 |  |  |  |   DivInstrumentType prefType; | 
					
						
							| 
									
										
										
										
											2024-11-08 03:19:49 -05:00
										 |  |  |  |   if (refChan>chans) { | 
					
						
							|  |  |  |  |     refChan=chans-1; | 
					
						
							|  |  |  |  |   } | 
					
						
							| 
									
										
										
										
											2022-11-13 18:29:37 -05:00
										 |  |  |  |   if (refChan<0) { | 
					
						
							|  |  |  |  |     prefType=fallbackType; | 
					
						
							|  |  |  |  |   } else { | 
					
						
							|  |  |  |  |     prefType=getPreferInsType(refChan); | 
					
						
							|  |  |  |  |   } | 
					
						
							| 
									
										
										
										
											2022-04-21 03:29:20 -04:00
										 |  |  |  |   switch (prefType) { | 
					
						
							|  |  |  |  |     case DIV_INS_OPLL: | 
					
						
							|  |  |  |  |       *ins=song.nullInsOPLL; | 
					
						
							|  |  |  |  |       break; | 
					
						
							|  |  |  |  |     case DIV_INS_OPL: | 
					
						
							|  |  |  |  |       *ins=song.nullInsOPL; | 
					
						
							|  |  |  |  |       break; | 
					
						
							| 
									
										
										
										
											2022-05-31 18:44:52 -04:00
										 |  |  |  |     case DIV_INS_OPL_DRUMS: | 
					
						
							|  |  |  |  |       *ins=song.nullInsOPLDrums; | 
					
						
							|  |  |  |  |       break; | 
					
						
							| 
									
										
										
										
											2023-10-14 07:10:37 -04:00
										 |  |  |  |     case DIV_INS_ESFM: | 
					
						
							|  |  |  |  |       *ins=song.nullInsESFM; | 
					
						
							|  |  |  |  |       break; | 
					
						
							| 
									
										
										
										
											2022-04-21 03:29:20 -04:00
										 |  |  |  |     default: | 
					
						
							|  |  |  |  |       break; | 
					
						
							|  |  |  |  |   } | 
					
						
							| 
									
										
										
										
											2022-11-13 18:29:37 -05:00
										 |  |  |  |   if (refChan>=0) { | 
					
						
							|  |  |  |  |     if (sysOfChan[refChan]==DIV_SYSTEM_QSOUND) { | 
					
						
							|  |  |  |  |       *ins=song.nullInsQSound; | 
					
						
							|  |  |  |  |     } | 
					
						
							| 
									
										
										
										
											2022-04-29 00:58:11 -04:00
										 |  |  |  |   } | 
					
						
							| 
									
										
										
										
											2024-05-27 05:02:07 -04:00
										 |  |  |  |   ins->name=fmt::sprintf(_("Instrument %d"),insCount); | 
					
						
							| 
									
										
										
										
											2022-05-13 04:18:14 -04:00
										 |  |  |  |   if (prefType!=DIV_INS_NULL) { | 
					
						
							|  |  |  |  |     ins->type=prefType; | 
					
						
							|  |  |  |  |   } | 
					
						
							| 
									
										
										
										
											2022-03-21 17:17:51 -04:00
										 |  |  |  |   saveLock.lock(); | 
					
						
							| 
									
										
										
										
											2021-12-17 03:33:12 -05:00
										 |  |  |  |   song.ins.push_back(ins); | 
					
						
							|  |  |  |  |   song.insLen=insCount+1; | 
					
						
							| 
									
										
										
										
											2023-05-16 00:27:45 -04:00
										 |  |  |  |   checkAssetDir(song.insDir,song.ins.size()); | 
					
						
							| 
									
										
										
										
											2022-03-21 17:17:51 -04:00
										 |  |  |  |   saveLock.unlock(); | 
					
						
							| 
									
										
										
										
											2022-03-23 22:38:28 -04:00
										 |  |  |  |   BUSY_END; | 
					
						
							| 
									
										
										
										
											2021-12-17 03:33:12 -05:00
										 |  |  |  |   return insCount; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-03 02:56:49 -04:00
										 |  |  |  | int DivEngine::addInstrumentPtr(DivInstrument* which) { | 
					
						
							| 
									
										
										
										
											2022-05-14 18:51:05 -04:00
										 |  |  |  |   if (song.ins.size()>=256) { | 
					
						
							|  |  |  |  |     delete which; | 
					
						
							|  |  |  |  |     return -1; | 
					
						
							|  |  |  |  |   } | 
					
						
							| 
									
										
										
										
											2022-04-03 02:56:49 -04:00
										 |  |  |  |   BUSY_BEGIN; | 
					
						
							|  |  |  |  |   saveLock.lock(); | 
					
						
							|  |  |  |  |   song.ins.push_back(which); | 
					
						
							|  |  |  |  |   song.insLen=song.ins.size(); | 
					
						
							| 
									
										
										
										
											2023-05-16 00:27:45 -04:00
										 |  |  |  |   checkAssetDir(song.insDir,song.ins.size()); | 
					
						
							| 
									
										
										
										
											2023-08-11 20:32:08 -04:00
										 |  |  |  |   checkAssetDir(song.waveDir,song.wave.size()); | 
					
						
							|  |  |  |  |   checkAssetDir(song.sampleDir,song.sample.size()); | 
					
						
							| 
									
										
										
										
											2022-04-03 02:56:49 -04:00
										 |  |  |  |   saveLock.unlock(); | 
					
						
							|  |  |  |  |   BUSY_END; | 
					
						
							|  |  |  |  |   return song.insLen; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-25 19:23:12 -04:00
										 |  |  |  | void DivEngine::loadTempIns(DivInstrument* which) { | 
					
						
							|  |  |  |  |   BUSY_BEGIN; | 
					
						
							|  |  |  |  |   if (tempIns==NULL) { | 
					
						
							|  |  |  |  |     tempIns=new DivInstrument; | 
					
						
							|  |  |  |  |   } | 
					
						
							| 
									
										
										
										
											2022-04-25 19:41:24 -04:00
										 |  |  |  |   *tempIns=*which; | 
					
						
							| 
									
										
										
										
											2022-04-25 19:23:12 -04:00
										 |  |  |  |   BUSY_END; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-09-08 03:52:15 -04:00
										 |  |  |  | void DivEngine::delInstrumentUnsafe(int index) { | 
					
						
							| 
									
										
										
										
											2021-12-17 03:33:12 -05:00
										 |  |  |  |   if (index>=0 && index<(int)song.ins.size()) { | 
					
						
							| 
									
										
										
										
											2022-01-13 17:40:29 -05:00
										 |  |  |  |     for (int i=0; i<song.systemLen; i++) { | 
					
						
							|  |  |  |  |       disCont[i].dispatch->notifyInsDeletion(song.ins[index]); | 
					
						
							|  |  |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-12-17 03:33:12 -05:00
										 |  |  |  |     delete song.ins[index]; | 
					
						
							|  |  |  |  |     song.ins.erase(song.ins.begin()+index); | 
					
						
							|  |  |  |  |     song.insLen=song.ins.size(); | 
					
						
							| 
									
										
										
										
											2022-02-07 23:42:54 -05:00
										 |  |  |  |     for (int i=0; i<chans; i++) { | 
					
						
							| 
									
										
										
										
											2022-08-15 23:54:31 -04:00
										 |  |  |  |       for (size_t j=0; j<song.subsong.size(); j++) { | 
					
						
							| 
									
										
										
										
											2022-12-09 00:30:27 -05:00
										 |  |  |  |         for (int k=0; k<DIV_MAX_PATTERNS; k++) { | 
					
						
							| 
									
										
										
										
											2022-08-15 23:54:31 -04:00
										 |  |  |  |           if (song.subsong[j]->pat[i].data[k]==NULL) continue; | 
					
						
							|  |  |  |  |           for (int l=0; l<song.subsong[j]->patLen; l++) { | 
					
						
							|  |  |  |  |             if (song.subsong[j]->pat[i].data[k]->data[l][2]>index) { | 
					
						
							|  |  |  |  |               song.subsong[j]->pat[i].data[k]->data[l][2]--; | 
					
						
							|  |  |  |  |             } | 
					
						
							| 
									
										
										
										
											2022-02-07 23:42:54 -05:00
										 |  |  |  |           } | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  |       } | 
					
						
							|  |  |  |  |     } | 
					
						
							| 
									
										
										
										
											2023-05-16 00:27:45 -04:00
										 |  |  |  |     removeAsset(song.insDir,index); | 
					
						
							|  |  |  |  |     checkAssetDir(song.insDir,song.ins.size()); | 
					
						
							| 
									
										
										
										
											2021-12-17 03:33:12 -05:00
										 |  |  |  |   } | 
					
						
							| 
									
										
										
										
											2023-09-08 03:52:15 -04:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | void DivEngine::delInstrument(int index) { | 
					
						
							|  |  |  |  |   BUSY_BEGIN; | 
					
						
							|  |  |  |  |   saveLock.lock(); | 
					
						
							|  |  |  |  |   delInstrumentUnsafe(index); | 
					
						
							| 
									
										
										
										
											2022-03-21 17:17:51 -04:00
										 |  |  |  |   saveLock.unlock(); | 
					
						
							| 
									
										
										
										
											2022-03-23 22:38:28 -04:00
										 |  |  |  |   BUSY_END; | 
					
						
							| 
									
										
										
										
											2021-12-17 03:33:12 -05:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | int DivEngine::addWave() { | 
					
						
							| 
									
										
										
										
											2022-08-13 05:17:32 -04:00
										 |  |  |  |   if (song.wave.size()>=256) { | 
					
						
							| 
									
										
										
										
											2024-05-27 05:02:07 -04:00
										 |  |  |  |     lastError=_("too many wavetables!"); | 
					
						
							| 
									
										
										
										
											2022-08-13 05:17:32 -04:00
										 |  |  |  |     return -1; | 
					
						
							|  |  |  |  |   } | 
					
						
							| 
									
										
										
										
											2022-03-23 22:38:28 -04:00
										 |  |  |  |   BUSY_BEGIN; | 
					
						
							| 
									
										
										
										
											2022-03-21 17:17:51 -04:00
										 |  |  |  |   saveLock.lock(); | 
					
						
							| 
									
										
										
										
											2021-12-17 03:33:12 -05:00
										 |  |  |  |   DivWavetable* wave=new DivWavetable; | 
					
						
							|  |  |  |  |   int waveCount=(int)song.wave.size(); | 
					
						
							|  |  |  |  |   song.wave.push_back(wave); | 
					
						
							|  |  |  |  |   song.waveLen=waveCount+1; | 
					
						
							| 
									
										
										
										
											2023-05-16 00:27:45 -04:00
										 |  |  |  |   checkAssetDir(song.waveDir,song.wave.size()); | 
					
						
							| 
									
										
										
										
											2022-03-21 17:17:51 -04:00
										 |  |  |  |   saveLock.unlock(); | 
					
						
							| 
									
										
										
										
											2022-03-23 22:38:28 -04:00
										 |  |  |  |   BUSY_END; | 
					
						
							| 
									
										
										
										
											2021-12-17 03:33:12 -05:00
										 |  |  |  |   return waveCount; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-13 05:17:32 -04:00
										 |  |  |  | int DivEngine::addWavePtr(DivWavetable* which) { | 
					
						
							| 
									
										
										
										
											2022-05-14 18:51:05 -04:00
										 |  |  |  |   if (song.wave.size()>=256) { | 
					
						
							| 
									
										
										
										
											2024-05-27 05:02:07 -04:00
										 |  |  |  |     lastError=_("too many wavetables!"); | 
					
						
							| 
									
										
										
										
											2022-08-13 05:17:32 -04:00
										 |  |  |  |     delete which; | 
					
						
							|  |  |  |  |     return -1; | 
					
						
							| 
									
										
										
										
											2022-05-14 18:51:05 -04:00
										 |  |  |  |   } | 
					
						
							| 
									
										
										
										
											2022-08-13 05:17:32 -04:00
										 |  |  |  |   BUSY_BEGIN; | 
					
						
							|  |  |  |  |   saveLock.lock(); | 
					
						
							|  |  |  |  |   int waveCount=(int)song.wave.size(); | 
					
						
							|  |  |  |  |   song.wave.push_back(which); | 
					
						
							|  |  |  |  |   song.waveLen=waveCount+1; | 
					
						
							| 
									
										
										
										
											2023-05-16 00:27:45 -04:00
										 |  |  |  |   checkAssetDir(song.waveDir,song.wave.size()); | 
					
						
							| 
									
										
										
										
											2022-08-13 05:17:32 -04:00
										 |  |  |  |   saveLock.unlock(); | 
					
						
							|  |  |  |  |   BUSY_END; | 
					
						
							|  |  |  |  |   return song.waveLen; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | DivWavetable* DivEngine::waveFromFile(const char* path, bool addRaw) { | 
					
						
							| 
									
										
										
										
											2022-01-21 17:59:48 -05:00
										 |  |  |  |   FILE* f=ps_fopen(path,"rb"); | 
					
						
							|  |  |  |  |   if (f==NULL) { | 
					
						
							| 
									
										
										
										
											2022-05-14 18:51:05 -04:00
										 |  |  |  |     lastError=fmt::sprintf("%s",strerror(errno)); | 
					
						
							| 
									
										
										
										
											2022-08-13 05:17:32 -04:00
										 |  |  |  |     return NULL; | 
					
						
							| 
									
										
										
										
											2022-01-21 17:59:48 -05:00
										 |  |  |  |   } | 
					
						
							|  |  |  |  |   unsigned char* buf; | 
					
						
							|  |  |  |  |   ssize_t len; | 
					
						
							|  |  |  |  |   if (fseek(f,0,SEEK_END)!=0) { | 
					
						
							|  |  |  |  |     fclose(f); | 
					
						
							| 
									
										
										
										
											2024-05-27 05:02:07 -04:00
										 |  |  |  |     lastError=fmt::sprintf(_("could not seek to end: %s"),strerror(errno)); | 
					
						
							| 
									
										
										
										
											2022-08-13 05:17:32 -04:00
										 |  |  |  |     return NULL; | 
					
						
							| 
									
										
										
										
											2022-01-21 17:59:48 -05:00
										 |  |  |  |   } | 
					
						
							|  |  |  |  |   len=ftell(f); | 
					
						
							|  |  |  |  |   if (len<0) { | 
					
						
							|  |  |  |  |     fclose(f); | 
					
						
							| 
									
										
										
										
											2024-05-27 05:02:07 -04:00
										 |  |  |  |     lastError=fmt::sprintf(_("could not determine file size: %s"),strerror(errno)); | 
					
						
							| 
									
										
										
										
											2022-08-13 05:17:32 -04:00
										 |  |  |  |     return NULL; | 
					
						
							| 
									
										
										
										
											2022-01-21 17:59:48 -05:00
										 |  |  |  |   } | 
					
						
							| 
									
										
										
										
											2022-04-26 02:07:28 -04:00
										 |  |  |  |   if (len==(SIZE_MAX>>1)) { | 
					
						
							|  |  |  |  |     fclose(f); | 
					
						
							| 
									
										
										
										
											2024-05-27 05:02:07 -04:00
										 |  |  |  |     lastError=_("file size is invalid!"); | 
					
						
							| 
									
										
										
										
											2022-08-13 05:17:32 -04:00
										 |  |  |  |     return NULL; | 
					
						
							| 
									
										
										
										
											2022-04-26 02:07:28 -04:00
										 |  |  |  |   } | 
					
						
							| 
									
										
										
										
											2022-01-21 17:59:48 -05:00
										 |  |  |  |   if (len==0) { | 
					
						
							|  |  |  |  |     fclose(f); | 
					
						
							| 
									
										
										
										
											2024-05-27 05:02:07 -04:00
										 |  |  |  |     lastError=_("file is empty"); | 
					
						
							| 
									
										
										
										
											2022-08-13 05:17:32 -04:00
										 |  |  |  |     return NULL; | 
					
						
							| 
									
										
										
										
											2022-01-21 17:59:48 -05:00
										 |  |  |  |   } | 
					
						
							|  |  |  |  |   if (fseek(f,0,SEEK_SET)!=0) { | 
					
						
							|  |  |  |  |     fclose(f); | 
					
						
							| 
									
										
										
										
											2024-05-27 05:02:07 -04:00
										 |  |  |  |     lastError=fmt::sprintf(_("could not seek to beginning: %s"),strerror(errno)); | 
					
						
							| 
									
										
										
										
											2022-08-13 05:17:32 -04:00
										 |  |  |  |     return NULL; | 
					
						
							| 
									
										
										
										
											2022-01-21 17:59:48 -05:00
										 |  |  |  |   } | 
					
						
							|  |  |  |  |   buf=new unsigned char[len]; | 
					
						
							|  |  |  |  |   if (fread(buf,1,len,f)!=(size_t)len) { | 
					
						
							| 
									
										
										
										
											2022-04-10 23:12:02 -04:00
										 |  |  |  |     logW("did not read entire wavetable file buffer!"); | 
					
						
							| 
									
										
										
										
											2022-01-21 17:59:48 -05:00
										 |  |  |  |     delete[] buf; | 
					
						
							| 
									
										
										
										
											2024-05-27 05:02:07 -04:00
										 |  |  |  |     lastError=fmt::sprintf(_("could not read entire file: %s"),strerror(errno)); | 
					
						
							| 
									
										
										
										
											2022-08-13 05:17:32 -04:00
										 |  |  |  |     return NULL; | 
					
						
							| 
									
										
										
										
											2022-01-21 17:59:48 -05:00
										 |  |  |  |   } | 
					
						
							|  |  |  |  |   fclose(f); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   SafeReader reader=SafeReader(buf,len); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   unsigned char magic[16]; | 
					
						
							|  |  |  |  |   bool isFurnaceTable=false; | 
					
						
							|  |  |  |  |   try { | 
					
						
							|  |  |  |  |     reader.read(magic,16); | 
					
						
							|  |  |  |  |     if (memcmp("-Furnace waveta-",magic,16)==0) { | 
					
						
							|  |  |  |  |       isFurnaceTable=true; | 
					
						
							|  |  |  |  |     } | 
					
						
							| 
									
										
										
										
											2022-03-19 02:13:00 -04:00
										 |  |  |  |   } catch (EndOfFileException& e) { | 
					
						
							| 
									
										
										
										
											2022-01-21 17:59:48 -05:00
										 |  |  |  |     reader.seek(0,SEEK_SET); | 
					
						
							|  |  |  |  |   } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   DivWavetable* wave=new DivWavetable; | 
					
						
							|  |  |  |  |   try { | 
					
						
							|  |  |  |  |     if (isFurnaceTable) { | 
					
						
							|  |  |  |  |       reader.seek(16,SEEK_SET); | 
					
						
							|  |  |  |  |       short version=reader.readS(); | 
					
						
							|  |  |  |  |       reader.readS(); // reserved
 | 
					
						
							| 
									
										
										
										
											2022-01-21 18:17:05 -05:00
										 |  |  |  |       reader.seek(20,SEEK_SET); | 
					
						
							|  |  |  |  |       if (wave->readWaveData(reader,version)!=DIV_DATA_SUCCESS) { | 
					
						
							| 
									
										
										
										
											2024-05-27 05:02:07 -04:00
										 |  |  |  |         lastError=_("invalid wavetable header/data!"); | 
					
						
							| 
									
										
										
										
											2022-01-21 18:17:05 -05:00
										 |  |  |  |         delete wave; | 
					
						
							| 
									
										
										
										
											2022-01-21 17:59:48 -05:00
										 |  |  |  |         delete[] buf; | 
					
						
							| 
									
										
										
										
											2022-08-13 05:17:32 -04:00
										 |  |  |  |         return NULL; | 
					
						
							| 
									
										
										
										
											2022-01-21 17:59:48 -05:00
										 |  |  |  |       } | 
					
						
							|  |  |  |  |     } else { | 
					
						
							|  |  |  |  |       try { | 
					
						
							|  |  |  |  |         // read as .dmw
 | 
					
						
							|  |  |  |  |         reader.seek(0,SEEK_SET); | 
					
						
							|  |  |  |  |         int len=reader.readI(); | 
					
						
							| 
									
										
										
										
											2023-05-05 00:11:44 -04:00
										 |  |  |  |         logD("wave length %d",len); | 
					
						
							| 
									
										
										
										
											2022-05-29 23:21:37 -04:00
										 |  |  |  |         if (len<=0 || len>256) { | 
					
						
							|  |  |  |  |           throw EndOfFileException(&reader,reader.size()); | 
					
						
							|  |  |  |  |         } | 
					
						
							| 
									
										
										
										
											2023-05-05 00:11:44 -04:00
										 |  |  |  |         wave->len=len; | 
					
						
							| 
									
										
										
										
											2022-01-21 17:59:48 -05:00
										 |  |  |  |         wave->max=(unsigned char)reader.readC(); | 
					
						
							| 
									
										
										
										
											2022-02-20 22:16:43 -05:00
										 |  |  |  |         if (wave->max==255) { // new wavetable format
 | 
					
						
							|  |  |  |  |           unsigned char waveVersion=reader.readC(); | 
					
						
							| 
									
										
										
										
											2022-04-10 23:12:02 -04:00
										 |  |  |  |           logI("reading modern .dmw..."); | 
					
						
							|  |  |  |  |           logD("wave version %d",waveVersion); | 
					
						
							| 
									
										
										
										
											2023-05-05 00:11:44 -04:00
										 |  |  |  |           wave->max=(unsigned char)reader.readC(); | 
					
						
							| 
									
										
										
										
											2022-02-20 22:16:43 -05:00
										 |  |  |  |           for (int i=0; i<len; i++) { | 
					
						
							|  |  |  |  |             wave->data[i]=reader.readI(); | 
					
						
							|  |  |  |  |           } | 
					
						
							|  |  |  |  |         } else if (reader.size()==(size_t)(len+5)) { | 
					
						
							| 
									
										
										
										
											2022-01-21 17:59:48 -05:00
										 |  |  |  |           // read as .dmw
 | 
					
						
							| 
									
										
										
										
											2022-04-10 23:12:02 -04:00
										 |  |  |  |           logI("reading .dmw..."); | 
					
						
							| 
									
										
										
										
											2022-01-21 17:59:48 -05:00
										 |  |  |  |           if (len>256) len=256; | 
					
						
							|  |  |  |  |           for (int i=0; i<len; i++) { | 
					
						
							|  |  |  |  |             wave->data[i]=(unsigned char)reader.readC(); | 
					
						
							|  |  |  |  |           } | 
					
						
							|  |  |  |  |         } else { | 
					
						
							|  |  |  |  |           // read as binary
 | 
					
						
							| 
									
										
										
										
											2022-05-28 19:51:05 -04:00
										 |  |  |  |           if (addRaw) { | 
					
						
							|  |  |  |  |             logI("reading binary..."); | 
					
						
							|  |  |  |  |             len=reader.size(); | 
					
						
							|  |  |  |  |             if (len>256) len=256; | 
					
						
							|  |  |  |  |             reader.seek(0,SEEK_SET); | 
					
						
							|  |  |  |  |             for (int i=0; i<len; i++) { | 
					
						
							|  |  |  |  |               wave->data[i]=(unsigned char)reader.readC(); | 
					
						
							|  |  |  |  |               if (wave->max<wave->data[i]) wave->max=wave->data[i]; | 
					
						
							|  |  |  |  |             } | 
					
						
							|  |  |  |  |             wave->len=len; | 
					
						
							|  |  |  |  |           } else { | 
					
						
							|  |  |  |  |             delete wave; | 
					
						
							|  |  |  |  |             delete[] buf; | 
					
						
							| 
									
										
										
										
											2022-08-13 05:17:32 -04:00
										 |  |  |  |             return NULL; | 
					
						
							| 
									
										
										
										
											2022-05-28 19:51:05 -04:00
										 |  |  |  |           } | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  |       } catch (EndOfFileException& e) { | 
					
						
							|  |  |  |  |         // read as binary
 | 
					
						
							|  |  |  |  |         if (addRaw) { | 
					
						
							| 
									
										
										
										
											2022-01-21 17:59:48 -05:00
										 |  |  |  |           len=reader.size(); | 
					
						
							| 
									
										
										
										
											2022-05-28 19:51:05 -04:00
										 |  |  |  |           logI("reading binary for being too small..."); | 
					
						
							| 
									
										
										
										
											2022-01-21 17:59:48 -05:00
										 |  |  |  |           if (len>256) len=256; | 
					
						
							|  |  |  |  |           reader.seek(0,SEEK_SET); | 
					
						
							|  |  |  |  |           for (int i=0; i<len; i++) { | 
					
						
							|  |  |  |  |             wave->data[i]=(unsigned char)reader.readC(); | 
					
						
							|  |  |  |  |             if (wave->max<wave->data[i]) wave->max=wave->data[i]; | 
					
						
							|  |  |  |  |           } | 
					
						
							|  |  |  |  |           wave->len=len; | 
					
						
							| 
									
										
										
										
											2022-05-28 19:51:05 -04:00
										 |  |  |  |         } else { | 
					
						
							|  |  |  |  |           delete wave; | 
					
						
							|  |  |  |  |           delete[] buf; | 
					
						
							| 
									
										
										
										
											2022-08-13 05:17:32 -04:00
										 |  |  |  |           return NULL; | 
					
						
							| 
									
										
										
										
											2022-01-21 17:59:48 -05:00
										 |  |  |  |         } | 
					
						
							|  |  |  |  |       } | 
					
						
							|  |  |  |  |     } | 
					
						
							| 
									
										
										
										
											2022-03-19 02:13:00 -04:00
										 |  |  |  |   } catch (EndOfFileException& e) { | 
					
						
							| 
									
										
										
										
											2022-01-21 17:59:48 -05:00
										 |  |  |  |     delete wave; | 
					
						
							|  |  |  |  |     delete[] buf; | 
					
						
							| 
									
										
										
										
											2024-05-27 05:02:07 -04:00
										 |  |  |  |     lastError=_("premature end of file"); | 
					
						
							| 
									
										
										
										
											2022-08-13 05:17:32 -04:00
										 |  |  |  |     return NULL; | 
					
						
							| 
									
										
										
										
											2022-01-21 17:59:48 -05:00
										 |  |  |  |   } | 
					
						
							|  |  |  |  |    | 
					
						
							| 
									
										
										
										
											2022-08-13 05:17:32 -04:00
										 |  |  |  |   return wave; | 
					
						
							| 
									
										
										
										
											2022-01-11 16:25:55 -05:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-09-08 03:52:15 -04:00
										 |  |  |  | void DivEngine::delWaveUnsafe(int index) { | 
					
						
							| 
									
										
										
										
											2021-12-17 03:33:12 -05:00
										 |  |  |  |   if (index>=0 && index<(int)song.wave.size()) { | 
					
						
							|  |  |  |  |     delete song.wave[index]; | 
					
						
							|  |  |  |  |     song.wave.erase(song.wave.begin()+index); | 
					
						
							|  |  |  |  |     song.waveLen=song.wave.size(); | 
					
						
							| 
									
										
										
										
											2023-05-16 00:27:45 -04:00
										 |  |  |  |     removeAsset(song.waveDir,index); | 
					
						
							|  |  |  |  |     checkAssetDir(song.waveDir,song.wave.size()); | 
					
						
							| 
									
										
										
										
											2021-12-17 03:33:12 -05:00
										 |  |  |  |   } | 
					
						
							| 
									
										
										
										
											2023-09-08 03:52:15 -04:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | void DivEngine::delWave(int index) { | 
					
						
							|  |  |  |  |   BUSY_BEGIN; | 
					
						
							|  |  |  |  |   saveLock.lock(); | 
					
						
							|  |  |  |  |   delWaveUnsafe(index); | 
					
						
							| 
									
										
										
										
											2022-03-21 17:17:51 -04:00
										 |  |  |  |   saveLock.unlock(); | 
					
						
							| 
									
										
										
										
											2022-03-23 22:38:28 -04:00
										 |  |  |  |   BUSY_END; | 
					
						
							| 
									
										
										
										
											2021-12-17 03:33:12 -05:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | int DivEngine::addSample() { | 
					
						
							| 
									
										
										
										
											2022-08-13 05:17:32 -04:00
										 |  |  |  |   if (song.sample.size()>=256) { | 
					
						
							| 
									
										
										
										
											2024-05-27 05:02:07 -04:00
										 |  |  |  |     lastError=_("too many samples!"); | 
					
						
							| 
									
										
										
										
											2022-08-13 05:17:32 -04:00
										 |  |  |  |     return -1; | 
					
						
							|  |  |  |  |   } | 
					
						
							| 
									
										
										
										
											2022-03-23 22:38:28 -04:00
										 |  |  |  |   BUSY_BEGIN; | 
					
						
							| 
									
										
										
										
											2022-03-21 17:17:51 -04:00
										 |  |  |  |   saveLock.lock(); | 
					
						
							| 
									
										
										
										
											2021-12-17 03:33:12 -05:00
										 |  |  |  |   DivSample* sample=new DivSample; | 
					
						
							|  |  |  |  |   int sampleCount=(int)song.sample.size(); | 
					
						
							| 
									
										
										
										
											2024-05-27 05:02:07 -04:00
										 |  |  |  |   sample->name=fmt::sprintf(_("Sample %d"),sampleCount); | 
					
						
							| 
									
										
										
										
											2025-02-06 02:14:21 -05:00
										 |  |  |  |   sample->centerRate=getCenterRate(); | 
					
						
							| 
									
										
										
										
											2021-12-17 03:33:12 -05:00
										 |  |  |  |   song.sample.push_back(sample); | 
					
						
							|  |  |  |  |   song.sampleLen=sampleCount+1; | 
					
						
							| 
									
										
										
										
											2022-06-17 00:39:38 -04:00
										 |  |  |  |   sPreview.sample=-1; | 
					
						
							|  |  |  |  |   sPreview.pos=0; | 
					
						
							| 
									
										
										
										
											2022-08-11 09:21:54 -04:00
										 |  |  |  |   sPreview.dir=false; | 
					
						
							| 
									
										
										
										
											2023-05-16 00:27:45 -04:00
										 |  |  |  |   checkAssetDir(song.sampleDir,song.sample.size()); | 
					
						
							| 
									
										
										
										
											2022-03-21 17:17:51 -04:00
										 |  |  |  |   saveLock.unlock(); | 
					
						
							| 
									
										
										
										
											2021-12-17 03:33:12 -05:00
										 |  |  |  |   renderSamples(); | 
					
						
							| 
									
										
										
										
											2022-03-23 22:38:28 -04:00
										 |  |  |  |   BUSY_END; | 
					
						
							| 
									
										
										
										
											2021-12-17 03:33:12 -05:00
										 |  |  |  |   return sampleCount; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-13 05:17:32 -04:00
										 |  |  |  | int DivEngine::addSamplePtr(DivSample* which) { | 
					
						
							| 
									
										
										
										
											2022-05-14 18:51:05 -04:00
										 |  |  |  |   if (song.sample.size()>=256) { | 
					
						
							| 
									
										
										
										
											2024-05-27 05:02:07 -04:00
										 |  |  |  |     lastError=_("too many samples!"); | 
					
						
							| 
									
										
										
										
											2022-08-13 05:17:32 -04:00
										 |  |  |  |     delete which; | 
					
						
							| 
									
										
										
										
											2022-05-14 18:51:05 -04:00
										 |  |  |  |     return -1; | 
					
						
							|  |  |  |  |   } | 
					
						
							| 
									
										
										
										
											2022-08-13 05:17:32 -04:00
										 |  |  |  |   int sampleCount=(int)song.sample.size(); | 
					
						
							|  |  |  |  |   BUSY_BEGIN; | 
					
						
							|  |  |  |  |   saveLock.lock(); | 
					
						
							|  |  |  |  |   song.sample.push_back(which); | 
					
						
							|  |  |  |  |   song.sampleLen=sampleCount+1; | 
					
						
							| 
									
										
										
										
											2023-05-16 00:27:45 -04:00
										 |  |  |  |   checkAssetDir(song.sampleDir,song.sample.size()); | 
					
						
							| 
									
										
										
										
											2022-08-13 05:17:32 -04:00
										 |  |  |  |   saveLock.unlock(); | 
					
						
							|  |  |  |  |   renderSamples(); | 
					
						
							|  |  |  |  |   BUSY_END; | 
					
						
							|  |  |  |  |   return sampleCount; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-09-08 03:52:15 -04:00
										 |  |  |  | void DivEngine::delSampleUnsafe(int index, bool render) { | 
					
						
							| 
									
										
										
										
											2022-06-17 00:39:38 -04:00
										 |  |  |  |   sPreview.sample=-1; | 
					
						
							|  |  |  |  |   sPreview.pos=0; | 
					
						
							| 
									
										
										
										
											2022-08-11 09:21:54 -04:00
										 |  |  |  |   sPreview.dir=false; | 
					
						
							| 
									
										
										
										
											2021-12-17 03:33:12 -05:00
										 |  |  |  |   if (index>=0 && index<(int)song.sample.size()) { | 
					
						
							|  |  |  |  |     delete song.sample[index]; | 
					
						
							|  |  |  |  |     song.sample.erase(song.sample.begin()+index); | 
					
						
							|  |  |  |  |     song.sampleLen=song.sample.size(); | 
					
						
							| 
									
										
										
										
											2023-05-16 00:27:45 -04:00
										 |  |  |  |     removeAsset(song.sampleDir,index); | 
					
						
							|  |  |  |  |     checkAssetDir(song.sampleDir,song.sample.size()); | 
					
						
							| 
									
										
										
										
											2023-07-26 18:58:57 -04:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |     // compensate
 | 
					
						
							|  |  |  |  |     for (DivInstrument* i: song.ins) { | 
					
						
							|  |  |  |  |       if (i->amiga.initSample==index) { | 
					
						
							|  |  |  |  |         i->amiga.initSample=-1; | 
					
						
							|  |  |  |  |       } else if (i->amiga.initSample>index) { | 
					
						
							|  |  |  |  |         i->amiga.initSample--; | 
					
						
							|  |  |  |  |       } | 
					
						
							|  |  |  |  |       for (int j=0; j<120; j++) { | 
					
						
							|  |  |  |  |         if (i->amiga.noteMap[j].map==index) { | 
					
						
							|  |  |  |  |           i->amiga.noteMap[j].map=-1; | 
					
						
							|  |  |  |  |         } else if (i->amiga.noteMap[j].map>index) { | 
					
						
							|  |  |  |  |           i->amiga.noteMap[j].map--; | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  |       } | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-09-08 03:52:15 -04:00
										 |  |  |  |     if (render) renderSamples(); | 
					
						
							| 
									
										
										
										
											2021-12-17 03:33:12 -05:00
										 |  |  |  |   } | 
					
						
							| 
									
										
										
										
											2023-09-08 03:52:15 -04:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | void DivEngine::delSample(int index) { | 
					
						
							|  |  |  |  |   BUSY_BEGIN; | 
					
						
							|  |  |  |  |   saveLock.lock(); | 
					
						
							|  |  |  |  |   delSampleUnsafe(index); | 
					
						
							| 
									
										
										
										
											2022-03-21 17:17:51 -04:00
										 |  |  |  |   saveLock.unlock(); | 
					
						
							| 
									
										
										
										
											2022-03-23 22:38:28 -04:00
										 |  |  |  |   BUSY_END; | 
					
						
							| 
									
										
										
										
											2021-12-17 03:33:12 -05:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-05 19:06:13 -05:00
										 |  |  |  | void DivEngine::addOrder(int pos, bool duplicate, bool where) { | 
					
						
							| 
									
										
										
										
											2022-01-09 22:17:03 -05:00
										 |  |  |  |   unsigned char order[DIV_MAX_CHANS]; | 
					
						
							| 
									
										
										
										
											2022-12-09 00:30:27 -05:00
										 |  |  |  |   if (curSubSong->ordersLen>=(DIV_MAX_PATTERNS-1)) return; | 
					
						
							| 
									
										
										
										
											2022-05-30 15:02:54 -04:00
										 |  |  |  |   memset(order,0,DIV_MAX_CHANS); | 
					
						
							| 
									
										
										
										
											2022-03-23 22:38:28 -04:00
										 |  |  |  |   BUSY_BEGIN_SOFT; | 
					
						
							| 
									
										
										
										
											2021-12-22 17:39:16 -05:00
										 |  |  |  |   if (duplicate) { | 
					
						
							| 
									
										
										
										
											2022-01-08 17:15:12 -05:00
										 |  |  |  |     for (int i=0; i<DIV_MAX_CHANS; i++) { | 
					
						
							| 
									
										
										
										
											2023-02-05 19:06:13 -05:00
										 |  |  |  |       order[i]=curOrders->ord[i][pos]; | 
					
						
							| 
									
										
										
										
											2021-12-22 17:39:16 -05:00
										 |  |  |  |     } | 
					
						
							|  |  |  |  |   } else { | 
					
						
							| 
									
										
										
										
											2022-12-09 00:30:27 -05:00
										 |  |  |  |     bool used[DIV_MAX_PATTERNS]; | 
					
						
							| 
									
										
										
										
											2021-12-22 17:39:16 -05:00
										 |  |  |  |     for (int i=0; i<chans; i++) { | 
					
						
							| 
									
										
										
										
											2022-12-09 00:30:27 -05:00
										 |  |  |  |       memset(used,0,sizeof(bool)*DIV_MAX_PATTERNS); | 
					
						
							| 
									
										
										
										
											2022-05-15 02:42:49 -04:00
										 |  |  |  |       for (int j=0; j<curSubSong->ordersLen; j++) { | 
					
						
							|  |  |  |  |         used[curOrders->ord[i][j]]=true; | 
					
						
							| 
									
										
										
										
											2021-12-22 17:39:16 -05:00
										 |  |  |  |       } | 
					
						
							| 
									
										
										
										
											2022-12-09 00:30:27 -05:00
										 |  |  |  |       order[i]=(DIV_MAX_PATTERNS-1); | 
					
						
							|  |  |  |  |       for (int j=0; j<DIV_MAX_PATTERNS; j++) { | 
					
						
							| 
									
										
										
										
											2021-12-22 17:39:16 -05:00
										 |  |  |  |         if (!used[j]) { | 
					
						
							|  |  |  |  |           order[i]=j; | 
					
						
							|  |  |  |  |           break; | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  |       } | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  |   } | 
					
						
							|  |  |  |  |   if (where) { // at the end
 | 
					
						
							| 
									
										
										
										
											2022-03-21 17:17:51 -04:00
										 |  |  |  |     saveLock.lock(); | 
					
						
							| 
									
										
										
										
											2022-01-08 17:15:12 -05:00
										 |  |  |  |     for (int i=0; i<DIV_MAX_CHANS; i++) { | 
					
						
							| 
									
										
										
										
											2022-05-15 02:42:49 -04:00
										 |  |  |  |       curOrders->ord[i][curSubSong->ordersLen]=order[i]; | 
					
						
							| 
									
										
										
										
											2021-12-22 17:39:16 -05:00
										 |  |  |  |     } | 
					
						
							| 
									
										
										
										
											2022-05-15 02:42:49 -04:00
										 |  |  |  |     curSubSong->ordersLen++; | 
					
						
							| 
									
										
										
										
											2022-03-21 17:17:51 -04:00
										 |  |  |  |     saveLock.unlock(); | 
					
						
							| 
									
										
										
										
											2021-12-22 17:39:16 -05:00
										 |  |  |  |   } else { // after current order
 | 
					
						
							| 
									
										
										
										
											2022-03-21 17:17:51 -04:00
										 |  |  |  |     saveLock.lock(); | 
					
						
							| 
									
										
										
										
											2022-01-08 17:15:12 -05:00
										 |  |  |  |     for (int i=0; i<DIV_MAX_CHANS; i++) { | 
					
						
							| 
									
										
										
										
											2023-02-05 19:06:13 -05:00
										 |  |  |  |       for (int j=curSubSong->ordersLen; j>pos; j--) { | 
					
						
							| 
									
										
										
										
											2022-05-15 02:42:49 -04:00
										 |  |  |  |         curOrders->ord[i][j]=curOrders->ord[i][j-1]; | 
					
						
							| 
									
										
										
										
											2021-12-22 17:39:16 -05:00
										 |  |  |  |       } | 
					
						
							| 
									
										
										
										
											2023-02-05 19:06:13 -05:00
										 |  |  |  |       curOrders->ord[i][pos+1]=order[i]; | 
					
						
							| 
									
										
										
										
											2021-12-22 17:39:16 -05:00
										 |  |  |  |     } | 
					
						
							| 
									
										
										
										
											2022-05-15 02:42:49 -04:00
										 |  |  |  |     curSubSong->ordersLen++; | 
					
						
							| 
									
										
										
										
											2022-03-21 17:17:51 -04:00
										 |  |  |  |     saveLock.unlock(); | 
					
						
							| 
									
										
										
										
											2023-03-09 00:36:18 -05:00
										 |  |  |  |     curOrder=pos+1; | 
					
						
							|  |  |  |  |     prevOrder=curOrder; | 
					
						
							| 
									
										
										
										
											2021-12-28 18:23:57 -05:00
										 |  |  |  |     if (playing && !freelance) { | 
					
						
							| 
									
										
										
										
											2021-12-22 17:39:16 -05:00
										 |  |  |  |       playSub(false); | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  |   } | 
					
						
							| 
									
										
										
										
											2022-03-23 22:38:28 -04:00
										 |  |  |  |   BUSY_END; | 
					
						
							| 
									
										
										
										
											2021-12-22 17:39:16 -05:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-05 19:06:13 -05:00
										 |  |  |  | void DivEngine::deepCloneOrder(int pos, bool where) { | 
					
						
							| 
									
										
										
										
											2022-02-12 03:59:05 -05:00
										 |  |  |  |   unsigned char order[DIV_MAX_CHANS]; | 
					
						
							| 
									
										
										
										
											2022-12-09 00:30:27 -05:00
										 |  |  |  |   if (curSubSong->ordersLen>=(DIV_MAX_PATTERNS-1)) return; | 
					
						
							| 
									
										
										
										
											2022-02-12 18:02:33 -05:00
										 |  |  |  |   warnings=""; | 
					
						
							| 
									
										
										
										
											2022-03-23 22:38:28 -04:00
										 |  |  |  |   BUSY_BEGIN_SOFT; | 
					
						
							| 
									
										
										
										
											2022-02-12 03:59:05 -05:00
										 |  |  |  |   for (int i=0; i<chans; i++) { | 
					
						
							| 
									
										
										
										
											2022-02-20 23:07:46 -05:00
										 |  |  |  |     bool didNotFind=true; | 
					
						
							| 
									
										
										
										
											2022-04-10 23:12:02 -04:00
										 |  |  |  |     logD("channel %d",i); | 
					
						
							| 
									
										
										
										
											2023-02-05 19:06:13 -05:00
										 |  |  |  |     order[i]=curOrders->ord[i][pos]; | 
					
						
							| 
									
										
										
										
											2022-02-12 03:59:05 -05:00
										 |  |  |  |     // find free slot
 | 
					
						
							| 
									
										
										
										
											2022-12-09 00:30:27 -05:00
										 |  |  |  |     for (int j=0; j<DIV_MAX_PATTERNS; j++) { | 
					
						
							| 
									
										
										
										
											2022-04-10 23:12:02 -04:00
										 |  |  |  |       logD("finding free slot in %d...",j); | 
					
						
							| 
									
										
										
										
											2022-05-15 02:42:49 -04:00
										 |  |  |  |       if (curPat[i].data[j]==NULL) { | 
					
						
							| 
									
										
										
										
											2022-02-12 03:59:05 -05:00
										 |  |  |  |         int origOrd=order[i]; | 
					
						
							|  |  |  |  |         order[i]=j; | 
					
						
							| 
									
										
										
										
											2022-05-15 02:42:49 -04:00
										 |  |  |  |         DivPattern* oldPat=curPat[i].getPattern(origOrd,false); | 
					
						
							|  |  |  |  |         DivPattern* pat=curPat[i].getPattern(j,true); | 
					
						
							| 
									
										
										
										
											2022-12-09 13:47:10 -05:00
										 |  |  |  |         memcpy(pat->data,oldPat->data,DIV_MAX_ROWS*DIV_MAX_COLS*sizeof(short)); | 
					
						
							| 
									
										
										
										
											2022-04-10 23:12:02 -04:00
										 |  |  |  |         logD("found at %d",j); | 
					
						
							| 
									
										
										
										
											2022-02-20 23:07:46 -05:00
										 |  |  |  |         didNotFind=false; | 
					
						
							| 
									
										
										
										
											2022-02-12 03:59:05 -05:00
										 |  |  |  |         break; | 
					
						
							|  |  |  |  |       } | 
					
						
							|  |  |  |  |     } | 
					
						
							| 
									
										
										
										
											2022-02-20 23:07:46 -05:00
										 |  |  |  |     if (didNotFind) { | 
					
						
							| 
									
										
										
										
											2024-05-27 05:02:07 -04:00
										 |  |  |  |       addWarning(fmt::sprintf(_("no free patterns in channel %d!"),i)); | 
					
						
							| 
									
										
										
										
											2022-02-12 18:02:33 -05:00
										 |  |  |  |     } | 
					
						
							| 
									
										
										
										
											2022-02-12 03:59:05 -05:00
										 |  |  |  |   } | 
					
						
							|  |  |  |  |   if (where) { // at the end
 | 
					
						
							| 
									
										
										
										
											2022-03-21 17:17:51 -04:00
										 |  |  |  |     saveLock.lock(); | 
					
						
							| 
									
										
										
										
											2022-02-12 03:59:05 -05:00
										 |  |  |  |     for (int i=0; i<chans; i++) { | 
					
						
							| 
									
										
										
										
											2022-05-15 02:42:49 -04:00
										 |  |  |  |       curOrders->ord[i][curSubSong->ordersLen]=order[i]; | 
					
						
							| 
									
										
										
										
											2022-02-12 03:59:05 -05:00
										 |  |  |  |     } | 
					
						
							| 
									
										
										
										
											2022-05-15 02:42:49 -04:00
										 |  |  |  |     curSubSong->ordersLen++; | 
					
						
							| 
									
										
										
										
											2022-03-21 17:17:51 -04:00
										 |  |  |  |     saveLock.unlock(); | 
					
						
							| 
									
										
										
										
											2022-02-12 03:59:05 -05:00
										 |  |  |  |   } else { // after current order
 | 
					
						
							| 
									
										
										
										
											2022-03-21 17:17:51 -04:00
										 |  |  |  |     saveLock.lock(); | 
					
						
							| 
									
										
										
										
											2022-02-12 03:59:05 -05:00
										 |  |  |  |     for (int i=0; i<chans; i++) { | 
					
						
							| 
									
										
										
										
											2023-02-05 19:06:13 -05:00
										 |  |  |  |       for (int j=curSubSong->ordersLen; j>pos; j--) { | 
					
						
							| 
									
										
										
										
											2022-05-15 02:42:49 -04:00
										 |  |  |  |         curOrders->ord[i][j]=curOrders->ord[i][j-1]; | 
					
						
							| 
									
										
										
										
											2022-02-12 03:59:05 -05:00
										 |  |  |  |       } | 
					
						
							| 
									
										
										
										
											2023-02-05 19:06:13 -05:00
										 |  |  |  |       curOrders->ord[i][pos+1]=order[i]; | 
					
						
							| 
									
										
										
										
											2022-02-12 03:59:05 -05:00
										 |  |  |  |     } | 
					
						
							| 
									
										
										
										
											2022-05-15 02:42:49 -04:00
										 |  |  |  |     curSubSong->ordersLen++; | 
					
						
							| 
									
										
										
										
											2022-03-21 17:17:51 -04:00
										 |  |  |  |     saveLock.unlock(); | 
					
						
							| 
									
										
										
										
											2023-02-05 19:06:13 -05:00
										 |  |  |  |     if (pos<=curOrder) curOrder++; | 
					
						
							| 
									
										
										
										
											2022-02-12 03:59:05 -05:00
										 |  |  |  |     if (playing && !freelance) { | 
					
						
							|  |  |  |  |       playSub(false); | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  |   } | 
					
						
							| 
									
										
										
										
											2022-03-23 22:38:28 -04:00
										 |  |  |  |   BUSY_END; | 
					
						
							| 
									
										
										
										
											2022-02-12 03:59:05 -05:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-05 19:06:13 -05:00
										 |  |  |  | void DivEngine::deleteOrder(int pos) { | 
					
						
							| 
									
										
										
										
											2022-05-15 02:42:49 -04:00
										 |  |  |  |   if (curSubSong->ordersLen<=1) return; | 
					
						
							| 
									
										
										
										
											2022-03-23 22:38:28 -04:00
										 |  |  |  |   BUSY_BEGIN_SOFT; | 
					
						
							| 
									
										
										
										
											2022-03-21 17:17:51 -04:00
										 |  |  |  |   saveLock.lock(); | 
					
						
							| 
									
										
										
										
											2022-01-08 17:15:12 -05:00
										 |  |  |  |   for (int i=0; i<DIV_MAX_CHANS; i++) { | 
					
						
							| 
									
										
										
										
											2023-02-05 19:06:13 -05:00
										 |  |  |  |     for (int j=pos; j<curSubSong->ordersLen; j++) { | 
					
						
							| 
									
										
										
										
											2022-05-15 02:42:49 -04:00
										 |  |  |  |       curOrders->ord[i][j]=curOrders->ord[i][j+1]; | 
					
						
							| 
									
										
										
										
											2021-12-22 17:39:16 -05:00
										 |  |  |  |     } | 
					
						
							|  |  |  |  |   } | 
					
						
							| 
									
										
										
										
											2022-05-15 02:42:49 -04:00
										 |  |  |  |   curSubSong->ordersLen--; | 
					
						
							| 
									
										
										
										
											2022-03-21 17:17:51 -04:00
										 |  |  |  |   saveLock.unlock(); | 
					
						
							| 
									
										
										
										
											2023-02-05 19:06:13 -05:00
										 |  |  |  |   if (curOrder>pos) curOrder--; | 
					
						
							| 
									
										
										
										
											2022-05-15 02:42:49 -04:00
										 |  |  |  |   if (curOrder>=curSubSong->ordersLen) curOrder=curSubSong->ordersLen-1; | 
					
						
							| 
									
										
										
										
											2021-12-28 18:23:57 -05:00
										 |  |  |  |   if (playing && !freelance) { | 
					
						
							| 
									
										
										
										
											2021-12-22 17:39:16 -05:00
										 |  |  |  |     playSub(false); | 
					
						
							|  |  |  |  |   } | 
					
						
							| 
									
										
										
										
											2022-03-23 22:38:28 -04:00
										 |  |  |  |   BUSY_END; | 
					
						
							| 
									
										
										
										
											2021-12-22 17:39:16 -05:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-05 19:06:13 -05:00
										 |  |  |  | void DivEngine::moveOrderUp(int& pos) { | 
					
						
							| 
									
										
										
										
											2022-03-23 22:38:28 -04:00
										 |  |  |  |   BUSY_BEGIN_SOFT; | 
					
						
							| 
									
										
										
										
											2023-02-05 19:06:13 -05:00
										 |  |  |  |   if (pos<1) { | 
					
						
							| 
									
										
										
										
											2022-03-23 22:38:28 -04:00
										 |  |  |  |     BUSY_END; | 
					
						
							| 
									
										
										
										
											2021-12-22 17:39:16 -05:00
										 |  |  |  |     return; | 
					
						
							|  |  |  |  |   } | 
					
						
							| 
									
										
										
										
											2022-03-21 17:17:51 -04:00
										 |  |  |  |   saveLock.lock(); | 
					
						
							| 
									
										
										
										
											2022-01-08 17:15:12 -05:00
										 |  |  |  |   for (int i=0; i<DIV_MAX_CHANS; i++) { | 
					
						
							| 
									
										
										
										
											2023-02-05 19:06:13 -05:00
										 |  |  |  |     curOrders->ord[i][pos]^=curOrders->ord[i][pos-1]; | 
					
						
							|  |  |  |  |     curOrders->ord[i][pos-1]^=curOrders->ord[i][pos]; | 
					
						
							|  |  |  |  |     curOrders->ord[i][pos]^=curOrders->ord[i][pos-1]; | 
					
						
							| 
									
										
										
										
											2021-12-22 17:39:16 -05:00
										 |  |  |  |   } | 
					
						
							| 
									
										
										
										
											2022-03-21 17:17:51 -04:00
										 |  |  |  |   saveLock.unlock(); | 
					
						
							| 
									
										
										
										
											2023-02-05 19:06:13 -05:00
										 |  |  |  |   if (curOrder==pos) { | 
					
						
							|  |  |  |  |     curOrder--; | 
					
						
							|  |  |  |  |   } | 
					
						
							|  |  |  |  |   pos--; | 
					
						
							| 
									
										
										
										
											2021-12-28 18:23:57 -05:00
										 |  |  |  |   if (playing && !freelance) { | 
					
						
							| 
									
										
										
										
											2021-12-22 17:39:16 -05:00
										 |  |  |  |     playSub(false); | 
					
						
							|  |  |  |  |   } | 
					
						
							| 
									
										
										
										
											2022-03-23 22:38:28 -04:00
										 |  |  |  |   BUSY_END; | 
					
						
							| 
									
										
										
										
											2021-12-22 17:39:16 -05:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-05 19:06:13 -05:00
										 |  |  |  | void DivEngine::moveOrderDown(int& pos) { | 
					
						
							| 
									
										
										
										
											2022-03-23 22:38:28 -04:00
										 |  |  |  |   BUSY_BEGIN_SOFT; | 
					
						
							| 
									
										
										
										
											2023-02-05 19:06:13 -05:00
										 |  |  |  |   if (pos>=curSubSong->ordersLen-1) { | 
					
						
							| 
									
										
										
										
											2022-03-23 22:38:28 -04:00
										 |  |  |  |     BUSY_END; | 
					
						
							| 
									
										
										
										
											2021-12-22 17:39:16 -05:00
										 |  |  |  |     return; | 
					
						
							|  |  |  |  |   } | 
					
						
							| 
									
										
										
										
											2022-03-21 17:17:51 -04:00
										 |  |  |  |   saveLock.lock(); | 
					
						
							| 
									
										
										
										
											2022-01-08 17:15:12 -05:00
										 |  |  |  |   for (int i=0; i<DIV_MAX_CHANS; i++) { | 
					
						
							| 
									
										
										
										
											2023-02-05 19:06:13 -05:00
										 |  |  |  |     curOrders->ord[i][pos]^=curOrders->ord[i][pos+1]; | 
					
						
							|  |  |  |  |     curOrders->ord[i][pos+1]^=curOrders->ord[i][pos]; | 
					
						
							|  |  |  |  |     curOrders->ord[i][pos]^=curOrders->ord[i][pos+1]; | 
					
						
							| 
									
										
										
										
											2021-12-22 17:39:16 -05:00
										 |  |  |  |   } | 
					
						
							| 
									
										
										
										
											2022-03-21 17:17:51 -04:00
										 |  |  |  |   saveLock.unlock(); | 
					
						
							| 
									
										
										
										
											2023-02-05 19:06:13 -05:00
										 |  |  |  |   if (curOrder==pos) { | 
					
						
							|  |  |  |  |     curOrder++; | 
					
						
							|  |  |  |  |   } | 
					
						
							|  |  |  |  |   pos++; | 
					
						
							| 
									
										
										
										
											2021-12-28 18:23:57 -05:00
										 |  |  |  |   if (playing && !freelance) { | 
					
						
							| 
									
										
										
										
											2021-12-22 17:39:16 -05:00
										 |  |  |  |     playSub(false); | 
					
						
							|  |  |  |  |   } | 
					
						
							| 
									
										
										
										
											2022-03-23 22:38:28 -04:00
										 |  |  |  |   BUSY_END; | 
					
						
							| 
									
										
										
										
											2021-12-22 17:39:16 -05:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-02-07 23:42:54 -05:00
										 |  |  |  | void DivEngine::exchangeIns(int one, int two) { | 
					
						
							|  |  |  |  |   for (int i=0; i<chans; i++) { | 
					
						
							| 
									
										
										
										
											2022-08-15 23:54:31 -04:00
										 |  |  |  |     for (size_t j=0; j<song.subsong.size(); j++) { | 
					
						
							| 
									
										
										
										
											2022-12-09 00:30:27 -05:00
										 |  |  |  |       for (int k=0; k<DIV_MAX_PATTERNS; k++) { | 
					
						
							| 
									
										
										
										
											2022-08-15 23:54:31 -04:00
										 |  |  |  |         if (song.subsong[j]->pat[i].data[k]==NULL) continue; | 
					
						
							| 
									
										
										
										
											2022-10-03 13:16:50 -04:00
										 |  |  |  |         for (int l=0; l<song.subsong[j]->patLen; l++) { | 
					
						
							| 
									
										
										
										
											2022-08-15 23:54:31 -04:00
										 |  |  |  |           if (song.subsong[j]->pat[i].data[k]->data[l][2]==one) { | 
					
						
							|  |  |  |  |             song.subsong[j]->pat[i].data[k]->data[l][2]=two; | 
					
						
							|  |  |  |  |           } else if (song.subsong[j]->pat[i].data[k]->data[l][2]==two) { | 
					
						
							|  |  |  |  |             song.subsong[j]->pat[i].data[k]->data[l][2]=one; | 
					
						
							|  |  |  |  |           } | 
					
						
							| 
									
										
										
										
											2022-02-07 23:42:54 -05:00
										 |  |  |  |         } | 
					
						
							|  |  |  |  |       } | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  |   } | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-07-26 18:58:57 -04:00
										 |  |  |  | void DivEngine::exchangeWave(int one, int two) { | 
					
						
							|  |  |  |  |   // TODO
 | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | void DivEngine::exchangeSample(int one, int two) { | 
					
						
							|  |  |  |  |   for (DivInstrument* i: song.ins) { | 
					
						
							|  |  |  |  |     if (i->amiga.initSample==one) { | 
					
						
							|  |  |  |  |       i->amiga.initSample=two; | 
					
						
							|  |  |  |  |     } else if (i->amiga.initSample==two) { | 
					
						
							|  |  |  |  |       i->amiga.initSample=one; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  |     for (int j=0; j<120; j++) { | 
					
						
							|  |  |  |  |       if (i->amiga.noteMap[j].map==one) { | 
					
						
							|  |  |  |  |         i->amiga.noteMap[j].map=two; | 
					
						
							|  |  |  |  |       } else if (i->amiga.noteMap[j].map==two) { | 
					
						
							|  |  |  |  |         i->amiga.noteMap[j].map=one; | 
					
						
							|  |  |  |  |       } | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  |   } | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-01-11 03:52:11 -05:00
										 |  |  |  | bool DivEngine::moveInsUp(int which) { | 
					
						
							|  |  |  |  |   if (which<1 || which>=(int)song.ins.size()) return false; | 
					
						
							| 
									
										
										
										
											2022-03-23 22:38:28 -04:00
										 |  |  |  |   BUSY_BEGIN; | 
					
						
							| 
									
										
										
										
											2022-01-11 03:52:11 -05:00
										 |  |  |  |   DivInstrument* prev=song.ins[which]; | 
					
						
							| 
									
										
										
										
											2022-03-21 17:17:51 -04:00
										 |  |  |  |   saveLock.lock(); | 
					
						
							| 
									
										
										
										
											2022-01-11 03:52:11 -05:00
										 |  |  |  |   song.ins[which]=song.ins[which-1]; | 
					
						
							|  |  |  |  |   song.ins[which-1]=prev; | 
					
						
							| 
									
										
										
										
											2023-05-16 04:04:16 -04:00
										 |  |  |  |   moveAsset(song.insDir,which,which-1); | 
					
						
							| 
									
										
										
										
											2022-02-07 23:42:54 -05:00
										 |  |  |  |   exchangeIns(which,which-1); | 
					
						
							| 
									
										
										
										
											2022-03-21 17:17:51 -04:00
										 |  |  |  |   saveLock.unlock(); | 
					
						
							| 
									
										
										
										
											2022-03-23 22:38:28 -04:00
										 |  |  |  |   BUSY_END; | 
					
						
							| 
									
										
										
										
											2022-01-11 03:52:11 -05:00
										 |  |  |  |   return true; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | bool DivEngine::moveWaveUp(int which) { | 
					
						
							|  |  |  |  |   if (which<1 || which>=(int)song.wave.size()) return false; | 
					
						
							| 
									
										
										
										
											2022-03-23 22:38:28 -04:00
										 |  |  |  |   BUSY_BEGIN; | 
					
						
							| 
									
										
										
										
											2022-01-11 03:52:11 -05:00
										 |  |  |  |   DivWavetable* prev=song.wave[which]; | 
					
						
							| 
									
										
										
										
											2022-03-21 17:17:51 -04:00
										 |  |  |  |   saveLock.lock(); | 
					
						
							| 
									
										
										
										
											2022-01-11 03:52:11 -05:00
										 |  |  |  |   song.wave[which]=song.wave[which-1]; | 
					
						
							|  |  |  |  |   song.wave[which-1]=prev; | 
					
						
							| 
									
										
										
										
											2023-05-16 04:04:16 -04:00
										 |  |  |  |   moveAsset(song.waveDir,which,which-1); | 
					
						
							| 
									
										
										
										
											2023-07-26 18:58:57 -04:00
										 |  |  |  |   exchangeWave(which,which-1); | 
					
						
							| 
									
										
										
										
											2022-03-21 17:17:51 -04:00
										 |  |  |  |   saveLock.unlock(); | 
					
						
							| 
									
										
										
										
											2022-03-23 22:38:28 -04:00
										 |  |  |  |   BUSY_END; | 
					
						
							| 
									
										
										
										
											2022-01-11 03:52:11 -05:00
										 |  |  |  |   return true; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | bool DivEngine::moveSampleUp(int which) { | 
					
						
							|  |  |  |  |   if (which<1 || which>=(int)song.sample.size()) return false; | 
					
						
							| 
									
										
										
										
											2022-03-23 22:38:28 -04:00
										 |  |  |  |   BUSY_BEGIN; | 
					
						
							| 
									
										
										
										
											2022-06-17 00:39:38 -04:00
										 |  |  |  |   sPreview.sample=-1; | 
					
						
							|  |  |  |  |   sPreview.pos=0; | 
					
						
							| 
									
										
										
										
											2022-08-11 09:21:54 -04:00
										 |  |  |  |   sPreview.dir=false; | 
					
						
							| 
									
										
										
										
											2022-01-11 03:52:11 -05:00
										 |  |  |  |   DivSample* prev=song.sample[which]; | 
					
						
							| 
									
										
										
										
											2022-03-21 17:17:51 -04:00
										 |  |  |  |   saveLock.lock(); | 
					
						
							| 
									
										
										
										
											2022-01-11 03:52:11 -05:00
										 |  |  |  |   song.sample[which]=song.sample[which-1]; | 
					
						
							|  |  |  |  |   song.sample[which-1]=prev; | 
					
						
							| 
									
										
										
										
											2023-05-16 04:04:16 -04:00
										 |  |  |  |   moveAsset(song.sampleDir,which,which-1); | 
					
						
							| 
									
										
										
										
											2023-07-26 18:58:57 -04:00
										 |  |  |  |   exchangeSample(which,which-1); | 
					
						
							| 
									
										
										
										
											2022-03-21 17:17:51 -04:00
										 |  |  |  |   saveLock.unlock(); | 
					
						
							| 
									
										
										
										
											2022-10-02 03:08:33 -04:00
										 |  |  |  |   renderSamples(); | 
					
						
							| 
									
										
										
										
											2022-03-23 22:38:28 -04:00
										 |  |  |  |   BUSY_END; | 
					
						
							| 
									
										
										
										
											2022-01-11 03:52:11 -05:00
										 |  |  |  |   return true; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | bool DivEngine::moveInsDown(int which) { | 
					
						
							|  |  |  |  |   if (which<0 || which>=((int)song.ins.size())-1) return false; | 
					
						
							| 
									
										
										
										
											2022-03-23 22:38:28 -04:00
										 |  |  |  |   BUSY_BEGIN; | 
					
						
							| 
									
										
										
										
											2022-01-11 03:52:11 -05:00
										 |  |  |  |   DivInstrument* prev=song.ins[which]; | 
					
						
							| 
									
										
										
										
											2022-03-21 17:17:51 -04:00
										 |  |  |  |   saveLock.lock(); | 
					
						
							| 
									
										
										
										
											2022-01-11 03:52:11 -05:00
										 |  |  |  |   song.ins[which]=song.ins[which+1]; | 
					
						
							|  |  |  |  |   song.ins[which+1]=prev; | 
					
						
							| 
									
										
										
										
											2022-02-07 23:42:54 -05:00
										 |  |  |  |   exchangeIns(which,which+1); | 
					
						
							| 
									
										
										
										
											2023-05-16 04:04:16 -04:00
										 |  |  |  |   moveAsset(song.insDir,which,which+1); | 
					
						
							| 
									
										
										
										
											2022-03-21 17:17:51 -04:00
										 |  |  |  |   saveLock.unlock(); | 
					
						
							| 
									
										
										
										
											2022-03-23 22:38:28 -04:00
										 |  |  |  |   BUSY_END; | 
					
						
							| 
									
										
										
										
											2022-01-11 03:52:11 -05:00
										 |  |  |  |   return true; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | bool DivEngine::moveWaveDown(int which) { | 
					
						
							|  |  |  |  |   if (which<0 || which>=((int)song.wave.size())-1) return false; | 
					
						
							| 
									
										
										
										
											2022-03-23 22:38:28 -04:00
										 |  |  |  |   BUSY_BEGIN; | 
					
						
							| 
									
										
										
										
											2022-01-11 03:52:11 -05:00
										 |  |  |  |   DivWavetable* prev=song.wave[which]; | 
					
						
							| 
									
										
										
										
											2022-03-21 17:17:51 -04:00
										 |  |  |  |   saveLock.lock(); | 
					
						
							| 
									
										
										
										
											2022-01-11 03:52:11 -05:00
										 |  |  |  |   song.wave[which]=song.wave[which+1]; | 
					
						
							|  |  |  |  |   song.wave[which+1]=prev; | 
					
						
							| 
									
										
										
										
											2023-07-26 18:58:57 -04:00
										 |  |  |  |   exchangeWave(which,which+1); | 
					
						
							| 
									
										
										
										
											2023-05-16 04:04:16 -04:00
										 |  |  |  |   moveAsset(song.waveDir,which,which+1); | 
					
						
							| 
									
										
										
										
											2022-03-21 17:17:51 -04:00
										 |  |  |  |   saveLock.unlock(); | 
					
						
							| 
									
										
										
										
											2022-03-23 22:38:28 -04:00
										 |  |  |  |   BUSY_END; | 
					
						
							| 
									
										
										
										
											2022-01-11 03:52:11 -05:00
										 |  |  |  |   return true; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | bool DivEngine::moveSampleDown(int which) { | 
					
						
							|  |  |  |  |   if (which<0 || which>=((int)song.sample.size())-1) return false; | 
					
						
							| 
									
										
										
										
											2022-03-23 22:38:28 -04:00
										 |  |  |  |   BUSY_BEGIN; | 
					
						
							| 
									
										
										
										
											2022-06-17 00:39:38 -04:00
										 |  |  |  |   sPreview.sample=-1; | 
					
						
							|  |  |  |  |   sPreview.pos=0; | 
					
						
							| 
									
										
										
										
											2022-08-11 09:21:54 -04:00
										 |  |  |  |   sPreview.dir=false; | 
					
						
							| 
									
										
										
										
											2022-01-11 03:52:11 -05:00
										 |  |  |  |   DivSample* prev=song.sample[which]; | 
					
						
							| 
									
										
										
										
											2022-03-21 17:17:51 -04:00
										 |  |  |  |   saveLock.lock(); | 
					
						
							| 
									
										
										
										
											2022-01-11 03:52:11 -05:00
										 |  |  |  |   song.sample[which]=song.sample[which+1]; | 
					
						
							|  |  |  |  |   song.sample[which+1]=prev; | 
					
						
							| 
									
										
										
										
											2023-07-26 18:58:57 -04:00
										 |  |  |  |   exchangeSample(which,which+1); | 
					
						
							| 
									
										
										
										
											2023-05-16 04:04:16 -04:00
										 |  |  |  |   moveAsset(song.sampleDir,which,which+1); | 
					
						
							| 
									
										
										
										
											2022-03-21 17:17:51 -04:00
										 |  |  |  |   saveLock.unlock(); | 
					
						
							| 
									
										
										
										
											2022-10-02 03:00:31 -04:00
										 |  |  |  |   renderSamples(); | 
					
						
							| 
									
										
										
										
											2022-03-23 22:38:28 -04:00
										 |  |  |  |   BUSY_END; | 
					
						
							| 
									
										
										
										
											2022-01-11 03:52:11 -05:00
										 |  |  |  |   return true; | 
					
						
							| 
									
										
										
										
											2024-12-19 13:04:10 -05:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | bool DivEngine::swapInstruments(int a, int b) { | 
					
						
							|  |  |  |  |   if (a<0 || a>=(int)song.ins.size() || b<0 || b>=(int)song.ins.size()) return false; | 
					
						
							|  |  |  |  |   BUSY_BEGIN; | 
					
						
							|  |  |  |  |   DivInstrument* temp=song.ins[a]; | 
					
						
							|  |  |  |  |   saveLock.lock(); | 
					
						
							|  |  |  |  |   song.ins[a]=song.ins[b]; | 
					
						
							|  |  |  |  |   song.ins[b]=temp; | 
					
						
							|  |  |  |  |   moveAsset(song.insDir,a,b); | 
					
						
							|  |  |  |  |   exchangeIns(a,b); | 
					
						
							|  |  |  |  |   saveLock.unlock(); | 
					
						
							|  |  |  |  |   BUSY_END; | 
					
						
							|  |  |  |  |   return true; | 
					
						
							| 
									
										
										
										
											2022-01-11 03:52:11 -05:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-12-19 13:51:21 -05:00
										 |  |  |  | bool DivEngine::swapWaves(int a, int b) { | 
					
						
							|  |  |  |  |   if (a<0 || a>=(int)song.wave.size() || b<0 || b>=(int)song.wave.size()) return false; | 
					
						
							|  |  |  |  |   BUSY_BEGIN; | 
					
						
							|  |  |  |  |   DivWavetable* temp=song.wave[a]; | 
					
						
							|  |  |  |  |   saveLock.lock(); | 
					
						
							|  |  |  |  |   song.wave[a]=song.wave[b]; | 
					
						
							|  |  |  |  |   song.wave[b]=temp; | 
					
						
							|  |  |  |  |   exchangeWave(a,b); | 
					
						
							|  |  |  |  |   moveAsset(song.waveDir,a,b); | 
					
						
							|  |  |  |  |   saveLock.unlock(); | 
					
						
							|  |  |  |  |   BUSY_END; | 
					
						
							|  |  |  |  |   return true; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | bool DivEngine::swapSamples(int a, int b) { | 
					
						
							|  |  |  |  |   if (a<0 || a>=(int)song.sample.size() || b<0 || b>=(int)song.sample.size()) return false; | 
					
						
							|  |  |  |  |   BUSY_BEGIN; | 
					
						
							|  |  |  |  |   sPreview.sample=-1; | 
					
						
							|  |  |  |  |   sPreview.pos=0; | 
					
						
							|  |  |  |  |   sPreview.dir=false; | 
					
						
							|  |  |  |  |   DivSample* temp=song.sample[a]; | 
					
						
							|  |  |  |  |   saveLock.lock(); | 
					
						
							|  |  |  |  |   song.sample[a]=song.sample[b]; | 
					
						
							|  |  |  |  |   song.sample[b]=temp; | 
					
						
							|  |  |  |  |   exchangeSample(a,b); | 
					
						
							|  |  |  |  |   moveAsset(song.sampleDir,a,b); | 
					
						
							|  |  |  |  |   saveLock.unlock(); | 
					
						
							|  |  |  |  |   renderSamples(); | 
					
						
							|  |  |  |  |   BUSY_END; | 
					
						
							|  |  |  |  |   return true; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-01-06 18:44:20 -05:00
										 |  |  |  | void DivEngine::autoPatchbay() { | 
					
						
							|  |  |  |  |   song.patchbay.clear(); | 
					
						
							|  |  |  |  |   for (unsigned int i=0; i<song.systemLen; i++) { | 
					
						
							|  |  |  |  |     if (disCont[i].dispatch==NULL) continue; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     unsigned int outs=disCont[i].dispatch->getOutputCount(); | 
					
						
							|  |  |  |  |     if (outs>16) outs=16; | 
					
						
							|  |  |  |  |     if (outs<2) { | 
					
						
							| 
									
										
										
										
											2023-08-24 12:08:06 -04:00
										 |  |  |  |       song.patchbay.reserve(DIV_MAX_OUTPUTS); | 
					
						
							| 
									
										
										
										
											2023-01-06 18:44:20 -05:00
										 |  |  |  |       for (unsigned int j=0; j<DIV_MAX_OUTPUTS; j++) { | 
					
						
							|  |  |  |  |         song.patchbay.push_back((i<<20)|j); | 
					
						
							|  |  |  |  |       } | 
					
						
							|  |  |  |  |     } else { | 
					
						
							| 
									
										
										
										
											2023-08-24 12:08:06 -04:00
										 |  |  |  |       song.patchbay.reserve(outs); | 
					
						
							| 
									
										
										
										
											2023-01-06 18:44:20 -05:00
										 |  |  |  |       for (unsigned int j=0; j<outs; j++) { | 
					
						
							|  |  |  |  |         song.patchbay.push_back((i<<20)|(j<<16)|j); | 
					
						
							|  |  |  |  |       } | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  |   } | 
					
						
							| 
									
										
										
										
											2023-01-10 19:09:26 -05:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |   // wave/sample preview
 | 
					
						
							| 
									
										
										
										
											2023-08-24 12:08:06 -04:00
										 |  |  |  |   song.patchbay.reserve(DIV_MAX_OUTPUTS); | 
					
						
							| 
									
										
										
										
											2023-01-10 19:09:26 -05:00
										 |  |  |  |   for (unsigned int j=0; j<DIV_MAX_OUTPUTS; j++) { | 
					
						
							|  |  |  |  |     song.patchbay.push_back(0xffd00000|j); | 
					
						
							|  |  |  |  |   } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   // metronome
 | 
					
						
							| 
									
										
										
										
											2023-08-24 12:08:06 -04:00
										 |  |  |  |   song.patchbay.reserve(DIV_MAX_OUTPUTS); | 
					
						
							| 
									
										
										
										
											2023-01-10 19:09:26 -05:00
										 |  |  |  |   for (unsigned int j=0; j<DIV_MAX_OUTPUTS; j++) { | 
					
						
							|  |  |  |  |     song.patchbay.push_back(0xffe00000|j); | 
					
						
							|  |  |  |  |   } | 
					
						
							| 
									
										
										
										
											2023-01-06 18:44:20 -05:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-01-10 15:58:15 -05:00
										 |  |  |  | void DivEngine::autoPatchbayP() { | 
					
						
							|  |  |  |  |   BUSY_BEGIN; | 
					
						
							|  |  |  |  |   saveLock.lock(); | 
					
						
							|  |  |  |  |   autoPatchbay(); | 
					
						
							|  |  |  |  |   saveLock.unlock(); | 
					
						
							|  |  |  |  |   BUSY_END; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-05-21 06:40:10 -04:00
										 |  |  |  | void DivEngine::recalcPatchbay() { | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-01-08 19:23:17 -05:00
										 |  |  |  | bool DivEngine::patchConnect(unsigned int src, unsigned int dest) { | 
					
						
							|  |  |  |  |   unsigned int armed=(src<<16)|(dest&0xffff); | 
					
						
							|  |  |  |  |   for (unsigned int i: song.patchbay) { | 
					
						
							|  |  |  |  |     if (i==armed) return false; | 
					
						
							|  |  |  |  |   } | 
					
						
							|  |  |  |  |   BUSY_BEGIN; | 
					
						
							|  |  |  |  |   saveLock.lock(); | 
					
						
							|  |  |  |  |   song.patchbay.push_back(armed); | 
					
						
							| 
									
										
										
										
											2023-01-10 15:58:15 -05:00
										 |  |  |  |   song.patchbayAuto=false; | 
					
						
							| 
									
										
										
										
											2023-01-08 19:23:17 -05:00
										 |  |  |  |   saveLock.unlock(); | 
					
						
							|  |  |  |  |   BUSY_END; | 
					
						
							|  |  |  |  |   return true; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | bool DivEngine::patchDisconnect(unsigned int src, unsigned int dest) { | 
					
						
							|  |  |  |  |   unsigned int armed=(src<<16)|(dest&0xffff); | 
					
						
							|  |  |  |  |   for (auto i=song.patchbay.begin(); i!=song.patchbay.end(); i++) { | 
					
						
							|  |  |  |  |     if (*i==armed) { | 
					
						
							|  |  |  |  |       BUSY_BEGIN; | 
					
						
							|  |  |  |  |       saveLock.lock(); | 
					
						
							|  |  |  |  |       song.patchbay.erase(i); | 
					
						
							| 
									
										
										
										
											2023-01-10 15:58:15 -05:00
										 |  |  |  |       song.patchbayAuto=false; | 
					
						
							| 
									
										
										
										
											2023-01-08 19:23:17 -05:00
										 |  |  |  |       saveLock.unlock(); | 
					
						
							|  |  |  |  |       BUSY_END; | 
					
						
							|  |  |  |  |       return true; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  |   } | 
					
						
							|  |  |  |  |   return false; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-01-12 03:56:18 -05:00
										 |  |  |  | void DivEngine::patchDisconnectAll(unsigned int portSet) { | 
					
						
							|  |  |  |  |   BUSY_BEGIN; | 
					
						
							|  |  |  |  |   saveLock.lock(); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   if (portSet&0x1000) { | 
					
						
							|  |  |  |  |     portSet&=0xfff; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     for (size_t i=0; i<song.patchbay.size(); i++) { | 
					
						
							|  |  |  |  |       if ((song.patchbay[i]&0xfff0)==(portSet<<4)) { | 
					
						
							|  |  |  |  |         song.patchbay.erase(song.patchbay.begin()+i); | 
					
						
							|  |  |  |  |         i--; | 
					
						
							|  |  |  |  |       } | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  |   } else { | 
					
						
							|  |  |  |  |     portSet&=0xfff; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     for (size_t i=0; i<song.patchbay.size(); i++) { | 
					
						
							|  |  |  |  |       if ((song.patchbay[i]&0xfff00000)==(portSet<<20)) { | 
					
						
							|  |  |  |  |         song.patchbay.erase(song.patchbay.begin()+i); | 
					
						
							|  |  |  |  |         i--; | 
					
						
							|  |  |  |  |       } | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  |   } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   saveLock.unlock(); | 
					
						
							|  |  |  |  |   BUSY_END; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-12-28 18:23:57 -05:00
										 |  |  |  | void DivEngine::noteOn(int chan, int ins, int note, int vol) { | 
					
						
							| 
									
										
										
										
											2022-01-24 17:13:47 -05:00
										 |  |  |  |   if (chan<0 || chan>=chans) return; | 
					
						
							| 
									
										
										
										
											2022-03-23 22:38:28 -04:00
										 |  |  |  |   BUSY_BEGIN; | 
					
						
							| 
									
										
										
										
											2022-08-27 01:37:32 -04:00
										 |  |  |  |   pendingNotes.push_back(DivNoteEvent(chan,ins,note,vol,true)); | 
					
						
							| 
									
										
										
										
											2021-12-28 18:23:57 -05:00
										 |  |  |  |   if (!playing) { | 
					
						
							|  |  |  |  |     reset(); | 
					
						
							|  |  |  |  |     freelance=true; | 
					
						
							|  |  |  |  |     playing=true; | 
					
						
							|  |  |  |  |   } | 
					
						
							| 
									
										
										
										
											2022-03-23 22:38:28 -04:00
										 |  |  |  |   BUSY_END; | 
					
						
							| 
									
										
										
										
											2021-12-28 18:23:57 -05:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | void DivEngine::noteOff(int chan) { | 
					
						
							| 
									
										
										
										
											2022-01-24 17:13:47 -05:00
										 |  |  |  |   if (chan<0 || chan>=chans) return; | 
					
						
							| 
									
										
										
										
											2022-03-23 22:38:28 -04:00
										 |  |  |  |   BUSY_BEGIN; | 
					
						
							| 
									
										
										
										
											2022-08-27 01:37:32 -04:00
										 |  |  |  |   pendingNotes.push_back(DivNoteEvent(chan,-1,-1,-1,false)); | 
					
						
							| 
									
										
										
										
											2021-12-28 18:23:57 -05:00
										 |  |  |  |   if (!playing) { | 
					
						
							|  |  |  |  |     reset(); | 
					
						
							|  |  |  |  |     freelance=true; | 
					
						
							|  |  |  |  |     playing=true; | 
					
						
							|  |  |  |  |   } | 
					
						
							| 
									
										
										
										
											2022-03-23 22:38:28 -04:00
										 |  |  |  |   BUSY_END; | 
					
						
							| 
									
										
										
										
											2021-12-28 18:23:57 -05:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-10-15 18:02:25 -04:00
										 |  |  |  | bool DivEngine::autoNoteOn(int ch, int ins, int note, int vol) { | 
					
						
							| 
									
										
										
										
											2022-04-28 06:04:34 -04:00
										 |  |  |  |   bool isViable[DIV_MAX_CHANS]; | 
					
						
							|  |  |  |  |   bool canPlayAnyway=false; | 
					
						
							| 
									
										
										
										
											2022-04-28 19:32:24 -04:00
										 |  |  |  |   bool notInViableChannel=false; | 
					
						
							| 
									
										
										
										
											2022-03-31 02:51:57 -04:00
										 |  |  |  |   if (midiBaseChan<0) midiBaseChan=0; | 
					
						
							|  |  |  |  |   if (midiBaseChan>=chans) midiBaseChan=chans-1; | 
					
						
							|  |  |  |  |   int finalChan=midiBaseChan; | 
					
						
							| 
									
										
										
										
											2022-04-28 06:04:34 -04:00
										 |  |  |  |   int finalChanType=getChannelType(finalChan); | 
					
						
							| 
									
										
										
										
											2022-03-31 02:51:57 -04:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |   if (!playing) { | 
					
						
							|  |  |  |  |     reset(); | 
					
						
							|  |  |  |  |     freelance=true; | 
					
						
							|  |  |  |  |     playing=true; | 
					
						
							|  |  |  |  |   } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-28 06:04:34 -04:00
										 |  |  |  |   // 1. check which channels are viable for this instrument
 | 
					
						
							|  |  |  |  |   DivInstrument* insInst=getIns(ins); | 
					
						
							| 
									
										
										
										
											2022-05-13 04:18:14 -04:00
										 |  |  |  |   if (getPreferInsType(finalChan)!=insInst->type && getPreferInsSecondType(finalChan)!=insInst->type && getPreferInsType(finalChan)!=DIV_INS_NULL) notInViableChannel=true; | 
					
						
							| 
									
										
										
										
											2022-04-28 06:04:34 -04:00
										 |  |  |  |   for (int i=0; i<chans; i++) { | 
					
						
							| 
									
										
										
										
											2022-05-13 04:18:14 -04:00
										 |  |  |  |     if (ins==-1 || ins>=song.insLen || getPreferInsType(i)==insInst->type || (getPreferInsType(i)==DIV_INS_NULL && finalChanType==DIV_CH_NOISE) || getPreferInsSecondType(i)==insInst->type) { | 
					
						
							| 
									
										
										
										
											2022-04-28 06:04:34 -04:00
										 |  |  |  |       if (insInst->type==DIV_INS_OPL) { | 
					
						
							|  |  |  |  |         if (insInst->fm.ops==2 || getChannelType(i)==DIV_CH_OP) { | 
					
						
							|  |  |  |  |           isViable[i]=true; | 
					
						
							|  |  |  |  |           canPlayAnyway=true; | 
					
						
							|  |  |  |  |         } else { | 
					
						
							|  |  |  |  |           isViable[i]=false; | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  |       } else { | 
					
						
							|  |  |  |  |         isViable[i]=true; | 
					
						
							|  |  |  |  |         canPlayAnyway=true; | 
					
						
							|  |  |  |  |       } | 
					
						
							|  |  |  |  |     } else { | 
					
						
							|  |  |  |  |       isViable[i]=false; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  |   } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-10-15 18:02:25 -04:00
										 |  |  |  |   if (!canPlayAnyway) return false; | 
					
						
							| 
									
										
										
										
											2022-04-28 06:04:34 -04:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |   // 2. find a free channel
 | 
					
						
							| 
									
										
										
										
											2022-03-31 02:51:57 -04:00
										 |  |  |  |   do { | 
					
						
							| 
									
										
										
										
											2022-06-03 19:05:07 -04:00
										 |  |  |  |     if ((!midiPoly) || (isViable[finalChan] && chan[finalChan].midiNote==-1 && (insInst->type==DIV_INS_OPL || getChannelType(finalChan)==finalChanType || notInViableChannel))) { | 
					
						
							| 
									
										
										
										
											2022-03-31 02:51:57 -04:00
										 |  |  |  |       chan[finalChan].midiNote=note; | 
					
						
							| 
									
										
										
										
											2022-04-28 06:04:34 -04:00
										 |  |  |  |       chan[finalChan].midiAge=midiAgeCounter++; | 
					
						
							| 
									
										
										
										
											2022-08-27 01:37:32 -04:00
										 |  |  |  |       pendingNotes.push_back(DivNoteEvent(finalChan,ins,note,vol,true)); | 
					
						
							| 
									
										
										
										
											2023-10-15 18:02:25 -04:00
										 |  |  |  |       return true; | 
					
						
							| 
									
										
										
										
											2022-04-28 06:04:34 -04:00
										 |  |  |  |     } | 
					
						
							|  |  |  |  |     if (++finalChan>=chans) { | 
					
						
							|  |  |  |  |       finalChan=0; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  |   } while (finalChan!=midiBaseChan); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   // 3. find the oldest channel
 | 
					
						
							|  |  |  |  |   int candidate=finalChan; | 
					
						
							|  |  |  |  |   do { | 
					
						
							| 
									
										
										
										
											2022-04-28 19:32:24 -04:00
										 |  |  |  |     if (isViable[finalChan] && (insInst->type==DIV_INS_OPL || getChannelType(finalChan)==finalChanType || notInViableChannel) && chan[finalChan].midiAge<chan[candidate].midiAge) { | 
					
						
							| 
									
										
										
										
											2022-04-28 06:04:34 -04:00
										 |  |  |  |       candidate=finalChan; | 
					
						
							| 
									
										
										
										
											2022-03-31 02:51:57 -04:00
										 |  |  |  |     } | 
					
						
							|  |  |  |  |     if (++finalChan>=chans) { | 
					
						
							|  |  |  |  |       finalChan=0; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  |   } while (finalChan!=midiBaseChan); | 
					
						
							| 
									
										
										
										
											2022-04-28 06:04:34 -04:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |   chan[candidate].midiNote=note; | 
					
						
							|  |  |  |  |   chan[candidate].midiAge=midiAgeCounter++; | 
					
						
							| 
									
										
										
										
											2022-08-27 01:37:32 -04:00
										 |  |  |  |   pendingNotes.push_back(DivNoteEvent(candidate,ins,note,vol,true)); | 
					
						
							| 
									
										
										
										
											2023-10-15 18:02:25 -04:00
										 |  |  |  |   return true; | 
					
						
							| 
									
										
										
										
											2022-03-31 02:51:57 -04:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | void DivEngine::autoNoteOff(int ch, int note, int vol) { | 
					
						
							|  |  |  |  |   if (!playing) { | 
					
						
							| 
									
										
										
										
											2023-07-15 03:59:06 -04:00
										 |  |  |  |     return; | 
					
						
							| 
									
										
										
										
											2022-03-31 02:51:57 -04:00
										 |  |  |  |   } | 
					
						
							|  |  |  |  |   //if (ch<0 || ch>=chans) return;
 | 
					
						
							|  |  |  |  |   for (int i=0; i<chans; i++) { | 
					
						
							|  |  |  |  |     if (chan[i].midiNote==note) { | 
					
						
							| 
									
										
										
										
											2022-08-27 01:37:32 -04:00
										 |  |  |  |       pendingNotes.push_back(DivNoteEvent(i,-1,-1,-1,false)); | 
					
						
							| 
									
										
										
										
											2022-03-31 02:51:57 -04:00
										 |  |  |  |       chan[i].midiNote=-1; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  |   } | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-16 01:10:52 -04:00
										 |  |  |  | void DivEngine::autoNoteOffAll() { | 
					
						
							|  |  |  |  |   if (!playing) { | 
					
						
							| 
									
										
										
										
											2023-07-15 03:59:06 -04:00
										 |  |  |  |     return; | 
					
						
							| 
									
										
										
										
											2022-04-16 01:10:52 -04:00
										 |  |  |  |   } | 
					
						
							|  |  |  |  |   for (int i=0; i<chans; i++) { | 
					
						
							|  |  |  |  |     if (chan[i].midiNote!=-1) { | 
					
						
							| 
									
										
										
										
											2022-08-27 01:37:32 -04:00
										 |  |  |  |       pendingNotes.push_back(DivNoteEvent(i,-1,-1,-1,false)); | 
					
						
							| 
									
										
										
										
											2022-04-16 01:10:52 -04:00
										 |  |  |  |       chan[i].midiNote=-1; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  |   } | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-06-03 19:05:07 -04:00
										 |  |  |  | void DivEngine::setAutoNotePoly(bool poly) { | 
					
						
							|  |  |  |  |   midiPoly=poly; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-12-11 03:34:43 -05:00
										 |  |  |  | void DivEngine::setOrder(unsigned char order) { | 
					
						
							| 
									
										
										
										
											2022-03-23 22:38:28 -04:00
										 |  |  |  |   BUSY_BEGIN_SOFT; | 
					
						
							| 
									
										
										
										
											2021-12-11 03:34:43 -05:00
										 |  |  |  |   curOrder=order; | 
					
						
							| 
									
										
										
										
											2022-05-15 02:42:49 -04:00
										 |  |  |  |   if (order>=curSubSong->ordersLen) curOrder=0; | 
					
						
							| 
									
										
										
										
											2022-06-06 02:05:06 -04:00
										 |  |  |  |   prevOrder=curOrder; | 
					
						
							| 
									
										
										
										
											2021-12-28 18:23:57 -05:00
										 |  |  |  |   if (playing && !freelance) { | 
					
						
							| 
									
										
										
										
											2021-12-21 02:02:25 -05:00
										 |  |  |  |     playSub(false); | 
					
						
							| 
									
										
										
										
											2021-12-11 03:34:43 -05:00
										 |  |  |  |   } | 
					
						
							| 
									
										
										
										
											2022-03-23 22:38:28 -04:00
										 |  |  |  |   BUSY_END; | 
					
						
							| 
									
										
										
										
											2021-05-11 16:08:08 -04:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-10-11 03:48:39 -04:00
										 |  |  |  | void DivEngine::updateSysFlags(int system, bool restart, bool render) { | 
					
						
							| 
									
										
										
										
											2022-03-23 22:38:28 -04:00
										 |  |  |  |   BUSY_BEGIN_SOFT; | 
					
						
							| 
									
										
										
										
											2022-09-29 21:13:40 -04:00
										 |  |  |  |   disCont[system].dispatch->setFlags(song.systemFlags[system]); | 
					
						
							| 
									
										
										
										
											2022-01-28 18:12:56 -05:00
										 |  |  |  |   disCont[system].setRates(got.rate); | 
					
						
							| 
									
										
										
										
											2023-10-11 03:48:39 -04:00
										 |  |  |  |   if (render) renderSamples(); | 
					
						
							| 
									
										
										
										
											2023-01-10 15:58:15 -05:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |   // patchbay
 | 
					
						
							|  |  |  |  |   if (song.patchbayAuto) { | 
					
						
							|  |  |  |  |     saveLock.lock(); | 
					
						
							|  |  |  |  |     autoPatchbay(); | 
					
						
							|  |  |  |  |     saveLock.unlock(); | 
					
						
							|  |  |  |  |   } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-08-14 17:28:43 -04:00
										 |  |  |  |   if (restart) { | 
					
						
							|  |  |  |  |     if (isPlaying()) { | 
					
						
							|  |  |  |  |       playSub(false); | 
					
						
							|  |  |  |  |     } else if (freelance) { | 
					
						
							|  |  |  |  |       reset(); | 
					
						
							|  |  |  |  |     } | 
					
						
							| 
									
										
										
										
											2022-02-07 17:24:26 -05:00
										 |  |  |  |   } | 
					
						
							| 
									
										
										
										
											2022-03-23 22:38:28 -04:00
										 |  |  |  |   BUSY_END; | 
					
						
							| 
									
										
										
										
											2022-01-28 18:12:56 -05:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-06-11 19:57:32 -04:00
										 |  |  |  | void DivEngine::setSongRate(float hz) { | 
					
						
							| 
									
										
										
										
											2022-03-23 22:38:28 -04:00
										 |  |  |  |   BUSY_BEGIN; | 
					
						
							| 
									
										
										
										
											2022-03-21 17:17:51 -04:00
										 |  |  |  |   saveLock.lock(); | 
					
						
							| 
									
										
										
										
											2022-05-15 02:42:49 -04:00
										 |  |  |  |   curSubSong->hz=hz; | 
					
						
							| 
									
										
										
										
											2023-06-11 19:57:32 -04:00
										 |  |  |  |   divider=curSubSong->hz; | 
					
						
							| 
									
										
										
										
											2022-03-21 17:17:51 -04:00
										 |  |  |  |   saveLock.unlock(); | 
					
						
							| 
									
										
										
										
											2022-03-23 22:38:28 -04:00
										 |  |  |  |   BUSY_END; | 
					
						
							| 
									
										
										
										
											2021-12-15 17:32:08 -05:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-09 04:33:03 -04:00
										 |  |  |  | void DivEngine::setAudio(DivAudioEngines which) { | 
					
						
							|  |  |  |  |   audioEngine=which; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | void DivEngine::setView(DivStatusView which) { | 
					
						
							|  |  |  |  |   view=which; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-01-04 00:02:41 -05:00
										 |  |  |  | bool DivEngine::getMetronome() { | 
					
						
							|  |  |  |  |   return metronome; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | void DivEngine::setMetronome(bool enable) { | 
					
						
							|  |  |  |  |   metronome=enable; | 
					
						
							|  |  |  |  |   metroAmp=0; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-13 03:29:07 -04:00
										 |  |  |  | void DivEngine::setMetronomeVol(float vol) { | 
					
						
							|  |  |  |  |   metroVol=vol; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-09-11 17:04:19 -04:00
										 |  |  |  | void DivEngine::setSamplePreviewVol(float vol) { | 
					
						
							|  |  |  |  |   previewVol=vol; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-04-23 05:38:08 -04:00
										 |  |  |  | void DivEngine::setConsoleMode(bool enable, bool statusOut) { | 
					
						
							| 
									
										
										
										
											2021-12-18 04:26:17 -05:00
										 |  |  |  |   consoleMode=enable; | 
					
						
							| 
									
										
										
										
											2024-04-23 05:38:08 -04:00
										 |  |  |  |   disableStatusOut=!statusOut; | 
					
						
							| 
									
										
										
										
											2021-12-18 04:26:17 -05:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-12-14 17:18:19 -05:00
										 |  |  |  | bool DivEngine::switchMaster(bool full) { | 
					
						
							| 
									
										
										
										
											2022-09-11 00:32:04 -04:00
										 |  |  |  |   logI("switching output..."); | 
					
						
							|  |  |  |  |   deinitAudioBackend(true); | 
					
						
							| 
									
										
										
										
											2022-12-14 17:18:19 -05:00
										 |  |  |  |   if (full) { | 
					
						
							|  |  |  |  |     quitDispatch(); | 
					
						
							|  |  |  |  |     initDispatch(); | 
					
						
							|  |  |  |  |   } | 
					
						
							| 
									
										
										
										
											2023-09-06 15:23:47 -04:00
										 |  |  |  |   if (renderPool!=NULL) { | 
					
						
							|  |  |  |  |     delete renderPool; | 
					
						
							|  |  |  |  |     renderPool=NULL; | 
					
						
							|  |  |  |  |   } | 
					
						
							| 
									
										
										
										
											2022-01-17 01:42:26 -05:00
										 |  |  |  |   if (initAudioBackend()) { | 
					
						
							|  |  |  |  |     for (int i=0; i<song.systemLen; i++) { | 
					
						
							|  |  |  |  |       disCont[i].setRates(got.rate); | 
					
						
							| 
									
										
										
										
											2023-10-29 16:25:23 -04:00
										 |  |  |  |       disCont[i].setQuality(lowQuality,dcHiPass); | 
					
						
							| 
									
										
										
										
											2022-01-17 01:42:26 -05:00
										 |  |  |  |     } | 
					
						
							|  |  |  |  |     if (!output->setRun(true)) { | 
					
						
							| 
									
										
										
										
											2022-04-10 23:12:02 -04:00
										 |  |  |  |       logE("error while activating audio!"); | 
					
						
							| 
									
										
										
										
											2022-02-05 23:48:56 -05:00
										 |  |  |  |       return false; | 
					
						
							| 
									
										
										
										
											2022-01-17 01:42:26 -05:00
										 |  |  |  |     } | 
					
						
							| 
									
										
										
										
											2022-02-05 23:48:56 -05:00
										 |  |  |  |   } else { | 
					
						
							|  |  |  |  |     return false; | 
					
						
							| 
									
										
										
										
											2022-01-17 01:42:26 -05:00
										 |  |  |  |   } | 
					
						
							| 
									
										
										
										
											2022-05-02 03:54:23 -04:00
										 |  |  |  |   renderSamples(); | 
					
						
							| 
									
										
										
										
											2022-02-05 23:48:56 -05:00
										 |  |  |  |   return true; | 
					
						
							| 
									
										
										
										
											2022-01-17 01:42:26 -05:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-03-31 02:51:57 -04:00
										 |  |  |  | void DivEngine::setMidiBaseChan(int chan) { | 
					
						
							|  |  |  |  |   if (chan<0 || chan>=chans) chan=0; | 
					
						
							|  |  |  |  |   midiBaseChan=chan; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-01 03:21:10 -04:00
										 |  |  |  | void DivEngine::setMidiDirect(bool value) { | 
					
						
							|  |  |  |  |   midiIsDirect=value; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-12-17 15:30:51 -05:00
										 |  |  |  | void DivEngine::setMidiDirectProgram(bool value) { | 
					
						
							|  |  |  |  |   midiIsDirectProgram=value; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-12-16 19:52:37 -05:00
										 |  |  |  | void DivEngine::setMidiVolExp(float value) { | 
					
						
							|  |  |  |  |   midiVolExp=value; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-03-28 19:19:47 -04:00
										 |  |  |  | void DivEngine::setMidiCallback(std::function<int(const TAMidiMessage&)> what) { | 
					
						
							| 
									
										
										
										
											2022-03-28 16:24:09 -04:00
										 |  |  |  |   midiCallback=what; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-01-25 15:29:37 -05:00
										 |  |  |  | void DivEngine::setMidiDebug(bool enable) { | 
					
						
							|  |  |  |  |   midiDebug=enable; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-05-08 03:01:32 -04:00
										 |  |  |  | bool DivEngine::sendMidiMessage(TAMidiMessage& msg) { | 
					
						
							|  |  |  |  |   if (output==NULL) { | 
					
						
							|  |  |  |  |     logW("output is NULL!"); | 
					
						
							|  |  |  |  |     return false; | 
					
						
							|  |  |  |  |   } | 
					
						
							|  |  |  |  |   if (output->midiOut==NULL) { | 
					
						
							|  |  |  |  |     logW("MIDI output is NULL!"); | 
					
						
							|  |  |  |  |     return false; | 
					
						
							|  |  |  |  |   } | 
					
						
							|  |  |  |  |   BUSY_BEGIN; | 
					
						
							|  |  |  |  |   logD("sending MIDI message..."); | 
					
						
							|  |  |  |  |   bool ret=(output->midiOut->send(msg)); | 
					
						
							|  |  |  |  |   BUSY_END; | 
					
						
							|  |  |  |  |   return ret; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-03-19 04:42:44 -04:00
										 |  |  |  | void DivEngine::synchronized(const std::function<void()>& what) { | 
					
						
							| 
									
										
										
										
											2022-03-23 22:38:28 -04:00
										 |  |  |  |   BUSY_BEGIN; | 
					
						
							| 
									
										
										
										
											2022-03-19 04:42:44 -04:00
										 |  |  |  |   what(); | 
					
						
							| 
									
										
										
										
											2022-03-23 22:38:28 -04:00
										 |  |  |  |   BUSY_END; | 
					
						
							| 
									
										
										
										
											2022-03-19 04:42:44 -04:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-08-13 17:50:13 -04:00
										 |  |  |  | void DivEngine::synchronizedSoft(const std::function<void()>& what) { | 
					
						
							|  |  |  |  |   BUSY_BEGIN_SOFT; | 
					
						
							|  |  |  |  |   what(); | 
					
						
							|  |  |  |  |   BUSY_END; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-03-21 17:17:51 -04:00
										 |  |  |  | void DivEngine::lockSave(const std::function<void()>& what) { | 
					
						
							|  |  |  |  |   saveLock.lock(); | 
					
						
							|  |  |  |  |   what(); | 
					
						
							|  |  |  |  |   saveLock.unlock(); | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | void DivEngine::lockEngine(const std::function<void()>& what) { | 
					
						
							| 
									
										
										
										
											2022-03-23 22:38:28 -04:00
										 |  |  |  |   BUSY_BEGIN; | 
					
						
							| 
									
										
										
										
											2022-03-21 17:17:51 -04:00
										 |  |  |  |   saveLock.lock(); | 
					
						
							|  |  |  |  |   what(); | 
					
						
							|  |  |  |  |   saveLock.unlock(); | 
					
						
							| 
									
										
										
										
											2022-03-23 22:38:28 -04:00
										 |  |  |  |   BUSY_END; | 
					
						
							| 
									
										
										
										
											2022-03-21 17:17:51 -04:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-02-05 21:26:24 -05:00
										 |  |  |  | TAAudioDesc& DivEngine::getAudioDescWant() { | 
					
						
							|  |  |  |  |   return want; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | TAAudioDesc& DivEngine::getAudioDescGot() { | 
					
						
							|  |  |  |  |   return got; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-02-13 21:42:57 -05:00
										 |  |  |  | std::vector<String>& DivEngine::getAudioDevices() { | 
					
						
							|  |  |  |  |   return audioDevs; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-03-28 04:46:50 -04:00
										 |  |  |  | std::vector<String>& DivEngine::getMidiIns() { | 
					
						
							|  |  |  |  |   return midiIns; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | std::vector<String>& DivEngine::getMidiOuts() { | 
					
						
							|  |  |  |  |   return midiOuts; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-02-13 21:42:57 -05:00
										 |  |  |  | void DivEngine::rescanAudioDevices() { | 
					
						
							|  |  |  |  |   audioDevs.clear(); | 
					
						
							|  |  |  |  |   if (output!=NULL) { | 
					
						
							|  |  |  |  |     audioDevs=output->listAudioDevices(); | 
					
						
							| 
									
										
										
										
											2023-12-28 04:11:30 -05:00
										 |  |  |  |   } | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | void DivEngine::rescanMidiDevices() { | 
					
						
							|  |  |  |  |   if (output!=NULL) { | 
					
						
							|  |  |  |  |     logV("re-scanning midi..."); | 
					
						
							| 
									
										
										
										
											2022-03-28 04:46:50 -04:00
										 |  |  |  |     if (output->midiIn!=NULL) { | 
					
						
							|  |  |  |  |       midiIns=output->midiIn->listDevices(); | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  |     if (output->midiOut!=NULL) { | 
					
						
							|  |  |  |  |       midiOuts=output->midiOut->listDevices(); | 
					
						
							|  |  |  |  |     } | 
					
						
							| 
									
										
										
										
											2022-02-13 21:42:57 -05:00
										 |  |  |  |   } | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-08-18 02:40:18 -04:00
										 |  |  |  | void DivEngine::initDispatch(bool isRender) { | 
					
						
							| 
									
										
										
										
											2022-03-23 22:38:28 -04:00
										 |  |  |  |   BUSY_BEGIN; | 
					
						
							| 
									
										
										
										
											2023-08-26 20:18:39 -04:00
										 |  |  |  |   logV("initializing dispatch..."); | 
					
						
							| 
									
										
										
										
											2023-08-18 02:40:18 -04:00
										 |  |  |  |   if (isRender) logI("render cores set"); | 
					
						
							| 
									
										
										
										
											2023-10-29 16:25:23 -04:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |   lowQuality=getConfInt("audioQuality",0); | 
					
						
							|  |  |  |  |   dcHiPass=getConfInt("audioHiPass",1); | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-04-08 00:12:27 -04:00
										 |  |  |  |   if (lowQuality) { | 
					
						
							|  |  |  |  |     blip_add_delta=blip_add_delta_fast; | 
					
						
							|  |  |  |  |   } else { | 
					
						
							|  |  |  |  |     blip_add_delta=blip_add_delta_slow; | 
					
						
							|  |  |  |  |   } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-01-08 16:03:32 -05:00
										 |  |  |  |   for (int i=0; i<song.systemLen; i++) { | 
					
						
							| 
									
										
										
										
											2023-08-18 02:40:18 -04:00
										 |  |  |  |     disCont[i].init(song.system[i],this,getChannelCount(song.system[i]),got.rate,song.systemFlags[i],isRender); | 
					
						
							| 
									
										
										
										
											2022-01-08 16:03:32 -05:00
										 |  |  |  |     disCont[i].setRates(got.rate); | 
					
						
							| 
									
										
										
										
											2023-10-29 16:25:23 -04:00
										 |  |  |  |     disCont[i].setQuality(lowQuality,dcHiPass); | 
					
						
							| 
									
										
										
										
											2021-12-13 14:40:03 -05:00
										 |  |  |  |   } | 
					
						
							| 
									
										
										
										
											2023-01-10 19:09:26 -05:00
										 |  |  |  |   if (song.patchbayAuto) { | 
					
						
							|  |  |  |  |     saveLock.lock(); | 
					
						
							|  |  |  |  |     autoPatchbay(); | 
					
						
							|  |  |  |  |     saveLock.unlock(); | 
					
						
							|  |  |  |  |   } | 
					
						
							| 
									
										
										
										
											2022-01-08 16:03:32 -05:00
										 |  |  |  |   recalcChans(); | 
					
						
							| 
									
										
										
										
											2022-03-23 22:38:28 -04:00
										 |  |  |  |   BUSY_END; | 
					
						
							| 
									
										
										
										
											2021-12-13 14:40:03 -05:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | void DivEngine::quitDispatch() { | 
					
						
							| 
									
										
										
										
											2022-03-23 22:38:28 -04:00
										 |  |  |  |   BUSY_BEGIN; | 
					
						
							| 
									
										
										
										
											2023-08-26 20:18:39 -04:00
										 |  |  |  |   logV("terminating dispatch..."); | 
					
						
							| 
									
										
										
										
											2022-01-08 16:03:32 -05:00
										 |  |  |  |   for (int i=0; i<song.systemLen; i++) { | 
					
						
							|  |  |  |  |     disCont[i].quit(); | 
					
						
							|  |  |  |  |   } | 
					
						
							| 
									
										
										
										
											2022-01-12 17:45:07 -05:00
										 |  |  |  |   cycles=0; | 
					
						
							|  |  |  |  |   clockDrift=0; | 
					
						
							| 
									
										
										
										
											2023-05-09 06:05:53 -04:00
										 |  |  |  |   midiClockCycles=0; | 
					
						
							|  |  |  |  |   midiClockDrift=0; | 
					
						
							|  |  |  |  |   midiTimeCycles=0; | 
					
						
							|  |  |  |  |   midiTimeDrift=0; | 
					
						
							| 
									
										
										
										
											2021-12-15 00:37:27 -05:00
										 |  |  |  |   chans=0; | 
					
						
							|  |  |  |  |   playing=false; | 
					
						
							| 
									
										
										
										
											2023-02-05 02:56:39 -05:00
										 |  |  |  |   curSpeed=0; | 
					
						
							| 
									
										
										
										
											2021-12-15 00:37:27 -05:00
										 |  |  |  |   endOfSong=false; | 
					
						
							| 
									
										
										
										
											2023-09-10 04:00:55 -04:00
										 |  |  |  |   stepPlay=0; | 
					
						
							| 
									
										
										
										
											2021-12-15 00:37:27 -05:00
										 |  |  |  |   ticks=0; | 
					
						
							| 
									
										
										
										
											2022-05-18 01:05:25 -04:00
										 |  |  |  |   tempoAccum=0; | 
					
						
							| 
									
										
										
										
											2021-12-15 00:37:27 -05:00
										 |  |  |  |   curRow=0; | 
					
						
							|  |  |  |  |   curOrder=0; | 
					
						
							| 
									
										
										
										
											2022-06-06 02:05:06 -04:00
										 |  |  |  |   prevRow=0; | 
					
						
							|  |  |  |  |   prevOrder=0; | 
					
						
							| 
									
										
										
										
											2021-12-15 00:37:27 -05:00
										 |  |  |  |   nextSpeed=3; | 
					
						
							|  |  |  |  |   changeOrd=-1; | 
					
						
							|  |  |  |  |   changePos=0; | 
					
						
							|  |  |  |  |   totalTicks=0; | 
					
						
							| 
									
										
										
										
											2024-12-09 13:56:20 -05:00
										 |  |  |  |   totalTicksOff=0; | 
					
						
							| 
									
										
										
										
											2022-01-12 02:45:26 -05:00
										 |  |  |  |   totalSeconds=0; | 
					
						
							|  |  |  |  |   totalTicksR=0; | 
					
						
							| 
									
										
										
										
											2023-05-09 23:12:14 -04:00
										 |  |  |  |   curMidiClock=0; | 
					
						
							|  |  |  |  |   curMidiTime=0; | 
					
						
							| 
									
										
										
										
											2023-05-10 03:57:59 -04:00
										 |  |  |  |   curMidiTimeCode=0; | 
					
						
							|  |  |  |  |   curMidiTimePiece=0; | 
					
						
							| 
									
										
										
										
											2021-12-15 00:37:27 -05:00
										 |  |  |  |   totalCmds=0; | 
					
						
							|  |  |  |  |   lastCmds=0; | 
					
						
							|  |  |  |  |   cmdsPerSecond=0; | 
					
						
							| 
									
										
										
										
											2022-01-08 01:57:37 -05:00
										 |  |  |  |   for (int i=0; i<DIV_MAX_CHANS; i++) { | 
					
						
							| 
									
										
										
										
											2021-12-18 03:25:42 -05:00
										 |  |  |  |     isMuted[i]=0; | 
					
						
							|  |  |  |  |   } | 
					
						
							| 
									
										
										
										
											2023-09-07 02:16:27 -04:00
										 |  |  |  |   if (renderPool!=NULL) { | 
					
						
							|  |  |  |  |     delete renderPool; | 
					
						
							|  |  |  |  |     renderPool=NULL; | 
					
						
							|  |  |  |  |   } | 
					
						
							| 
									
										
										
										
											2022-03-23 22:38:28 -04:00
										 |  |  |  |   BUSY_END; | 
					
						
							| 
									
										
										
										
											2021-12-13 14:40:03 -05:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-01-17 01:20:02 -05:00
										 |  |  |  | bool DivEngine::initAudioBackend() { | 
					
						
							| 
									
										
										
										
											2022-01-17 01:42:26 -05:00
										 |  |  |  |   // load values
 | 
					
						
							| 
									
										
										
										
											2022-09-11 00:32:04 -04:00
										 |  |  |  |   logI("initializing audio."); | 
					
						
							| 
									
										
										
										
											2022-01-22 23:50:49 -05:00
										 |  |  |  |   if (audioEngine==DIV_AUDIO_NULL) { | 
					
						
							|  |  |  |  |     if (getConfString("audioEngine","SDL")=="JACK") { | 
					
						
							|  |  |  |  |       audioEngine=DIV_AUDIO_JACK; | 
					
						
							| 
									
										
										
										
											2023-08-30 18:32:51 -04:00
										 |  |  |  |     } else if (getConfString("audioEngine","SDL")=="PortAudio") { | 
					
						
							|  |  |  |  |       audioEngine=DIV_AUDIO_PORTAUDIO; | 
					
						
							| 
									
										
										
										
											2022-01-22 23:50:49 -05:00
										 |  |  |  |     } else { | 
					
						
							|  |  |  |  |       audioEngine=DIV_AUDIO_SDL; | 
					
						
							|  |  |  |  |     } | 
					
						
							| 
									
										
										
										
											2022-01-17 01:42:26 -05:00
										 |  |  |  |   } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-07-06 22:14:25 -04:00
										 |  |  |  | #ifdef HAVE_SDL2
 | 
					
						
							|  |  |  |  |   if (audioEngine==DIV_AUDIO_SDL) { | 
					
						
							|  |  |  |  |     String audioDriver=getConfString("sdlAudioDriver",""); | 
					
						
							|  |  |  |  |     if (!audioDriver.empty()) { | 
					
						
							|  |  |  |  |       SDL_SetHint("SDL_HINT_AUDIODRIVER",audioDriver.c_str()); | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  |   } | 
					
						
							|  |  |  |  | #endif
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-02-04 17:04:36 -05:00
										 |  |  |  |   forceMono=getConfInt("forceMono",0); | 
					
						
							| 
									
										
										
										
											2022-07-25 19:41:47 -04:00
										 |  |  |  |   clampSamples=getConfInt("clampSamples",0); | 
					
						
							| 
									
										
										
										
											2022-04-15 06:37:23 -04:00
										 |  |  |  |   lowLatency=getConfInt("lowLatency",0); | 
					
						
							| 
									
										
										
										
											2022-04-13 03:29:07 -04:00
										 |  |  |  |   metroVol=(float)(getConfInt("metroVol",100))/100.0f; | 
					
						
							| 
									
										
										
										
											2023-09-11 17:04:19 -04:00
										 |  |  |  |   previewVol=(float)(getConfInt("sampleVol",50))/100.0f; | 
					
						
							| 
									
										
										
										
											2022-09-26 02:27:36 -04:00
										 |  |  |  |   midiOutClock=getConfInt("midiOutClock",0); | 
					
						
							| 
									
										
										
										
											2023-05-10 03:57:59 -04:00
										 |  |  |  |   midiOutTime=getConfInt("midiOutTime",0); | 
					
						
							|  |  |  |  |   midiOutTimeRate=getConfInt("midiOutTimeRate",0); | 
					
						
							| 
									
										
										
										
											2023-07-06 22:14:25 -04:00
										 |  |  |  |   midiOutProgramChange=getConfInt("midiOutProgramChange",0); | 
					
						
							| 
									
										
										
										
											2022-09-26 02:27:36 -04:00
										 |  |  |  |   midiOutMode=getConfInt("midiOutMode",DIV_MIDI_MODE_NOTE); | 
					
						
							| 
									
										
										
										
											2022-04-13 03:29:07 -04:00
										 |  |  |  |   if (metroVol<0.0f) metroVol=0.0f; | 
					
						
							|  |  |  |  |   if (metroVol>2.0f) metroVol=2.0f; | 
					
						
							| 
									
										
										
										
											2023-09-11 17:04:19 -04:00
										 |  |  |  |   if (previewVol<0.0f) previewVol=0.0f; | 
					
						
							|  |  |  |  |   if (previewVol>1.0f) previewVol=1.0f; | 
					
						
							| 
									
										
										
										
											2023-09-06 15:23:47 -04:00
										 |  |  |  |   renderPoolThreads=getConfInt("renderPoolThreads",0); | 
					
						
							| 
									
										
										
										
											2022-01-17 01:42:26 -05:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-15 15:38:25 -04:00
										 |  |  |  |   if (lowLatency) logI("using low latency mode."); | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-01-17 01:20:02 -05:00
										 |  |  |  |   switch (audioEngine) { | 
					
						
							|  |  |  |  |     case DIV_AUDIO_JACK: | 
					
						
							|  |  |  |  | #ifndef HAVE_JACK
 | 
					
						
							| 
									
										
										
										
											2022-04-10 23:12:02 -04:00
										 |  |  |  |       logE("Furnace was not compiled with JACK support!"); | 
					
						
							| 
									
										
										
										
											2022-01-17 01:20:02 -05:00
										 |  |  |  |       setConf("audioEngine","SDL"); | 
					
						
							|  |  |  |  |       saveConf(); | 
					
						
							| 
									
										
										
										
											2022-05-22 20:01:50 -04:00
										 |  |  |  | #ifdef HAVE_SDL2
 | 
					
						
							| 
									
										
										
										
											2022-01-17 01:20:02 -05:00
										 |  |  |  |       output=new TAAudioSDL; | 
					
						
							| 
									
										
										
										
											2022-05-22 20:01:50 -04:00
										 |  |  |  | #else
 | 
					
						
							|  |  |  |  |       logE("Furnace was not compiled with SDL support either!"); | 
					
						
							|  |  |  |  |       output=new TAAudio; | 
					
						
							|  |  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2022-01-17 01:20:02 -05:00
										 |  |  |  | #else
 | 
					
						
							|  |  |  |  |       output=new TAAudioJACK; | 
					
						
							| 
									
										
										
										
											2023-08-30 18:32:51 -04:00
										 |  |  |  | #endif
 | 
					
						
							|  |  |  |  |       break; | 
					
						
							|  |  |  |  |     case DIV_AUDIO_PORTAUDIO: | 
					
						
							|  |  |  |  | #ifndef HAVE_PA
 | 
					
						
							|  |  |  |  |       logE("Furnace was not compiled with PortAudio!"); | 
					
						
							|  |  |  |  |       setConf("audioEngine","SDL"); | 
					
						
							|  |  |  |  |       saveConf(); | 
					
						
							|  |  |  |  | #ifdef HAVE_SDL2
 | 
					
						
							|  |  |  |  |       output=new TAAudioSDL; | 
					
						
							|  |  |  |  | #else
 | 
					
						
							|  |  |  |  |       logE("Furnace was not compiled with SDL support either!"); | 
					
						
							|  |  |  |  |       output=new TAAudio; | 
					
						
							|  |  |  |  | #endif
 | 
					
						
							|  |  |  |  | #else
 | 
					
						
							|  |  |  |  |       output=new TAAudioPA; | 
					
						
							| 
									
										
										
										
											2022-01-17 01:20:02 -05:00
										 |  |  |  | #endif
 | 
					
						
							|  |  |  |  |       break; | 
					
						
							|  |  |  |  |     case DIV_AUDIO_SDL: | 
					
						
							| 
									
										
										
										
											2022-05-22 20:01:50 -04:00
										 |  |  |  | #ifdef HAVE_SDL2
 | 
					
						
							| 
									
										
										
										
											2022-01-17 01:20:02 -05:00
										 |  |  |  |       output=new TAAudioSDL; | 
					
						
							| 
									
										
										
										
											2022-05-22 20:01:50 -04:00
										 |  |  |  | #else
 | 
					
						
							|  |  |  |  |       logE("Furnace was not compiled with SDL support!"); | 
					
						
							|  |  |  |  |       output=new TAAudio; | 
					
						
							|  |  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2022-01-17 01:20:02 -05:00
										 |  |  |  |       break; | 
					
						
							| 
									
										
										
										
											2024-04-23 05:38:08 -04:00
										 |  |  |  |     case DIV_AUDIO_PIPE: | 
					
						
							|  |  |  |  |       output=new TAAudioPipe; | 
					
						
							|  |  |  |  |       break; | 
					
						
							| 
									
										
										
										
											2022-01-22 23:50:49 -05:00
										 |  |  |  |     case DIV_AUDIO_DUMMY: | 
					
						
							|  |  |  |  |       output=new TAAudio; | 
					
						
							|  |  |  |  |       break; | 
					
						
							| 
									
										
										
										
											2022-01-17 01:20:02 -05:00
										 |  |  |  |     default: | 
					
						
							| 
									
										
										
										
											2022-04-10 23:12:02 -04:00
										 |  |  |  |       logE("invalid audio engine!"); | 
					
						
							| 
									
										
										
										
											2022-01-17 01:20:02 -05:00
										 |  |  |  |       return false; | 
					
						
							|  |  |  |  |   } | 
					
						
							| 
									
										
										
										
											2022-02-13 21:42:57 -05:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-08-26 20:18:39 -04:00
										 |  |  |  |   logV("listing audio devices"); | 
					
						
							| 
									
										
										
										
											2022-02-13 21:42:57 -05:00
										 |  |  |  |   audioDevs=output->listAudioDevices(); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   want.deviceName=getConfString("audioDevice",""); | 
					
						
							| 
									
										
										
										
											2022-01-17 01:20:02 -05:00
										 |  |  |  |   want.bufsize=getConfInt("audioBufSize",1024); | 
					
						
							|  |  |  |  |   want.rate=getConfInt("audioRate",44100); | 
					
						
							|  |  |  |  |   want.fragments=2; | 
					
						
							|  |  |  |  |   want.inChans=0; | 
					
						
							| 
									
										
										
										
											2023-01-05 02:40:17 -05:00
										 |  |  |  |   want.outChans=getConfInt("audioChans",2); | 
					
						
							| 
									
										
										
										
											2022-01-17 01:20:02 -05:00
										 |  |  |  |   want.outFormat=TA_AUDIO_FORMAT_F32; | 
					
						
							| 
									
										
										
										
											2023-08-31 05:30:49 -04:00
										 |  |  |  |   want.wasapiEx=getConfInt("wasapiEx",0); | 
					
						
							| 
									
										
										
										
											2022-01-17 01:20:02 -05:00
										 |  |  |  |   want.name="Furnace"; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-01-05 02:40:17 -05:00
										 |  |  |  |   if (want.outChans<1) want.outChans=1; | 
					
						
							|  |  |  |  |   if (want.outChans>16) want.outChans=16; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-08-26 20:18:39 -04:00
										 |  |  |  |   logV("setting callback"); | 
					
						
							| 
									
										
										
										
											2022-01-17 01:20:02 -05:00
										 |  |  |  |   output->setCallback(process,this); | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-08-26 20:18:39 -04:00
										 |  |  |  |   logV("calling init"); | 
					
						
							| 
									
										
										
										
											2022-01-17 01:20:02 -05:00
										 |  |  |  |   if (!output->init(want,got)) { | 
					
						
							| 
									
										
										
										
											2022-04-10 23:12:02 -04:00
										 |  |  |  |     logE("error while initializing audio!"); | 
					
						
							| 
									
										
										
										
											2022-01-17 01:42:26 -05:00
										 |  |  |  |     delete output; | 
					
						
							|  |  |  |  |     output=NULL; | 
					
						
							| 
									
										
										
										
											2022-02-06 17:00:01 -05:00
										 |  |  |  |     audioEngine=DIV_AUDIO_NULL; | 
					
						
							| 
									
										
										
										
											2022-01-17 01:20:02 -05:00
										 |  |  |  |     return false; | 
					
						
							|  |  |  |  |   } | 
					
						
							| 
									
										
										
										
											2022-01-27 17:49:00 -05:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-08-26 20:18:39 -04:00
										 |  |  |  |   logV("allocating oscBuf..."); | 
					
						
							| 
									
										
										
										
											2023-01-05 02:40:17 -05:00
										 |  |  |  |   for (int i=0; i<got.outChans; i++) { | 
					
						
							|  |  |  |  |     if (oscBuf[i]==NULL) { | 
					
						
							|  |  |  |  |       oscBuf[i]=new float[32768]; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  |     memset(oscBuf[i],0,32768*sizeof(float)); | 
					
						
							|  |  |  |  |   } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-08-26 20:18:39 -04:00
										 |  |  |  |   logI("initializing MIDI."); | 
					
						
							| 
									
										
										
										
											2022-03-28 04:46:50 -04:00
										 |  |  |  |   if (output->initMidi(false)) { | 
					
						
							|  |  |  |  |     midiIns=output->midiIn->listDevices(); | 
					
						
							|  |  |  |  |     midiOuts=output->midiOut->listDevices(); | 
					
						
							|  |  |  |  |   } else { | 
					
						
							| 
									
										
										
										
											2022-04-10 23:12:02 -04:00
										 |  |  |  |     logW("error while initializing MIDI!"); | 
					
						
							| 
									
										
										
										
											2022-03-28 04:46:50 -04:00
										 |  |  |  |   } | 
					
						
							|  |  |  |  |   if (output->midiIn) { | 
					
						
							|  |  |  |  |     String inName=getConfString("midiInDevice",""); | 
					
						
							|  |  |  |  |     if (!inName.empty()) { | 
					
						
							|  |  |  |  |       // try opening device
 | 
					
						
							| 
									
										
										
										
											2022-04-10 23:12:02 -04:00
										 |  |  |  |       logI("opening MIDI input."); | 
					
						
							| 
									
										
										
										
											2022-03-28 04:46:50 -04:00
										 |  |  |  |       if (!output->midiIn->openDevice(inName)) { | 
					
						
							| 
									
										
										
										
											2022-04-10 23:12:02 -04:00
										 |  |  |  |         logW("could not open MIDI input device!"); | 
					
						
							| 
									
										
										
										
											2022-03-28 04:46:50 -04:00
										 |  |  |  |       } | 
					
						
							| 
									
										
										
										
											2022-06-17 00:55:17 -04:00
										 |  |  |  |     } else { | 
					
						
							|  |  |  |  |       logV("no MIDI input device selected."); | 
					
						
							| 
									
										
										
										
											2022-03-28 04:46:50 -04:00
										 |  |  |  |     } | 
					
						
							|  |  |  |  |   } | 
					
						
							| 
									
										
										
										
											2022-03-31 04:33:05 -04:00
										 |  |  |  |   if (output->midiOut) { | 
					
						
							|  |  |  |  |     String outName=getConfString("midiOutDevice",""); | 
					
						
							|  |  |  |  |     if (!outName.empty()) { | 
					
						
							|  |  |  |  |       // try opening device
 | 
					
						
							| 
									
										
										
										
											2022-04-10 23:12:02 -04:00
										 |  |  |  |       logI("opening MIDI output."); | 
					
						
							| 
									
										
										
										
											2022-03-31 04:33:05 -04:00
										 |  |  |  |       if (!output->midiOut->openDevice(outName)) { | 
					
						
							| 
									
										
										
										
											2022-04-10 23:12:02 -04:00
										 |  |  |  |         logW("could not open MIDI output device!"); | 
					
						
							| 
									
										
										
										
											2022-03-31 04:33:05 -04:00
										 |  |  |  |       } | 
					
						
							| 
									
										
										
										
											2022-06-17 00:55:17 -04:00
										 |  |  |  |     } else { | 
					
						
							|  |  |  |  |       logV("no MIDI output device selected."); | 
					
						
							| 
									
										
										
										
											2022-03-31 04:33:05 -04:00
										 |  |  |  |     } | 
					
						
							|  |  |  |  |   } | 
					
						
							| 
									
										
										
										
											2022-03-28 04:46:50 -04:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-08-26 20:18:39 -04:00
										 |  |  |  |   logV("initAudioBackend done"); | 
					
						
							| 
									
										
										
										
											2022-01-17 01:20:02 -05:00
										 |  |  |  |   return true; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-11 00:32:04 -04:00
										 |  |  |  | bool DivEngine::deinitAudioBackend(bool dueToSwitchMaster) { | 
					
						
							| 
									
										
										
										
											2022-01-17 01:20:02 -05:00
										 |  |  |  |   if (output!=NULL) { | 
					
						
							| 
									
										
										
										
											2022-09-11 00:32:04 -04:00
										 |  |  |  |     logI("closing audio output."); | 
					
						
							| 
									
										
										
										
											2022-08-07 18:37:07 -04:00
										 |  |  |  |     output->quit(); | 
					
						
							| 
									
										
										
										
											2022-03-28 04:46:50 -04:00
										 |  |  |  |     if (output->midiIn) { | 
					
						
							|  |  |  |  |       if (output->midiIn->isDeviceOpen()) { | 
					
						
							| 
									
										
										
										
											2022-04-10 23:12:02 -04:00
										 |  |  |  |         logI("closing MIDI input."); | 
					
						
							| 
									
										
										
										
											2022-03-28 04:46:50 -04:00
										 |  |  |  |         output->midiIn->closeDevice(); | 
					
						
							|  |  |  |  |       } | 
					
						
							|  |  |  |  |     } | 
					
						
							| 
									
										
										
										
											2022-03-31 04:33:05 -04:00
										 |  |  |  |     if (output->midiOut) { | 
					
						
							|  |  |  |  |       if (output->midiOut->isDeviceOpen()) { | 
					
						
							| 
									
										
										
										
											2022-04-10 23:12:02 -04:00
										 |  |  |  |         logI("closing MIDI output."); | 
					
						
							| 
									
										
										
										
											2022-03-31 04:33:05 -04:00
										 |  |  |  |         output->midiOut->closeDevice(); | 
					
						
							|  |  |  |  |       } | 
					
						
							|  |  |  |  |     } | 
					
						
							| 
									
										
										
										
											2022-03-28 04:46:50 -04:00
										 |  |  |  |     output->quitMidi(); | 
					
						
							| 
									
										
										
										
											2022-01-17 01:20:02 -05:00
										 |  |  |  |     delete output; | 
					
						
							|  |  |  |  |     output=NULL; | 
					
						
							| 
									
										
										
										
											2022-09-11 00:32:04 -04:00
										 |  |  |  |     if (dueToSwitchMaster) { | 
					
						
							|  |  |  |  |       audioEngine=DIV_AUDIO_NULL; | 
					
						
							|  |  |  |  |     } | 
					
						
							| 
									
										
										
										
											2022-01-17 01:20:02 -05:00
										 |  |  |  |   } | 
					
						
							|  |  |  |  |   return true; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-06-01 17:36:00 -04:00
										 |  |  |  | bool DivEngine::prePreInit() { | 
					
						
							|  |  |  |  |   // init config
 | 
					
						
							|  |  |  |  |   initConfDir(); | 
					
						
							|  |  |  |  |   logD("config path: %s",configPath.c_str()); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   configLoaded=true; | 
					
						
							|  |  |  |  |   return loadConf(); | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-10-16 15:55:01 -04:00
										 |  |  |  | bool DivEngine::preInit(bool noSafeMode) { | 
					
						
							|  |  |  |  |   bool wantSafe=false; | 
					
						
							| 
									
										
										
										
											2024-06-01 17:36:00 -04:00
										 |  |  |  |   if (!configLoaded) prePreInit(); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   logI("Furnace version " DIV_VERSION "."); | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-26 19:32:33 -04:00
										 |  |  |  |   // register systems
 | 
					
						
							| 
									
										
										
										
											2022-04-27 01:56:15 -04:00
										 |  |  |  |   if (!systemsRegistered) registerSystems(); | 
					
						
							| 
									
										
										
										
											2022-07-23 23:11:30 -04:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-08-13 05:40:26 -04:00
										 |  |  |  |   // register ROM exports
 | 
					
						
							|  |  |  |  |   if (!romExportsRegistered) registerROMExports(); | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-01-25 19:48:49 -05:00
										 |  |  |  |   // TODO: re-enable with a better approach
 | 
					
						
							|  |  |  |  |   // see issue #1581
 | 
					
						
							|  |  |  |  |   /*
 | 
					
						
							| 
									
										
										
										
											2023-10-16 15:55:01 -04:00
										 |  |  |  |   if (!noSafeMode) { | 
					
						
							|  |  |  |  |     String safeModePath=configPath+DIR_SEPARATOR_STR+"safemode"; | 
					
						
							|  |  |  |  |     if (touchFile(safeModePath.c_str())==-EEXIST) { | 
					
						
							|  |  |  |  |       wantSafe=true; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  |   } | 
					
						
							| 
									
										
										
										
											2024-01-25 19:48:49 -05:00
										 |  |  |  |   */ | 
					
						
							| 
									
										
										
										
											2023-10-16 15:55:01 -04:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-12-18 01:55:21 -05:00
										 |  |  |  |   String logPath=configPath+DIR_SEPARATOR_STR+"furnace.log"; | 
					
						
							|  |  |  |  |   startLogFile(logPath.c_str()); | 
					
						
							| 
									
										
										
										
											2023-02-21 22:55:07 -05:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-04-26 00:18:02 -04:00
										 |  |  |  |   if (!conf.has("opn1Core")) { | 
					
						
							|  |  |  |  |     if (conf.has("opnCore")) { | 
					
						
							|  |  |  |  |       conf.set("opn1Core",conf.getString("opnCore","")); | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  |   } | 
					
						
							|  |  |  |  |   if (!conf.has("opnaCore")) { | 
					
						
							|  |  |  |  |     if (conf.has("opnCore")) { | 
					
						
							|  |  |  |  |       conf.set("opnaCore",conf.getString("opnCore","")); | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  |   } | 
					
						
							|  |  |  |  |   if (!conf.has("opnbCore")) { | 
					
						
							|  |  |  |  |     if (conf.has("opnCore")) { | 
					
						
							|  |  |  |  |       conf.set("opnbCore",conf.getString("opnCore","")); | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  |   } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-08-26 20:18:39 -04:00
										 |  |  |  | #ifdef HAVE_SDL2
 | 
					
						
							|  |  |  |  |   String audioDriver=getConfString("sdlAudioDriver",""); | 
					
						
							|  |  |  |  |   if (!audioDriver.empty()) { | 
					
						
							|  |  |  |  |     SDL_SetHint("SDL_HINT_AUDIODRIVER",audioDriver.c_str()); | 
					
						
							|  |  |  |  |   } | 
					
						
							|  |  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2023-10-16 15:55:01 -04:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |   if (wantSafe) { | 
					
						
							|  |  |  |  |     logW("requesting safe mode."); | 
					
						
							|  |  |  |  |   } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   return wantSafe; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | void DivEngine::everythingOK() { | 
					
						
							| 
									
										
										
										
											2024-01-25 19:48:49 -05:00
										 |  |  |  |   // TODO: re-enable with a better approach
 | 
					
						
							|  |  |  |  |   // see issue #1581
 | 
					
						
							|  |  |  |  |   /*
 | 
					
						
							| 
									
										
										
										
											2023-10-16 15:55:01 -04:00
										 |  |  |  |   String safeModePath=configPath+DIR_SEPARATOR_STR+"safemode"; | 
					
						
							|  |  |  |  |   deleteFile(safeModePath.c_str()); | 
					
						
							| 
									
										
										
										
											2024-01-25 19:48:49 -05:00
										 |  |  |  |   */ | 
					
						
							| 
									
										
										
										
											2022-12-18 01:55:21 -05:00
										 |  |  |  | } | 
					
						
							| 
									
										
										
										
											2021-12-19 16:52:04 -05:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-12-18 01:55:21 -05:00
										 |  |  |  | bool DivEngine::init() { | 
					
						
							| 
									
										
										
										
											2022-05-12 23:15:03 -04:00
										 |  |  |  |   loadSampleROMs(); | 
					
						
							| 
									
										
										
										
											2022-05-11 17:39:36 -04:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-30 02:37:37 -04:00
										 |  |  |  |   // set default system preset
 | 
					
						
							|  |  |  |  |   if (!hasLoadedSomething) { | 
					
						
							| 
									
										
										
										
											2022-05-01 18:26:56 -04:00
										 |  |  |  |     logD("setting default preset"); | 
					
						
							| 
									
										
										
										
											2022-09-30 01:26:54 -04:00
										 |  |  |  |     String preset=getConfString("initialSys2",""); | 
					
						
							| 
									
										
										
										
											2023-01-06 17:43:08 -05:00
										 |  |  |  |     bool oldVol=getConfInt("configVersion",DIV_ENGINE_VERSION)<135; | 
					
						
							| 
									
										
										
										
											2022-09-30 01:26:54 -04:00
										 |  |  |  |     if (preset.empty()) { | 
					
						
							|  |  |  |  |       // try loading old preset
 | 
					
						
							| 
									
										
										
										
											2023-02-25 00:43:27 -05:00
										 |  |  |  |       logD("trying to load old preset"); | 
					
						
							| 
									
										
										
										
											2022-09-30 01:26:54 -04:00
										 |  |  |  |       preset=decodeSysDesc(getConfString("initialSys","")); | 
					
						
							| 
									
										
										
										
											2023-01-06 17:43:08 -05:00
										 |  |  |  |       oldVol=false; | 
					
						
							| 
									
										
										
										
											2022-09-30 01:26:54 -04:00
										 |  |  |  |     } | 
					
						
							| 
									
										
										
										
											2022-05-01 18:26:56 -04:00
										 |  |  |  |     logD("preset size %ld",preset.size()); | 
					
						
							| 
									
										
										
										
											2022-04-30 02:37:37 -04:00
										 |  |  |  |     if (preset.size()>0 && (preset.size()&3)==0) { | 
					
						
							| 
									
										
										
										
											2023-01-06 17:43:08 -05:00
										 |  |  |  |       initSongWithDesc(preset.c_str(),true,oldVol); | 
					
						
							| 
									
										
										
										
											2022-04-30 02:37:37 -04:00
										 |  |  |  |     } | 
					
						
							| 
									
										
										
										
											2022-07-23 18:02:03 -04:00
										 |  |  |  |     String sysName=getConfString("initialSysName",""); | 
					
						
							|  |  |  |  |     if (sysName=="") { | 
					
						
							| 
									
										
										
										
											2022-07-27 03:36:48 -04:00
										 |  |  |  |       song.systemName=getSongSystemLegacyName(song,!getConfInt("noMultiSystem",0)); | 
					
						
							| 
									
										
										
										
											2022-07-23 18:02:03 -04:00
										 |  |  |  |     } else { | 
					
						
							|  |  |  |  |       song.systemName=sysName; | 
					
						
							|  |  |  |  |     } | 
					
						
							| 
									
										
										
										
											2022-04-30 02:37:37 -04:00
										 |  |  |  |     hasLoadedSomething=true; | 
					
						
							|  |  |  |  |   } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-12-19 03:16:24 -05:00
										 |  |  |  |   // init the rest of engine
 | 
					
						
							| 
									
										
										
										
											2022-02-05 23:48:56 -05:00
										 |  |  |  |   bool haveAudio=false; | 
					
						
							|  |  |  |  |   if (!initAudioBackend()) { | 
					
						
							| 
									
										
										
										
											2022-04-10 23:12:02 -04:00
										 |  |  |  |     logE("no audio output available!"); | 
					
						
							| 
									
										
										
										
											2022-02-05 23:48:56 -05:00
										 |  |  |  |   } else { | 
					
						
							|  |  |  |  |     haveAudio=true; | 
					
						
							|  |  |  |  |   } | 
					
						
							| 
									
										
										
										
											2021-05-11 16:08:08 -04:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-08-26 20:18:39 -04:00
										 |  |  |  |   logV("creating blip_buf"); | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-01-08 16:03:32 -05:00
										 |  |  |  |   samp_bb=blip_new(32768); | 
					
						
							|  |  |  |  |   if (samp_bb==NULL) { | 
					
						
							| 
									
										
										
										
											2022-04-10 23:12:02 -04:00
										 |  |  |  |     logE("not enough memory!"); | 
					
						
							| 
									
										
										
										
											2021-05-12 04:58:55 -04:00
										 |  |  |  |     return false; | 
					
						
							|  |  |  |  |   } | 
					
						
							| 
									
										
										
										
											2023-10-29 16:25:23 -04:00
										 |  |  |  |   blip_set_dc(samp_bb,0); | 
					
						
							| 
									
										
										
										
											2021-05-12 04:58:55 -04:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-02-05 23:48:56 -05:00
										 |  |  |  |   samp_bbOut=new short[32768]; | 
					
						
							| 
									
										
										
										
											2021-12-21 13:06:14 -05:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-01-08 16:03:32 -05:00
										 |  |  |  |   samp_bbIn=new short[32768]; | 
					
						
							|  |  |  |  |   samp_bbInLen=32768; | 
					
						
							| 
									
										
										
										
											2023-01-12 03:31:43 -05:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |   metroBuf=new float[8192]; | 
					
						
							|  |  |  |  |   metroBufLen=8192; | 
					
						
							| 
									
										
										
										
											2023-08-26 20:18:39 -04:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |   logV("setting blip rate of samp_bb (%f)",got.rate); | 
					
						
							| 
									
										
										
										
											2021-12-21 13:06:14 -05:00
										 |  |  |  |    | 
					
						
							| 
									
										
										
										
											2022-01-08 16:03:32 -05:00
										 |  |  |  |   blip_set_rates(samp_bb,44100,got.rate); | 
					
						
							| 
									
										
										
										
											2021-12-06 05:21:42 -05:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-16 04:03:23 -04:00
										 |  |  |  |   for (int i=0; i<64; i++) { | 
					
						
							|  |  |  |  |     vibTable[i]=127*sin(((double)i/64.0)*(2*M_PI)); | 
					
						
							| 
									
										
										
										
											2021-05-14 15:16:48 -04:00
										 |  |  |  |   } | 
					
						
							| 
									
										
										
										
											2023-02-04 16:08:20 -05:00
										 |  |  |  |   for (int i=0; i<128; i++) { | 
					
						
							|  |  |  |  |     tremTable[i]=255*0.5*(1.0-cos(((double)i/128.0)*(2*M_PI))); | 
					
						
							|  |  |  |  |   } | 
					
						
							| 
									
										
										
										
											2022-03-25 03:52:41 -04:00
										 |  |  |  |   for (int i=0; i<4096; i++) { | 
					
						
							|  |  |  |  |     reversePitchTable[i]=round(1024.0*pow(2.0,(2048.0-(double)i)/(12.0*128.0))); | 
					
						
							|  |  |  |  |     pitchTable[i]=round(1024.0*pow(2.0,((double)i-2048.0)/(12.0*128.0))); | 
					
						
							|  |  |  |  |   } | 
					
						
							| 
									
										
										
										
											2021-05-14 15:16:48 -04:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-01-08 01:57:37 -05:00
										 |  |  |  |   for (int i=0; i<DIV_MAX_CHANS; i++) { | 
					
						
							| 
									
										
										
										
											2021-12-18 03:25:42 -05:00
										 |  |  |  |     isMuted[i]=0; | 
					
						
							| 
									
										
										
										
											2022-02-10 03:15:39 -05:00
										 |  |  |  |     keyHit[i]=false; | 
					
						
							| 
									
										
										
										
											2021-12-18 03:25:42 -05:00
										 |  |  |  |   } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-12-13 14:40:03 -05:00
										 |  |  |  |   initDispatch(); | 
					
						
							| 
									
										
										
										
											2022-05-01 13:57:44 -04:00
										 |  |  |  |   renderSamples(); | 
					
						
							| 
									
										
										
										
											2021-12-11 16:51:34 -05:00
										 |  |  |  |   reset(); | 
					
						
							| 
									
										
										
										
											2021-12-20 14:20:05 -05:00
										 |  |  |  |   active=true; | 
					
						
							| 
									
										
										
										
											2021-05-17 16:06:11 -04:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-02-05 23:48:56 -05:00
										 |  |  |  |   if (!haveAudio) { | 
					
						
							| 
									
										
										
										
											2022-01-16 23:32:13 -05:00
										 |  |  |  |     return false; | 
					
						
							| 
									
										
										
										
											2022-02-05 23:48:56 -05:00
										 |  |  |  |   } else { | 
					
						
							| 
									
										
										
										
											2023-01-03 01:09:46 -05:00
										 |  |  |  |     if (output==NULL) { | 
					
						
							|  |  |  |  |       logE("output is NULL!"); | 
					
						
							|  |  |  |  |       return false; | 
					
						
							|  |  |  |  |     } | 
					
						
							| 
									
										
										
										
											2022-02-05 23:48:56 -05:00
										 |  |  |  |     if (!output->setRun(true)) { | 
					
						
							| 
									
										
										
										
											2022-04-10 23:12:02 -04:00
										 |  |  |  |       logE("error while activating!"); | 
					
						
							| 
									
										
										
										
											2022-02-05 23:48:56 -05:00
										 |  |  |  |       return false; | 
					
						
							|  |  |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-05-11 16:08:08 -04:00
										 |  |  |  |   } | 
					
						
							|  |  |  |  |   return true; | 
					
						
							|  |  |  |  | } | 
					
						
							| 
									
										
										
										
											2021-12-13 17:09:46 -05:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-03-14 02:58:43 -04:00
										 |  |  |  | bool DivEngine::quit(bool saveConfig) { | 
					
						
							| 
									
										
										
										
											2022-01-17 01:20:02 -05:00
										 |  |  |  |   deinitAudioBackend(); | 
					
						
							| 
									
										
										
										
											2021-12-13 17:09:46 -05:00
										 |  |  |  |   quitDispatch(); | 
					
						
							| 
									
										
										
										
											2024-03-14 02:58:43 -04:00
										 |  |  |  |   if (saveConfig) { | 
					
						
							|  |  |  |  |     logI("saving config."); | 
					
						
							|  |  |  |  |     saveConf(); | 
					
						
							|  |  |  |  |   } | 
					
						
							| 
									
										
										
										
											2021-12-20 14:20:05 -05:00
										 |  |  |  |   active=false; | 
					
						
							| 
									
										
										
										
											2023-01-05 02:40:17 -05:00
										 |  |  |  |   for (int i=0; i<DIV_MAX_OUTPUTS; i++) { | 
					
						
							|  |  |  |  |     if (oscBuf[i]!=NULL) delete[] oscBuf[i]; | 
					
						
							|  |  |  |  |   } | 
					
						
							| 
									
										
										
										
											2023-01-12 03:31:43 -05:00
										 |  |  |  |   if (metroBuf!=NULL) { | 
					
						
							|  |  |  |  |     delete[] metroBuf; | 
					
						
							|  |  |  |  |     metroBuf=NULL; | 
					
						
							|  |  |  |  |     metroBufLen=0; | 
					
						
							|  |  |  |  |   } | 
					
						
							| 
									
										
										
										
											2022-05-12 23:15:03 -04:00
										 |  |  |  |   if (yrw801ROM!=NULL) delete[] yrw801ROM; | 
					
						
							|  |  |  |  |   if (tg100ROM!=NULL) delete[] tg100ROM; | 
					
						
							|  |  |  |  |   if (mu5ROM!=NULL) delete[] mu5ROM; | 
					
						
							| 
									
										
										
										
											2022-05-15 02:42:49 -04:00
										 |  |  |  |   song.unload(); | 
					
						
							| 
									
										
										
										
											2021-12-13 17:09:46 -05:00
										 |  |  |  |   return true; | 
					
						
							| 
									
										
										
										
											2021-12-14 12:33:26 -05:00
										 |  |  |  | } |