2022-02-15 03:12:20 +00: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.
|
|
|
|
*/
|
|
|
|
|
2022-04-20 16:52:37 +00:00
|
|
|
#ifndef _SAMPLE_H
|
|
|
|
#define _SAMPLE_H
|
|
|
|
|
|
|
|
#pragma once
|
|
|
|
|
2021-12-15 19:15:44 +00:00
|
|
|
#include "../ta-utils.h"
|
2022-03-22 23:36:29 +00:00
|
|
|
#include <deque>
|
2021-12-15 19:15:44 +00:00
|
|
|
|
2022-03-20 08:14:00 +00:00
|
|
|
enum DivResampleFilters {
|
|
|
|
DIV_RESAMPLE_NONE=0,
|
|
|
|
DIV_RESAMPLE_LINEAR,
|
|
|
|
DIV_RESAMPLE_CUBIC,
|
|
|
|
DIV_RESAMPLE_BLEP,
|
|
|
|
DIV_RESAMPLE_SINC,
|
2022-04-20 16:52:37 +00:00
|
|
|
DIV_RESAMPLE_BEST,
|
|
|
|
DIV_RESAMPLE_MAX // for identify boundary
|
|
|
|
};
|
|
|
|
|
|
|
|
enum DivSampleDepth: unsigned char {
|
|
|
|
DIV_SAMPLE_DEPTH_1BIT=0,
|
|
|
|
DIV_SAMPLE_DEPTH_1BIT_DPCM=1,
|
|
|
|
DIV_SAMPLE_DEPTH_QSOUND_ADPCM=4,
|
|
|
|
DIV_SAMPLE_DEPTH_ADPCM_A=5,
|
|
|
|
DIV_SAMPLE_DEPTH_ADPCM_B=6,
|
|
|
|
DIV_SAMPLE_DEPTH_X68K_ADPCM=7,
|
|
|
|
DIV_SAMPLE_DEPTH_8BIT=8,
|
|
|
|
DIV_SAMPLE_DEPTH_BRR=9,
|
|
|
|
DIV_SAMPLE_DEPTH_VOX=10,
|
|
|
|
DIV_SAMPLE_DEPTH_16BIT=16,
|
|
|
|
DIV_SAMPLE_DEPTH_MAX // for identify boundary
|
|
|
|
};
|
|
|
|
|
|
|
|
enum DivSampleLoopMode: unsigned char {
|
|
|
|
DIV_SAMPLE_LOOPMODE_ONESHOT=0,
|
2022-04-27 05:29:53 +00:00
|
|
|
DIV_SAMPLE_LOOPMODE_FORWARD,
|
2022-04-20 16:52:37 +00:00
|
|
|
DIV_SAMPLE_LOOPMODE_BACKWARD,
|
|
|
|
DIV_SAMPLE_LOOPMODE_PINGPONG,
|
|
|
|
DIV_SAMPLE_LOOPMODE_MAX // for identify boundary
|
2022-03-20 08:14:00 +00:00
|
|
|
};
|
|
|
|
|
2022-03-22 23:36:29 +00:00
|
|
|
struct DivSampleHistory {
|
|
|
|
unsigned char* data;
|
|
|
|
unsigned int length, samples;
|
2022-04-20 16:52:37 +00:00
|
|
|
DivSampleDepth depth;
|
2022-03-22 23:36:29 +00:00
|
|
|
int rate, centerRate, loopStart;
|
2022-04-20 16:52:37 +00:00
|
|
|
unsigned int loopEnd;
|
|
|
|
DivSampleLoopMode loopMode;
|
2022-03-22 23:36:29 +00:00
|
|
|
bool hasSample;
|
2022-04-20 16:52:37 +00:00
|
|
|
DivSampleHistory(void* d, unsigned int l, unsigned int s, DivSampleDepth de, int r, int cr, int ls, unsigned int le, DivSampleLoopMode lm):
|
2022-03-22 23:36:29 +00:00
|
|
|
data((unsigned char*)d),
|
|
|
|
length(l),
|
|
|
|
samples(s),
|
|
|
|
depth(de),
|
|
|
|
rate(r),
|
|
|
|
centerRate(cr),
|
|
|
|
loopStart(ls),
|
2022-04-20 16:52:37 +00:00
|
|
|
loopEnd(le),
|
|
|
|
loopMode(lm),
|
2022-03-22 23:36:29 +00:00
|
|
|
hasSample(true) {}
|
2022-04-20 16:52:37 +00:00
|
|
|
DivSampleHistory(DivSampleDepth de, int r, int cr, int ls, unsigned int le, DivSampleLoopMode lm):
|
2022-03-22 23:36:29 +00:00
|
|
|
data(NULL),
|
|
|
|
length(0),
|
|
|
|
samples(0),
|
|
|
|
depth(de),
|
|
|
|
rate(r),
|
|
|
|
centerRate(cr),
|
|
|
|
loopStart(ls),
|
2022-04-20 16:52:37 +00:00
|
|
|
loopEnd(le),
|
|
|
|
loopMode(lm),
|
2022-03-22 23:36:29 +00:00
|
|
|
hasSample(false) {}
|
|
|
|
~DivSampleHistory();
|
|
|
|
};
|
|
|
|
|
2021-05-11 20:08:08 +00:00
|
|
|
struct DivSample {
|
|
|
|
String name;
|
2022-02-24 08:57:45 +00:00
|
|
|
int rate, centerRate, loopStart, loopOffP;
|
2022-04-20 16:52:37 +00:00
|
|
|
unsigned int loopEnd;
|
|
|
|
// valid values are:
|
|
|
|
// - 0: One Shot (Loop disable)
|
|
|
|
// - 1: Foward loop
|
|
|
|
// - 2: Backward loop
|
|
|
|
// - 3: Pingpong loop
|
|
|
|
DivSampleLoopMode loopMode;
|
2022-02-03 04:17:16 +00:00
|
|
|
// valid values are:
|
2022-02-24 03:00:12 +00:00
|
|
|
// - 0: ZX Spectrum overlay drum (1-bit)
|
|
|
|
// - 1: 1-bit NES DPCM (1-bit)
|
2022-02-24 08:57:45 +00:00
|
|
|
// - 4: QSound ADPCM
|
2022-02-24 03:00:12 +00:00
|
|
|
// - 5: ADPCM-A
|
|
|
|
// - 6: ADPCM-B
|
|
|
|
// - 7: X68000 ADPCM
|
2022-02-03 04:17:16 +00:00
|
|
|
// - 8: 8-bit PCM
|
2022-02-24 03:00:12 +00:00
|
|
|
// - 9: BRR (SNES)
|
|
|
|
// - 10: VOX
|
2022-02-03 04:17:16 +00:00
|
|
|
// - 16: 16-bit PCM
|
2022-04-20 16:52:37 +00:00
|
|
|
DivSampleDepth depth;
|
2022-02-24 03:00:12 +00:00
|
|
|
|
|
|
|
// these are the new data structures.
|
2022-02-24 08:57:45 +00:00
|
|
|
signed char* data8; // 8
|
|
|
|
short* data16; // 16
|
|
|
|
unsigned char* data1; // 0
|
|
|
|
unsigned char* dataDPCM; // 1
|
|
|
|
unsigned char* dataQSoundA; // 4
|
|
|
|
unsigned char* dataA; // 5
|
|
|
|
unsigned char* dataB; // 6
|
|
|
|
unsigned char* dataX68; // 7
|
|
|
|
unsigned char* dataBRR; // 9
|
|
|
|
unsigned char* dataVOX; // 10
|
2022-02-24 03:00:12 +00:00
|
|
|
|
2022-02-24 08:57:45 +00:00
|
|
|
unsigned int length8, length16, length1, lengthDPCM, lengthQSoundA, lengthA, lengthB, lengthX68, lengthBRR, lengthVOX;
|
|
|
|
unsigned int off8, off16, off1, offDPCM, offQSoundA, offA, offB, offX68, offBRR, offVOX;
|
2022-04-20 16:52:37 +00:00
|
|
|
unsigned int offSegaPCM, offQSound, offX1_010, offES5506;
|
2022-02-24 03:00:12 +00:00
|
|
|
|
2022-02-24 08:57:45 +00:00
|
|
|
unsigned int samples;
|
2021-05-13 07:39:26 +00:00
|
|
|
|
2022-03-22 23:36:29 +00:00
|
|
|
std::deque<DivSampleHistory*> undoHist;
|
|
|
|
std::deque<DivSampleHistory*> redoHist;
|
|
|
|
|
2022-03-20 08:14:00 +00:00
|
|
|
/**
|
|
|
|
* @warning DO NOT USE - internal functions
|
|
|
|
*/
|
|
|
|
bool resampleNone(double rate);
|
|
|
|
bool resampleLinear(double rate);
|
|
|
|
bool resampleCubic(double rate);
|
|
|
|
bool resampleBlep(double rate);
|
|
|
|
bool resampleSinc(double rate);
|
|
|
|
|
2022-04-20 16:52:37 +00:00
|
|
|
/**
|
|
|
|
* check if sample is loopable.
|
|
|
|
* @return whether it was loopable.
|
|
|
|
*/
|
|
|
|
bool isLoopable();
|
|
|
|
|
2022-03-16 03:05:55 +00:00
|
|
|
/**
|
|
|
|
* save this sample to a file.
|
|
|
|
* @param path a path.
|
|
|
|
* @return whether saving succeeded or not.
|
|
|
|
*/
|
2021-12-18 06:03:59 +00:00
|
|
|
bool save(const char* path);
|
2022-03-16 03:05:55 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @warning DO NOT USE - internal function
|
|
|
|
* initialize sample data.
|
|
|
|
* @param d sample type.
|
|
|
|
* @param count number of samples.
|
|
|
|
* @return whether it was successful.
|
|
|
|
*/
|
2022-04-20 16:52:37 +00:00
|
|
|
bool initInternal(DivSampleDepth d, int count);
|
2022-03-16 03:05:55 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* initialize sample data. make sure you have set `depth` before doing so.
|
|
|
|
* @param count number of samples.
|
|
|
|
* @return whether it was successful.
|
|
|
|
*/
|
2022-02-24 08:57:45 +00:00
|
|
|
bool init(unsigned int count);
|
2022-03-16 03:05:55 +00:00
|
|
|
|
2022-03-19 08:42:44 +00:00
|
|
|
/**
|
|
|
|
* resize sample data. make sure the sample has been initialized before doing so.
|
|
|
|
* @warning do not attempt to resize a sample outside of a synchronized block!
|
|
|
|
* @param count number of samples.
|
|
|
|
* @return whether it was successful.
|
|
|
|
*/
|
|
|
|
bool resize(unsigned int count);
|
|
|
|
|
2022-03-21 07:43:52 +00:00
|
|
|
/**
|
|
|
|
* remove part of the sample data.
|
|
|
|
* @warning do not attempt to strip a sample outside of a synchronized block!
|
|
|
|
* @param start the beginning.
|
|
|
|
* @param end the end.
|
|
|
|
* @return whether it was successful.
|
|
|
|
*/
|
|
|
|
bool strip(unsigned int begin, unsigned int end);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* clip the sample data to specified boundaries.
|
|
|
|
* @warning do not attempt to trim a sample outside of a synchronized block!
|
|
|
|
* @param start the beginning.
|
|
|
|
* @param end the end.
|
|
|
|
* @return whether it was successful.
|
|
|
|
*/
|
|
|
|
bool trim(unsigned int begin, unsigned int end);
|
|
|
|
|
2022-03-22 09:54:01 +00:00
|
|
|
/**
|
|
|
|
* insert silence at specified position.
|
|
|
|
* @warning do not attempt to do this outside of a synchronized block!
|
|
|
|
* @param pos the beginning.
|
|
|
|
* @param length how many samples to insert.
|
|
|
|
* @return whether it was successful.
|
|
|
|
*/
|
|
|
|
bool insert(unsigned int pos, unsigned int length);
|
|
|
|
|
2022-03-20 08:14:00 +00:00
|
|
|
/**
|
|
|
|
* change the sample rate.
|
|
|
|
* @warning do not attempt to resample outside of a synchronized block!
|
|
|
|
* @param rate number of samples.
|
|
|
|
* @param filter the interpolation filter.
|
|
|
|
* @return whether it was successful.
|
|
|
|
*/
|
|
|
|
bool resample(double rate, int filter);
|
|
|
|
|
2022-03-16 03:05:55 +00:00
|
|
|
/**
|
|
|
|
* initialize the rest of sample formats for this sample.
|
|
|
|
*/
|
2022-02-24 08:57:45 +00:00
|
|
|
void render();
|
2022-03-16 03:05:55 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* get the sample data for the current depth.
|
|
|
|
* @return the sample data, or NULL if not created.
|
|
|
|
*/
|
2022-02-24 08:57:45 +00:00
|
|
|
void* getCurBuf();
|
2022-03-16 03:05:55 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* get the sample data length for the current depth.
|
|
|
|
* @return the sample data length.
|
|
|
|
*/
|
2022-02-24 08:57:45 +00:00
|
|
|
unsigned int getCurBufLen();
|
2022-03-22 23:36:29 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* prepare an undo step for this sample.
|
|
|
|
* @param data whether to include sample data.
|
2022-03-23 05:42:59 +00:00
|
|
|
* @param doNotPush if this is true, don't push the DivSampleHistory to the undo history.
|
2022-03-22 23:36:29 +00:00
|
|
|
* @return the undo step.
|
|
|
|
*/
|
2022-03-23 05:42:59 +00:00
|
|
|
DivSampleHistory* prepareUndo(bool data, bool doNotPush=false);
|
2022-03-22 23:36:29 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* undo. you may need to call DivEngine::renderSamples afterwards.
|
|
|
|
* @warning do not attempt to undo outside of a synchronized block!
|
|
|
|
* @return 0 on failure; 1 on success and 2 on success (data changed).
|
|
|
|
*/
|
|
|
|
int undo();
|
|
|
|
|
|
|
|
/**
|
|
|
|
* redo. you may need to call DivEngine::renderSamples afterwards.
|
|
|
|
* @warning do not attempt to redo outside of a synchronized block!
|
|
|
|
* @return 0 on failure; 1 on success and 2 on success (data changed).
|
|
|
|
*/
|
|
|
|
int redo();
|
2021-05-13 07:39:26 +00:00
|
|
|
DivSample():
|
|
|
|
name(""),
|
2021-12-23 23:04:44 +00:00
|
|
|
rate(32000),
|
2022-01-27 21:52:06 +00:00
|
|
|
centerRate(8363),
|
2022-04-20 16:52:37 +00:00
|
|
|
loopStart(0),
|
2022-01-24 22:13:47 +00:00
|
|
|
loopOffP(0),
|
2022-04-20 16:52:37 +00:00
|
|
|
loopEnd(~0),
|
|
|
|
loopMode(DIV_SAMPLE_LOOPMODE_ONESHOT),
|
|
|
|
depth(DIV_SAMPLE_DEPTH_16BIT),
|
2022-02-24 08:57:45 +00:00
|
|
|
data8(NULL),
|
|
|
|
data16(NULL),
|
|
|
|
data1(NULL),
|
|
|
|
dataDPCM(NULL),
|
|
|
|
dataQSoundA(NULL),
|
|
|
|
dataA(NULL),
|
|
|
|
dataB(NULL),
|
|
|
|
dataX68(NULL),
|
|
|
|
dataBRR(NULL),
|
|
|
|
dataVOX(NULL),
|
|
|
|
length8(0),
|
|
|
|
length16(0),
|
|
|
|
length1(0),
|
|
|
|
lengthDPCM(0),
|
|
|
|
lengthQSoundA(0),
|
|
|
|
lengthA(0),
|
|
|
|
lengthB(0),
|
|
|
|
lengthX68(0),
|
|
|
|
lengthBRR(0),
|
|
|
|
lengthVOX(0),
|
|
|
|
off8(0),
|
|
|
|
off16(0),
|
|
|
|
off1(0),
|
|
|
|
offDPCM(0),
|
|
|
|
offQSoundA(0),
|
|
|
|
offA(0),
|
|
|
|
offB(0),
|
|
|
|
offX68(0),
|
|
|
|
offBRR(0),
|
|
|
|
offVOX(0),
|
|
|
|
offSegaPCM(0),
|
|
|
|
offQSound(0),
|
2022-04-14 23:25:59 +00:00
|
|
|
offX1_010(0),
|
2022-02-24 08:57:45 +00:00
|
|
|
samples(0) {}
|
2021-12-15 19:15:44 +00:00
|
|
|
~DivSample();
|
2021-05-11 20:08:08 +00:00
|
|
|
};
|
2022-04-20 16:52:37 +00:00
|
|
|
|
|
|
|
#endif
|