| 
									
										
										
										
											2022-02-14 22:12:20 -05:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * Furnace Tracker - multi-system chiptune tracker | 
					
						
							|  |  |  |  * Copyright (C) 2021-2022 tildearrow and contributors | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * 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. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-12-15 14:15:44 -05:00
										 |  |  | #include "sample.h"
 | 
					
						
							| 
									
										
										
										
											2021-12-18 01:03:59 -05:00
										 |  |  | #include "../ta-log.h"
 | 
					
						
							| 
									
										
										
										
											2022-03-20 04:14:00 -04:00
										 |  |  | #include <math.h>
 | 
					
						
							| 
									
										
										
										
											2021-12-18 01:03:59 -05:00
										 |  |  | #include <string.h>
 | 
					
						
							|  |  |  | #include <sndfile.h>
 | 
					
						
							| 
									
										
										
										
											2022-03-20 04:14:00 -04:00
										 |  |  | #include "filter.h"
 | 
					
						
							| 
									
										
										
										
											2021-12-18 01:03:59 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-02-24 03:57:45 -05:00
										 |  |  | extern "C" { | 
					
						
							|  |  |  | #include "../../extern/adpcm/bs_codec.h"
 | 
					
						
							|  |  |  | #include "../../extern/adpcm/oki_codec.h"
 | 
					
						
							|  |  |  | #include "../../extern/adpcm/yma_codec.h"
 | 
					
						
							|  |  |  | #include "../../extern/adpcm/ymb_codec.h"
 | 
					
						
							|  |  |  | #include "../../extern/adpcm/ymz_codec.h"
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-03-22 19:36:29 -04:00
										 |  |  | DivSampleHistory::~DivSampleHistory() { | 
					
						
							|  |  |  |   if (data!=NULL) delete[] data; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-12-18 01:03:59 -05:00
										 |  |  | bool DivSample::save(const char* path) { | 
					
						
							|  |  |  |   SNDFILE* f; | 
					
						
							|  |  |  |   SF_INFO si; | 
					
						
							|  |  |  |   memset(&si,0,sizeof(SF_INFO)); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-02-24 03:57:45 -05:00
										 |  |  |   if (length16<1) return false; | 
					
						
							| 
									
										
										
										
											2021-12-18 01:03:59 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |   si.channels=1; | 
					
						
							| 
									
										
										
										
											2021-12-23 18:04:44 -05:00
										 |  |  |   si.samplerate=rate; | 
					
						
							| 
									
										
										
										
											2022-02-24 03:57:45 -05:00
										 |  |  |   si.format=SF_FORMAT_PCM_16|SF_FORMAT_WAV; | 
					
						
							| 
									
										
										
										
											2021-12-18 01:03:59 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |   f=sf_open(path,SFM_WRITE,&si); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (f==NULL) { | 
					
						
							| 
									
										
										
										
											2022-04-10 23:12:02 -04:00
										 |  |  |     logE("could not open wave file for saving! %s",sf_error_number(sf_error(f))); | 
					
						
							| 
									
										
										
										
											2021-12-18 01:03:59 -05:00
										 |  |  |     return false; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-02-23 08:32:35 -05:00
										 |  |  |   SF_INSTRUMENT inst; | 
					
						
							|  |  |  |   memset(&inst, 0, sizeof(inst)); | 
					
						
							|  |  |  |   inst.gain = 1; | 
					
						
							|  |  |  |   short pitch = (0x3c * 100) + 50 - (log2((double)centerRate/rate) * 12.0 * 100.0); | 
					
						
							|  |  |  |   inst.basenote = pitch / 100; | 
					
						
							|  |  |  |   inst.detune = 50 - (pitch % 100); | 
					
						
							|  |  |  |   inst.velocity_hi = 0x7f; | 
					
						
							|  |  |  |   inst.key_hi = 0x7f; | 
					
						
							|  |  |  |   if(loopStart != -1) | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |     inst.loop_count = 1; | 
					
						
							|  |  |  |     inst.loops[0].mode = SF_LOOP_FORWARD; | 
					
						
							|  |  |  |     inst.loops[0].start = loopStart; | 
					
						
							| 
									
										
										
										
											2022-02-24 03:57:45 -05:00
										 |  |  |     inst.loops[0].end = samples; | 
					
						
							| 
									
										
										
										
											2022-02-23 08:32:35 -05:00
										 |  |  |   } | 
					
						
							|  |  |  |   sf_command(f, SFC_SET_INSTRUMENT, &inst, sizeof(inst)); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-02-24 03:57:45 -05:00
										 |  |  |   sf_write_short(f,data16,length16); | 
					
						
							| 
									
										
										
										
											2022-02-23 08:32:35 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-12-18 01:03:59 -05:00
										 |  |  |   sf_close(f); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return true; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2021-12-15 14:15:44 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-02-24 16:16:02 -05:00
										 |  |  | // 16-bit memory is padded to 512, to make things easier for ADPCM-A/B.
 | 
					
						
							| 
									
										
										
										
											2022-02-24 03:57:45 -05:00
										 |  |  | bool DivSample::initInternal(unsigned char d, int count) { | 
					
						
							|  |  |  |   switch (d) { | 
					
						
							|  |  |  |     case 0: // 1-bit
 | 
					
						
							|  |  |  |       if (data1!=NULL) delete[] data1; | 
					
						
							|  |  |  |       length1=(count+7)/8; | 
					
						
							|  |  |  |       data1=new unsigned char[length1]; | 
					
						
							|  |  |  |       memset(data1,0,length1); | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  |     case 1: // DPCM
 | 
					
						
							|  |  |  |       if (dataDPCM!=NULL) delete[] dataDPCM; | 
					
						
							|  |  |  |       lengthDPCM=(count+7)/8; | 
					
						
							|  |  |  |       dataDPCM=new unsigned char[lengthDPCM]; | 
					
						
							|  |  |  |       memset(dataDPCM,0,lengthDPCM); | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  |     case 4: // QSound ADPCM
 | 
					
						
							|  |  |  |       if (dataQSoundA!=NULL) delete[] dataQSoundA; | 
					
						
							|  |  |  |       lengthQSoundA=(count+1)/2; | 
					
						
							|  |  |  |       dataQSoundA=new unsigned char[lengthQSoundA]; | 
					
						
							|  |  |  |       memset(dataQSoundA,0,lengthQSoundA); | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  |     case 5: // ADPCM-A
 | 
					
						
							|  |  |  |       if (dataA!=NULL) delete[] dataA; | 
					
						
							|  |  |  |       lengthA=(count+1)/2; | 
					
						
							| 
									
										
										
										
											2022-02-24 16:16:02 -05:00
										 |  |  |       dataA=new unsigned char[(lengthA+255)&(~0xff)]; | 
					
						
							|  |  |  |       memset(dataA,0,(lengthA+255)&(~0xff)); | 
					
						
							| 
									
										
										
										
											2022-02-24 03:57:45 -05:00
										 |  |  |       break; | 
					
						
							|  |  |  |     case 6: // ADPCM-B
 | 
					
						
							|  |  |  |       if (dataB!=NULL) delete[] dataB; | 
					
						
							|  |  |  |       lengthB=(count+1)/2; | 
					
						
							| 
									
										
										
										
											2022-02-24 16:16:02 -05:00
										 |  |  |       dataB=new unsigned char[(lengthB+255)&(~0xff)]; | 
					
						
							|  |  |  |       memset(dataB,0,(lengthB+255)&(~0xff)); | 
					
						
							| 
									
										
										
										
											2022-02-24 03:57:45 -05:00
										 |  |  |       break; | 
					
						
							|  |  |  |     case 7: // X68000 ADPCM
 | 
					
						
							|  |  |  |       if (dataX68!=NULL) delete[] dataX68; | 
					
						
							|  |  |  |       lengthX68=(count+1)/2; | 
					
						
							|  |  |  |       dataX68=new unsigned char[lengthX68]; | 
					
						
							|  |  |  |       memset(dataX68,0,lengthX68); | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  |     case 8: // 8-bit
 | 
					
						
							|  |  |  |       if (data8!=NULL) delete[] data8; | 
					
						
							|  |  |  |       length8=count; | 
					
						
							| 
									
										
										
										
											2022-03-07 10:15:21 -05:00
										 |  |  |       // for padding X1-010 sample
 | 
					
						
							| 
									
										
										
										
											2022-03-06 12:31:03 -05:00
										 |  |  |       data8=new signed char[(count+4095)&(~0xfff)]; | 
					
						
							|  |  |  |       memset(data8,0,(count+4095)&(~0xfff)); | 
					
						
							| 
									
										
										
										
											2022-02-24 03:57:45 -05:00
										 |  |  |       break; | 
					
						
							|  |  |  |     case 9: // BRR
 | 
					
						
							|  |  |  |       if (dataBRR!=NULL) delete[] dataBRR; | 
					
						
							|  |  |  |       lengthBRR=9*((count+15)/16); | 
					
						
							|  |  |  |       dataBRR=new unsigned char[lengthBRR]; | 
					
						
							|  |  |  |       memset(dataBRR,0,lengthBRR); | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  |     case 10: // VOX
 | 
					
						
							|  |  |  |       if (dataVOX!=NULL) delete[] dataVOX; | 
					
						
							|  |  |  |       lengthVOX=(count+1)/2; | 
					
						
							|  |  |  |       dataVOX=new unsigned char[lengthVOX]; | 
					
						
							|  |  |  |       memset(dataVOX,0,lengthVOX); | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  |     case 16: // 16-bit
 | 
					
						
							|  |  |  |       if (data16!=NULL) delete[] data16; | 
					
						
							|  |  |  |       length16=count*2; | 
					
						
							| 
									
										
										
										
											2022-02-24 16:16:02 -05:00
										 |  |  |       data16=new short[(count+511)&(~0x1ff)]; | 
					
						
							|  |  |  |       memset(data16,0,((count+511)&(~0x1ff))*sizeof(short)); | 
					
						
							| 
									
										
										
										
											2022-02-24 03:57:45 -05:00
										 |  |  |       break; | 
					
						
							|  |  |  |     default: | 
					
						
							|  |  |  |       return false; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool DivSample::init(unsigned int count) { | 
					
						
							|  |  |  |   if (!initInternal(depth,count)) return false; | 
					
						
							|  |  |  |   samples=count; | 
					
						
							|  |  |  |   return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-03-19 04:42:44 -04:00
										 |  |  | bool DivSample::resize(unsigned int count) { | 
					
						
							|  |  |  |   if (depth==8) { | 
					
						
							|  |  |  |     if (data8!=NULL) { | 
					
						
							|  |  |  |       signed char* oldData8=data8; | 
					
						
							|  |  |  |       data8=NULL; | 
					
						
							|  |  |  |       initInternal(8,count); | 
					
						
							|  |  |  |       memcpy(data8,oldData8,MIN(count,samples)); | 
					
						
							|  |  |  |       delete[] oldData8; | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |       initInternal(8,count); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     samples=count; | 
					
						
							|  |  |  |     return true; | 
					
						
							|  |  |  |   } else if (depth==16) { | 
					
						
							|  |  |  |     if (data16!=NULL) { | 
					
						
							|  |  |  |       short* oldData16=data16; | 
					
						
							|  |  |  |       data16=NULL; | 
					
						
							|  |  |  |       initInternal(16,count); | 
					
						
							|  |  |  |       memcpy(data16,oldData16,sizeof(short)*MIN(count,samples)); | 
					
						
							|  |  |  |       delete[] oldData16; | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |       initInternal(16,count); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     samples=count; | 
					
						
							|  |  |  |     return true; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-03-21 03:43:52 -04:00
										 |  |  | bool DivSample::strip(unsigned int begin, unsigned int end) { | 
					
						
							|  |  |  |   if (begin>samples) begin=samples; | 
					
						
							|  |  |  |   if (end>samples) end=samples; | 
					
						
							|  |  |  |   int count=samples-(end-begin); | 
					
						
							|  |  |  |   if (count<=0) return resize(0); | 
					
						
							|  |  |  |   if (depth==8) { | 
					
						
							|  |  |  |     if (data8!=NULL) { | 
					
						
							|  |  |  |       signed char* oldData8=data8; | 
					
						
							|  |  |  |       data8=NULL; | 
					
						
							|  |  |  |       initInternal(8,count); | 
					
						
							|  |  |  |       if (begin>0) { | 
					
						
							|  |  |  |         memcpy(data8,oldData8,begin); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       if (samples-end>0) { | 
					
						
							|  |  |  |         memcpy(data8+begin,oldData8+end,samples-end); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       delete[] oldData8; | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |       // do nothing
 | 
					
						
							|  |  |  |       return true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     samples=count; | 
					
						
							|  |  |  |     return true; | 
					
						
							|  |  |  |   } else if (depth==16) { | 
					
						
							|  |  |  |     if (data16!=NULL) { | 
					
						
							|  |  |  |       short* oldData16=data16; | 
					
						
							|  |  |  |       data16=NULL; | 
					
						
							|  |  |  |       initInternal(16,count); | 
					
						
							|  |  |  |       if (begin>0) { | 
					
						
							|  |  |  |         memcpy(data16,oldData16,sizeof(short)*begin); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       if (samples-end>0) { | 
					
						
							|  |  |  |         memcpy(&(data16[begin]),&(oldData16[end]),sizeof(short)*(samples-end)); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       delete[] oldData16; | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |       // do nothing
 | 
					
						
							|  |  |  |       return true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     samples=count; | 
					
						
							|  |  |  |     return true; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool DivSample::trim(unsigned int begin, unsigned int end) { | 
					
						
							|  |  |  |   int count=end-begin; | 
					
						
							|  |  |  |   if (count==0) return true; | 
					
						
							|  |  |  |   if (begin==0 && end==samples) return true; | 
					
						
							|  |  |  |   if (depth==8) { | 
					
						
							|  |  |  |     if (data8!=NULL) { | 
					
						
							|  |  |  |       signed char* oldData8=data8; | 
					
						
							|  |  |  |       data8=NULL; | 
					
						
							|  |  |  |       initInternal(8,count); | 
					
						
							|  |  |  |       memcpy(data8,oldData8+begin,count); | 
					
						
							|  |  |  |       delete[] oldData8; | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |       // do nothing
 | 
					
						
							|  |  |  |       return true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     samples=count; | 
					
						
							|  |  |  |     return true; | 
					
						
							|  |  |  |   } else if (depth==16) { | 
					
						
							|  |  |  |     if (data16!=NULL) { | 
					
						
							|  |  |  |       short* oldData16=data16; | 
					
						
							|  |  |  |       data16=NULL; | 
					
						
							|  |  |  |       initInternal(16,count); | 
					
						
							|  |  |  |       memcpy(data16,&(oldData16[begin]),sizeof(short)*count); | 
					
						
							|  |  |  |       delete[] oldData16; | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |       // do nothing
 | 
					
						
							|  |  |  |       return true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     samples=count; | 
					
						
							|  |  |  |     return true; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-03-22 05:54:01 -04:00
										 |  |  | bool DivSample::insert(unsigned int pos, unsigned int length) { | 
					
						
							|  |  |  |   unsigned int count=samples+length; | 
					
						
							|  |  |  |   if (depth==8) { | 
					
						
							|  |  |  |     if (data8!=NULL) { | 
					
						
							|  |  |  |       signed char* oldData8=data8; | 
					
						
							|  |  |  |       data8=NULL; | 
					
						
							|  |  |  |       initInternal(8,count); | 
					
						
							|  |  |  |       if (pos>0) { | 
					
						
							|  |  |  |         memcpy(data8,oldData8,pos); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       if (count-pos-length>0) { | 
					
						
							|  |  |  |         memcpy(data8+pos+length,oldData8+pos,count-pos-length); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       delete[] oldData8; | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |       initInternal(8,count); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     samples=count; | 
					
						
							|  |  |  |     return true; | 
					
						
							|  |  |  |   } else if (depth==16) { | 
					
						
							|  |  |  |     if (data16!=NULL) { | 
					
						
							|  |  |  |       short* oldData16=data16; | 
					
						
							|  |  |  |       data16=NULL; | 
					
						
							|  |  |  |       initInternal(16,count); | 
					
						
							| 
									
										
										
										
											2022-03-22 18:01:06 -04:00
										 |  |  |       if (pos>0) { | 
					
						
							|  |  |  |         memcpy(data16,oldData16,sizeof(short)*pos); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       if (count-pos-length>0) { | 
					
						
							|  |  |  |         memcpy(&(data16[pos+length]),&(oldData16[pos]),sizeof(short)*(count-pos-length)); | 
					
						
							|  |  |  |       } | 
					
						
							| 
									
										
										
										
											2022-03-22 05:54:01 -04:00
										 |  |  |       delete[] oldData16; | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |       initInternal(16,count); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     samples=count; | 
					
						
							|  |  |  |     return true; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-03-20 04:14:00 -04:00
										 |  |  | #define RESAMPLE_BEGIN \
 | 
					
						
							|  |  |  |   if (samples<1) return true; \ | 
					
						
							|  |  |  |   int finalCount=(double)samples*(r/(double)rate); \ | 
					
						
							|  |  |  |   signed char* oldData8=data8; \ | 
					
						
							|  |  |  |   short* oldData16=data16; \ | 
					
						
							|  |  |  |   if (depth==16) { \ | 
					
						
							|  |  |  |     if (data16!=NULL) { \ | 
					
						
							|  |  |  |       data16=NULL; \ | 
					
						
							|  |  |  |       initInternal(16,finalCount); \ | 
					
						
							|  |  |  |     } \ | 
					
						
							|  |  |  |   } else if (depth==8) {  \ | 
					
						
							|  |  |  |     if (data8!=NULL) { \ | 
					
						
							|  |  |  |       data8=NULL; \ | 
					
						
							|  |  |  |       initInternal(8,finalCount); \ | 
					
						
							|  |  |  |     } \ | 
					
						
							|  |  |  |   } else { \ | 
					
						
							|  |  |  |     return false; \ | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #define RESAMPLE_END \
 | 
					
						
							| 
									
										
										
										
											2022-03-21 22:47:27 -04:00
										 |  |  |   if (loopStart>=0) loopStart=(double)loopStart*(r/(double)rate); \ | 
					
						
							| 
									
										
										
										
											2022-04-12 03:15:12 -04:00
										 |  |  |   centerRate=(int)((double)centerRate*(r/(double)rate)); \ | 
					
						
							| 
									
										
										
										
											2022-03-22 00:54:01 -04:00
										 |  |  |   rate=r; \ | 
					
						
							|  |  |  |   samples=finalCount; \ | 
					
						
							| 
									
										
										
										
											2022-03-20 04:14:00 -04:00
										 |  |  |   if (depth==16) { \ | 
					
						
							|  |  |  |     delete[] oldData16; \ | 
					
						
							|  |  |  |   } else if (depth==8) { \ | 
					
						
							|  |  |  |     delete[] oldData8; \ | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool DivSample::resampleNone(double r) { | 
					
						
							|  |  |  |   RESAMPLE_BEGIN; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (depth==16) { | 
					
						
							|  |  |  |     for (int i=0; i<finalCount; i++) { | 
					
						
							|  |  |  |       unsigned int pos=(unsigned int)((double)i*((double)rate/r)); | 
					
						
							|  |  |  |       if (pos>=samples) { | 
					
						
							|  |  |  |         data16[i]=0; | 
					
						
							|  |  |  |       } else { | 
					
						
							|  |  |  |         data16[i]=oldData16[pos]; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } else if (depth==8) { | 
					
						
							|  |  |  |     for (int i=0; i<finalCount; i++) { | 
					
						
							|  |  |  |       unsigned int pos=(unsigned int)((double)i*((double)rate/r)); | 
					
						
							|  |  |  |       if (pos>=samples) { | 
					
						
							|  |  |  |         data8[i]=0; | 
					
						
							|  |  |  |       } else { | 
					
						
							|  |  |  |         data8[i]=oldData8[pos]; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   RESAMPLE_END; | 
					
						
							|  |  |  |   return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool DivSample::resampleLinear(double r) { | 
					
						
							|  |  |  |   RESAMPLE_BEGIN; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   double posFrac=0; | 
					
						
							|  |  |  |   unsigned int posInt=0; | 
					
						
							|  |  |  |   double factor=(double)rate/r; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (depth==16) { | 
					
						
							|  |  |  |     for (int i=0; i<finalCount; i++) { | 
					
						
							|  |  |  |       short s1=(posInt>=samples)?0:oldData16[posInt]; | 
					
						
							|  |  |  |       short s2=(posInt+1>=samples)?((loopStart>=0 && loopStart<(int)samples)?oldData16[loopStart]:0):oldData16[posInt+1]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       data16[i]=s1+(float)(s2-s1)*posFrac; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       posFrac+=factor; | 
					
						
							|  |  |  |       while (posFrac>=1.0) { | 
					
						
							|  |  |  |         posFrac-=1.0; | 
					
						
							|  |  |  |         posInt++; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } else if (depth==8) { | 
					
						
							|  |  |  |     for (int i=0; i<finalCount; i++) { | 
					
						
							|  |  |  |       short s1=(posInt>=samples)?0:oldData8[posInt]; | 
					
						
							|  |  |  |       short s2=(posInt+1>=samples)?((loopStart>=0 && loopStart<(int)samples)?oldData8[loopStart]:0):oldData8[posInt+1]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       data8[i]=s1+(float)(s2-s1)*posFrac; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       posFrac+=factor; | 
					
						
							|  |  |  |       while (posFrac>=1.0) { | 
					
						
							|  |  |  |         posFrac-=1.0; | 
					
						
							|  |  |  |         posInt++; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   RESAMPLE_END; | 
					
						
							|  |  |  |   return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool DivSample::resampleCubic(double r) { | 
					
						
							|  |  |  |   RESAMPLE_BEGIN; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   double posFrac=0; | 
					
						
							|  |  |  |   unsigned int posInt=0; | 
					
						
							|  |  |  |   double factor=(double)rate/r; | 
					
						
							|  |  |  |   float* cubicTable=DivFilterTables::getCubicTable(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (depth==16) { | 
					
						
							|  |  |  |     for (int i=0; i<finalCount; i++) { | 
					
						
							|  |  |  |       unsigned int n=((unsigned int)(posFrac*1024.0))&1023; | 
					
						
							|  |  |  |       float* t=&cubicTable[n<<2]; | 
					
						
							|  |  |  |       float s0=(posInt<1)?0:oldData16[posInt-1]; | 
					
						
							|  |  |  |       float s1=(posInt>=samples)?0:oldData16[posInt]; | 
					
						
							|  |  |  |       float s2=(posInt+1>=samples)?((loopStart>=0 && loopStart<(int)samples)?oldData16[loopStart]:0):oldData16[posInt+1]; | 
					
						
							|  |  |  |       float s3=(posInt+2>=samples)?((loopStart>=0 && loopStart<(int)samples)?oldData16[loopStart]:0):oldData16[posInt+2]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       float result=s0*t[0]+s1*t[1]+s2*t[2]+s3*t[3]; | 
					
						
							|  |  |  |       if (result<-32768) result=-32768; | 
					
						
							|  |  |  |       if (result>32767) result=32767; | 
					
						
							|  |  |  |       data16[i]=result; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       posFrac+=factor; | 
					
						
							|  |  |  |       while (posFrac>=1.0) { | 
					
						
							|  |  |  |         posFrac-=1.0; | 
					
						
							|  |  |  |         posInt++; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } else if (depth==8) { | 
					
						
							|  |  |  |     for (int i=0; i<finalCount; i++) { | 
					
						
							|  |  |  |       unsigned int n=((unsigned int)(posFrac*1024.0))&1023; | 
					
						
							|  |  |  |       float* t=&cubicTable[n<<2]; | 
					
						
							|  |  |  |       float s0=(posInt<1)?0:oldData8[posInt-1]; | 
					
						
							|  |  |  |       float s1=(posInt>=samples)?0:oldData8[posInt]; | 
					
						
							|  |  |  |       float s2=(posInt+1>=samples)?((loopStart>=0 && loopStart<(int)samples)?oldData8[loopStart]:0):oldData8[posInt+1]; | 
					
						
							|  |  |  |       float s3=(posInt+2>=samples)?((loopStart>=0 && loopStart<(int)samples)?oldData8[loopStart]:0):oldData8[posInt+2]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       float result=s0*t[0]+s1*t[1]+s2*t[2]+s3*t[3]; | 
					
						
							|  |  |  |       if (result<-128) result=-128; | 
					
						
							|  |  |  |       if (result>127) result=127; | 
					
						
							|  |  |  |       data8[i]=result; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       posFrac+=factor; | 
					
						
							|  |  |  |       while (posFrac>=1.0) { | 
					
						
							|  |  |  |         posFrac-=1.0; | 
					
						
							|  |  |  |         posInt++; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   RESAMPLE_END; | 
					
						
							|  |  |  |   return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool DivSample::resampleBlep(double r) { | 
					
						
							|  |  |  |   RESAMPLE_BEGIN; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   double posFrac=0; | 
					
						
							|  |  |  |   unsigned int posInt=0; | 
					
						
							|  |  |  |   double factor=r/(double)rate; | 
					
						
							|  |  |  |   float* sincITable=DivFilterTables::getSincIntegralTable(); | 
					
						
							|  |  |  |   float s[16]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   memset(s,0,16*sizeof(float)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (depth==16) { | 
					
						
							|  |  |  |     memset(data16,0,finalCount*sizeof(short)); | 
					
						
							|  |  |  |     for (int i=0; i<finalCount; i++) { | 
					
						
							|  |  |  |       if (posInt<samples) { | 
					
						
							|  |  |  |         int result=data16[i]+oldData16[posInt]; | 
					
						
							|  |  |  |         if (result<-32768) result=-32768; | 
					
						
							|  |  |  |         if (result>32767) result=32767; | 
					
						
							|  |  |  |         data16[i]=result; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |        | 
					
						
							|  |  |  |       posFrac+=1.0; | 
					
						
							|  |  |  |       while (posFrac>=1.0) { | 
					
						
							|  |  |  |         unsigned int n=((unsigned int)(posFrac*8192.0))&8191; | 
					
						
							|  |  |  |         posFrac-=factor; | 
					
						
							|  |  |  |         posInt++; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         float* t1=&sincITable[(8191-n)<<3]; | 
					
						
							|  |  |  |         float* t2=&sincITable[n<<3]; | 
					
						
							|  |  |  |         float delta=oldData16[posInt]-oldData16[posInt-1]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         for (int j=0; j<8; j++) { | 
					
						
							|  |  |  |           if (i-j>0) { | 
					
						
							|  |  |  |             float result=data16[i-j]+t1[j]*-delta; | 
					
						
							|  |  |  |             if (result<-32768) result=-32768; | 
					
						
							|  |  |  |             if (result>32767) result=32767; | 
					
						
							|  |  |  |             data16[i-j]=result; | 
					
						
							|  |  |  |           } | 
					
						
							|  |  |  |           if (i+j+1<finalCount) { | 
					
						
							|  |  |  |             float result=data16[i+j+1]+t2[j]*delta; | 
					
						
							|  |  |  |             if (result<-32768) result=-32768; | 
					
						
							|  |  |  |             if (result>32767) result=32767; | 
					
						
							|  |  |  |             data16[i+j+1]=result; | 
					
						
							|  |  |  |           } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } else if (depth==8) { | 
					
						
							| 
									
										
										
										
											2022-03-22 00:48:18 -04:00
										 |  |  |     memset(data8,0,finalCount); | 
					
						
							| 
									
										
										
										
											2022-03-20 04:14:00 -04:00
										 |  |  |     for (int i=0; i<finalCount; i++) { | 
					
						
							|  |  |  |       if (posInt<samples) { | 
					
						
							|  |  |  |         int result=data8[i]+oldData8[posInt]; | 
					
						
							|  |  |  |         if (result<-128) result=-128; | 
					
						
							|  |  |  |         if (result>127) result=127; | 
					
						
							|  |  |  |         data8[i]=result; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |        | 
					
						
							|  |  |  |       posFrac+=1.0; | 
					
						
							|  |  |  |       while (posFrac>=1.0) { | 
					
						
							|  |  |  |         unsigned int n=((unsigned int)(posFrac*8192.0))&8191; | 
					
						
							|  |  |  |         posFrac-=factor; | 
					
						
							|  |  |  |         posInt++; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         float* t1=&sincITable[(8191-n)<<3]; | 
					
						
							|  |  |  |         float* t2=&sincITable[n<<3]; | 
					
						
							|  |  |  |         float delta=oldData8[posInt]-oldData8[posInt-1]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         for (int j=0; j<8; j++) { | 
					
						
							|  |  |  |           if (i-j>0) { | 
					
						
							|  |  |  |             float result=data8[i-j]+t1[j]*-delta; | 
					
						
							|  |  |  |             if (result<-128) result=-128; | 
					
						
							|  |  |  |             if (result>127) result=127; | 
					
						
							|  |  |  |             data8[i-j]=result; | 
					
						
							|  |  |  |           } | 
					
						
							|  |  |  |           if (i+j+1<finalCount) { | 
					
						
							|  |  |  |             float result=data8[i+j+1]+t2[j]*delta; | 
					
						
							|  |  |  |             if (result<-128) result=-128; | 
					
						
							|  |  |  |             if (result>127) result=127; | 
					
						
							|  |  |  |             data8[i+j+1]=result; | 
					
						
							|  |  |  |           } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   RESAMPLE_END; | 
					
						
							|  |  |  |   return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool DivSample::resampleSinc(double r) { | 
					
						
							|  |  |  |   RESAMPLE_BEGIN; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   double posFrac=0; | 
					
						
							|  |  |  |   unsigned int posInt=0; | 
					
						
							|  |  |  |   double factor=(double)rate/r; | 
					
						
							|  |  |  |   float* sincTable=DivFilterTables::getSincTable(); | 
					
						
							|  |  |  |   float s[16]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   memset(s,0,16*sizeof(float)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (depth==16) { | 
					
						
							|  |  |  |     for (int i=0; i<finalCount+8; i++) { | 
					
						
							|  |  |  |       unsigned int n=((unsigned int)(posFrac*8192.0))&8191; | 
					
						
							|  |  |  |       float result=0; | 
					
						
							|  |  |  |       float* t1=&sincTable[(8191-n)<<3]; | 
					
						
							|  |  |  |       float* t2=&sincTable[n<<3]; | 
					
						
							|  |  |  |       for (int j=0; j<8; j++) { | 
					
						
							|  |  |  |         result+=s[j]*t2[7-j]; | 
					
						
							|  |  |  |         result+=s[8+j]*t1[j]; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       if (result<-32768) result=-32768; | 
					
						
							|  |  |  |       if (result>32767) result=32767; | 
					
						
							|  |  |  |       if (i>=8) { | 
					
						
							|  |  |  |         data16[i-8]=result; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |        | 
					
						
							|  |  |  |       posFrac+=factor; | 
					
						
							|  |  |  |       while (posFrac>=1.0) { | 
					
						
							|  |  |  |         posFrac-=1.0; | 
					
						
							|  |  |  |         posInt++; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         for (int j=0; j<15; j++) s[j]=s[j+1]; | 
					
						
							|  |  |  |         s[15]=(posInt>=samples)?0:oldData16[posInt]; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } else if (depth==8) { | 
					
						
							|  |  |  |     for (int i=0; i<finalCount+8; i++) { | 
					
						
							|  |  |  |       unsigned int n=((unsigned int)(posFrac*8192.0))&8191; | 
					
						
							|  |  |  |       float result=0; | 
					
						
							|  |  |  |       float* t1=&sincTable[(8191-n)<<3]; | 
					
						
							|  |  |  |       float* t2=&sincTable[n<<3]; | 
					
						
							|  |  |  |       for (int j=0; j<8; j++) { | 
					
						
							|  |  |  |         result+=s[j]*t2[7-j]; | 
					
						
							|  |  |  |         result+=s[8+j]*t1[j]; | 
					
						
							|  |  |  |       } | 
					
						
							| 
									
										
										
										
											2022-04-15 15:38:13 -04:00
										 |  |  |       if (result<-128) result=-128; | 
					
						
							|  |  |  |       if (result>127) result=127; | 
					
						
							| 
									
										
										
										
											2022-03-20 04:14:00 -04:00
										 |  |  |       if (i>=8) { | 
					
						
							|  |  |  |         data8[i-8]=result; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |        | 
					
						
							|  |  |  |       posFrac+=factor; | 
					
						
							|  |  |  |       while (posFrac>=1.0) { | 
					
						
							|  |  |  |         posFrac-=1.0; | 
					
						
							|  |  |  |         posInt++; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         for (int j=0; j<15; j++) s[j]=s[j+1]; | 
					
						
							|  |  |  |         s[15]=(posInt>=samples)?0:oldData8[posInt]; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   RESAMPLE_END; | 
					
						
							|  |  |  |   return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool DivSample::resample(double r, int filter) { | 
					
						
							|  |  |  |   if (depth!=8 && depth!=16) return false; | 
					
						
							|  |  |  |   switch (filter) { | 
					
						
							|  |  |  |     case DIV_RESAMPLE_NONE: | 
					
						
							|  |  |  |       return resampleNone(r); | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  |     case DIV_RESAMPLE_LINEAR: | 
					
						
							|  |  |  |       return resampleLinear(r); | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  |     case DIV_RESAMPLE_CUBIC: | 
					
						
							|  |  |  |       return resampleCubic(r); | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  |     case DIV_RESAMPLE_BLEP: | 
					
						
							|  |  |  |       return resampleBlep(r); | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  |     case DIV_RESAMPLE_SINC: | 
					
						
							|  |  |  |       return resampleSinc(r); | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  |     case DIV_RESAMPLE_BEST: | 
					
						
							|  |  |  |       if (r>rate) { | 
					
						
							|  |  |  |         return resampleSinc(r); | 
					
						
							|  |  |  |       } else { | 
					
						
							|  |  |  |         return resampleBlep(r); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-02-24 03:57:45 -05:00
										 |  |  | void DivSample::render() { | 
					
						
							|  |  |  |   // step 1: convert to 16-bit if needed
 | 
					
						
							|  |  |  |   if (depth!=16) { | 
					
						
							|  |  |  |     if (!initInternal(16,samples)) return; | 
					
						
							|  |  |  |     switch (depth) { | 
					
						
							|  |  |  |       case 0: // 1-bit
 | 
					
						
							|  |  |  |         for (unsigned int i=0; i<samples; i++) { | 
					
						
							|  |  |  |           data16[i]=((data1[i>>3]>>(i&7))&1)?0x7fff:-0x7fff; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |       case 1: { // DPCM
 | 
					
						
							|  |  |  |         int accum=0; | 
					
						
							|  |  |  |         for (unsigned int i=0; i<samples; i++) { | 
					
						
							| 
									
										
										
										
											2022-03-17 19:15:44 -04:00
										 |  |  |           accum+=((dataDPCM[i>>3]>>(i&7))&1)?1:-1; | 
					
						
							| 
									
										
										
										
											2022-02-24 03:57:45 -05:00
										 |  |  |           if (accum>63) accum=63; | 
					
						
							|  |  |  |           if (accum<-64) accum=-64; | 
					
						
							|  |  |  |           data16[i]=accum*512; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       case 4: // QSound ADPCM
 | 
					
						
							|  |  |  |         bs_decode(dataQSoundA,data16,samples); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |       case 5: // ADPCM-A
 | 
					
						
							|  |  |  |         yma_decode(dataA,data16,samples); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |       case 6: // ADPCM-B
 | 
					
						
							|  |  |  |         ymb_decode(dataB,data16,samples); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |       case 7: // X6800 ADPCM
 | 
					
						
							|  |  |  |         oki6258_decode(dataX68,data16,samples); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |       case 8: // 8-bit PCM
 | 
					
						
							|  |  |  |         for (unsigned int i=0; i<samples; i++) { | 
					
						
							|  |  |  |           data16[i]=data8[i]<<8; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |       case 9: // BRR
 | 
					
						
							|  |  |  |         // TODO!
 | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |       case 10: // VOX
 | 
					
						
							|  |  |  |         oki_decode(dataVOX,data16,samples); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |       default: | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   // step 2: render to other formats
 | 
					
						
							|  |  |  |   if (depth!=0) { // 1-bit
 | 
					
						
							|  |  |  |     if (!initInternal(0,samples)) return; | 
					
						
							|  |  |  |     for (unsigned int i=0; i<samples; i++) { | 
					
						
							|  |  |  |       if (data16[i]>0) { | 
					
						
							|  |  |  |         data1[i>>3]|=1<<(i&7); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   if (depth!=1) { // DPCM
 | 
					
						
							|  |  |  |     if (!initInternal(1,samples)) return; | 
					
						
							|  |  |  |     int accum=63; | 
					
						
							|  |  |  |     for (unsigned int i=0; i<samples; i++) { | 
					
						
							|  |  |  |       int next=((unsigned short)(data16[i]^0x8000))>>9; | 
					
						
							|  |  |  |       if (next>accum) { | 
					
						
							|  |  |  |         dataDPCM[i>>3]|=1<<(i&7); | 
					
						
							|  |  |  |         accum++; | 
					
						
							|  |  |  |       } else { | 
					
						
							|  |  |  |         accum--; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       if (accum<0) accum=0; | 
					
						
							|  |  |  |       if (accum>127) accum=127; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   if (depth!=4) { // QSound ADPCM
 | 
					
						
							|  |  |  |     if (!initInternal(4,samples)) return; | 
					
						
							|  |  |  |     bs_encode(data16,dataQSoundA,samples); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2022-02-24 16:16:02 -05:00
										 |  |  |   // TODO: pad to 256.
 | 
					
						
							| 
									
										
										
										
											2022-02-24 03:57:45 -05:00
										 |  |  |   if (depth!=5) { // ADPCM-A
 | 
					
						
							|  |  |  |     if (!initInternal(5,samples)) return; | 
					
						
							| 
									
										
										
										
											2022-02-24 16:16:02 -05:00
										 |  |  |     yma_encode(data16,dataA,(samples+511)&(~0x1ff)); | 
					
						
							| 
									
										
										
										
											2022-02-24 03:57:45 -05:00
										 |  |  |   } | 
					
						
							|  |  |  |   if (depth!=6) { // ADPCM-B
 | 
					
						
							|  |  |  |     if (!initInternal(6,samples)) return; | 
					
						
							| 
									
										
										
										
											2022-02-24 16:16:02 -05:00
										 |  |  |     ymb_encode(data16,dataB,(samples+511)&(~0x1ff)); | 
					
						
							| 
									
										
										
										
											2022-02-24 03:57:45 -05:00
										 |  |  |   } | 
					
						
							|  |  |  |   if (depth!=7) { // X68000 ADPCM
 | 
					
						
							|  |  |  |     if (!initInternal(7,samples)) return; | 
					
						
							|  |  |  |     oki6258_encode(data16,dataX68,samples); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   if (depth!=8) { // 8-bit PCM
 | 
					
						
							|  |  |  |     if (!initInternal(8,samples)) return; | 
					
						
							|  |  |  |     for (unsigned int i=0; i<samples; i++) { | 
					
						
							|  |  |  |       data8[i]=data16[i]>>8; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   // TODO: BRR!
 | 
					
						
							|  |  |  |   if (depth!=10) { // VOX
 | 
					
						
							|  |  |  |     if (!initInternal(10,samples)) return; | 
					
						
							|  |  |  |     oki_encode(data16,dataVOX,samples); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void* DivSample::getCurBuf() { | 
					
						
							|  |  |  |   switch (depth) { | 
					
						
							|  |  |  |     case 0: | 
					
						
							|  |  |  |       return data1; | 
					
						
							|  |  |  |     case 1: | 
					
						
							|  |  |  |       return dataDPCM; | 
					
						
							|  |  |  |     case 4: | 
					
						
							|  |  |  |       return dataQSoundA; | 
					
						
							|  |  |  |     case 5: | 
					
						
							|  |  |  |       return dataA; | 
					
						
							|  |  |  |     case 6: | 
					
						
							|  |  |  |       return dataB; | 
					
						
							|  |  |  |     case 7: | 
					
						
							|  |  |  |       return dataX68; | 
					
						
							|  |  |  |     case 8: | 
					
						
							|  |  |  |       return data8; | 
					
						
							|  |  |  |     case 9: | 
					
						
							|  |  |  |       return dataBRR; | 
					
						
							|  |  |  |     case 10: | 
					
						
							|  |  |  |       return dataVOX; | 
					
						
							|  |  |  |     case 16: | 
					
						
							|  |  |  |       return data16; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   return NULL; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | unsigned int DivSample::getCurBufLen() { | 
					
						
							|  |  |  |   switch (depth) { | 
					
						
							|  |  |  |     case 0: | 
					
						
							|  |  |  |       return length1; | 
					
						
							|  |  |  |     case 1: | 
					
						
							|  |  |  |       return lengthDPCM; | 
					
						
							|  |  |  |     case 4: | 
					
						
							|  |  |  |       return lengthQSoundA; | 
					
						
							|  |  |  |     case 5: | 
					
						
							|  |  |  |       return lengthA; | 
					
						
							|  |  |  |     case 6: | 
					
						
							|  |  |  |       return lengthB; | 
					
						
							|  |  |  |     case 7: | 
					
						
							|  |  |  |       return lengthX68; | 
					
						
							|  |  |  |     case 8: | 
					
						
							|  |  |  |       return length8; | 
					
						
							|  |  |  |     case 9: | 
					
						
							|  |  |  |       return lengthBRR; | 
					
						
							|  |  |  |     case 10: | 
					
						
							|  |  |  |       return lengthVOX; | 
					
						
							|  |  |  |     case 16: | 
					
						
							|  |  |  |       return length16; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-03-23 01:42:59 -04:00
										 |  |  | DivSampleHistory* DivSample::prepareUndo(bool data, bool doNotPush) { | 
					
						
							| 
									
										
										
										
											2022-03-22 19:36:29 -04:00
										 |  |  |   DivSampleHistory* h; | 
					
						
							|  |  |  |   if (data) { | 
					
						
							|  |  |  |     unsigned char* duplicate; | 
					
						
							|  |  |  |     if (getCurBuf()==NULL) { | 
					
						
							|  |  |  |       duplicate=NULL; | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |       duplicate=new unsigned char[getCurBufLen()]; | 
					
						
							|  |  |  |       memcpy(duplicate,getCurBuf(),getCurBufLen()); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     h=new DivSampleHistory(duplicate,getCurBufLen(),samples,depth,rate,centerRate,loopStart); | 
					
						
							|  |  |  |   } else { | 
					
						
							|  |  |  |     h=new DivSampleHistory(depth,rate,centerRate,loopStart); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2022-03-23 01:42:59 -04:00
										 |  |  |   if (!doNotPush) { | 
					
						
							|  |  |  |     while (!redoHist.empty()) { | 
					
						
							|  |  |  |       DivSampleHistory* h=redoHist.back(); | 
					
						
							|  |  |  |       delete h; | 
					
						
							|  |  |  |       redoHist.pop_back(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (undoHist.size()>100) undoHist.pop_front(); | 
					
						
							|  |  |  |     undoHist.push_back(h); | 
					
						
							| 
									
										
										
										
											2022-03-22 19:36:29 -04:00
										 |  |  |   } | 
					
						
							|  |  |  |   return h; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-03-23 01:42:59 -04:00
										 |  |  | #define applyHistory \
 | 
					
						
							|  |  |  |   depth=h->depth; \ | 
					
						
							|  |  |  |   if (h->hasSample) { \ | 
					
						
							|  |  |  |     initInternal(h->depth,h->samples); \ | 
					
						
							|  |  |  |     samples=h->samples; \ | 
					
						
							|  |  |  | \ | 
					
						
							| 
									
										
										
										
											2022-04-10 23:12:02 -04:00
										 |  |  |     if (h->length!=getCurBufLen()) logW("undo buffer length not equal to current buffer length! %d != %d",h->length,getCurBufLen()); \ | 
					
						
							| 
									
										
										
										
											2022-03-23 01:42:59 -04:00
										 |  |  | \ | 
					
						
							|  |  |  |     void* buf=getCurBuf(); \ | 
					
						
							|  |  |  | \ | 
					
						
							|  |  |  |     if (buf!=NULL && h->data!=NULL) { \ | 
					
						
							|  |  |  |       memcpy(buf,h->data,h->length); \ | 
					
						
							|  |  |  |     } \ | 
					
						
							|  |  |  |   } \ | 
					
						
							|  |  |  |   rate=h->rate; \ | 
					
						
							|  |  |  |   centerRate=h->centerRate; \ | 
					
						
							|  |  |  |   loopStart=h->loopStart; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-03-22 19:36:29 -04:00
										 |  |  | int DivSample::undo() { | 
					
						
							| 
									
										
										
										
											2022-03-23 01:42:59 -04:00
										 |  |  |   if (undoHist.empty()) return 0; | 
					
						
							|  |  |  |   DivSampleHistory* h=undoHist.back(); | 
					
						
							|  |  |  |   DivSampleHistory* redo=prepareUndo(h->hasSample,true); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   int ret=h->hasSample?2:1; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   applyHistory; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   redoHist.push_back(redo); | 
					
						
							|  |  |  |   delete h; | 
					
						
							|  |  |  |   undoHist.pop_back(); | 
					
						
							|  |  |  |   return ret; | 
					
						
							| 
									
										
										
										
											2022-03-22 19:36:29 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int DivSample::redo() { | 
					
						
							| 
									
										
										
										
											2022-03-23 01:42:59 -04:00
										 |  |  |   if (redoHist.empty()) return 0; | 
					
						
							|  |  |  |   DivSampleHistory* h=redoHist.back(); | 
					
						
							|  |  |  |   DivSampleHistory* undo=prepareUndo(h->hasSample,true); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   int ret=h->hasSample?2:1; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   applyHistory; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   undoHist.push_back(undo); | 
					
						
							|  |  |  |   delete h; | 
					
						
							|  |  |  |   redoHist.pop_back(); | 
					
						
							|  |  |  |   return ret; | 
					
						
							| 
									
										
										
										
											2022-03-22 19:36:29 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-12-15 14:15:44 -05:00
										 |  |  | DivSample::~DivSample() { | 
					
						
							| 
									
										
										
										
											2022-03-22 19:36:29 -04:00
										 |  |  |   while (!undoHist.empty()) { | 
					
						
							| 
									
										
										
										
											2022-03-23 01:42:59 -04:00
										 |  |  |     DivSampleHistory* h=undoHist.back(); | 
					
						
							| 
									
										
										
										
											2022-03-22 19:36:29 -04:00
										 |  |  |     delete h; | 
					
						
							| 
									
										
										
										
											2022-03-23 01:42:59 -04:00
										 |  |  |     undoHist.pop_back(); | 
					
						
							| 
									
										
										
										
											2022-03-22 19:36:29 -04:00
										 |  |  |   } | 
					
						
							|  |  |  |   while (!redoHist.empty()) { | 
					
						
							| 
									
										
										
										
											2022-03-23 01:42:59 -04:00
										 |  |  |     DivSampleHistory* h=redoHist.back(); | 
					
						
							| 
									
										
										
										
											2022-03-22 19:36:29 -04:00
										 |  |  |     delete h; | 
					
						
							| 
									
										
										
										
											2022-03-23 01:42:59 -04:00
										 |  |  |     redoHist.pop_back(); | 
					
						
							| 
									
										
										
										
											2022-03-22 19:36:29 -04:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2022-02-24 03:57:45 -05:00
										 |  |  |   if (data8) delete[] data8; | 
					
						
							|  |  |  |   if (data16) delete[] data16; | 
					
						
							|  |  |  |   if (data1) delete[] data1; | 
					
						
							|  |  |  |   if (dataDPCM) delete[] dataDPCM; | 
					
						
							|  |  |  |   if (dataQSoundA) delete[] dataQSoundA; | 
					
						
							|  |  |  |   if (dataA) delete[] dataA; | 
					
						
							|  |  |  |   if (dataB) delete[] dataB; | 
					
						
							|  |  |  |   if (dataX68) delete[] dataX68; | 
					
						
							|  |  |  |   if (dataBRR) delete[] dataBRR; | 
					
						
							|  |  |  |   if (dataVOX) delete[] dataVOX; | 
					
						
							| 
									
										
										
										
											2021-12-15 14:15:44 -05:00
										 |  |  | } |