allow loading wavetables

This commit is contained in:
tildearrow 2022-01-21 17:59:48 -05:00
parent c82440a7bc
commit 186e491c59
5 changed files with 126 additions and 8 deletions

View file

@ -1,3 +1,4 @@
#include <cstdint>
#define _USE_MATH_DEFINES #define _USE_MATH_DEFINES
#include "engine.h" #include "engine.h"
#include "instrument.h" #include "instrument.h"
@ -1431,14 +1432,10 @@ bool DivEngine::loadFur(unsigned char* file, size_t len) {
delete[] file; delete[] file;
return false; return false;
} }
reader.readI();
DivWavetable* wave=new DivWavetable; DivWavetable* wave=new DivWavetable;
reader.seek(wavePtr[i],SEEK_SET);
reader.readString(); // ignored for now wave->readWaveData(reader,ds.version);
wave->len=reader.readI();
wave->min=reader.readI();
wave->max=reader.readI();
reader.read(wave->data,4*wave->len);
ds.wave.push_back(wave); ds.wave.push_back(wave);
} }
@ -3079,8 +3076,113 @@ int DivEngine::addWave() {
} }
bool DivEngine::addWaveFromFile(const char* path) { bool DivEngine::addWaveFromFile(const char* path) {
FILE* f=ps_fopen(path,"rb");
if (f==NULL) {
return false;
}
unsigned char* buf;
ssize_t len;
if (fseek(f,0,SEEK_END)!=0) {
fclose(f);
return false;
}
len=ftell(f);
if (len<0) {
fclose(f);
return false;
}
if (len==0) {
fclose(f);
return false;
}
if (fseek(f,0,SEEK_SET)!=0) {
fclose(f);
return false;
}
buf=new unsigned char[len];
if (fread(buf,1,len,f)!=(size_t)len) {
logW("did not read entire wavetable file buffer!\n");
delete[] buf;
return false;
}
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;
}
} catch (EndOfFileException e) {
reader.seek(0,SEEK_SET);
}
DivWavetable* wave=new DivWavetable;
try {
if (isFurnaceTable) {
reader.seek(16,SEEK_SET);
short version=reader.readS();
reader.readS(); // reserved
reader.read(magic,4);
if (memcmp(magic,"WAVE",4)!=0) {
logE("invalid wavetable header!\n");
lastError="invalid wavetable header!";
delete[] buf;
return false;
}
reader.seek(20,SEEK_SET);
wave->readWaveData(reader,version);
} else {
try {
// read as .dmw
reader.seek(0,SEEK_SET);
int len=reader.readI();
wave->max=(unsigned char)reader.readC();
if (reader.size()==(size_t)(len+5)) {
// read as .dmw
logI("reading .dmw...\n");
if (len>256) len=256;
for (int i=0; i<len; i++) {
wave->data[i]=(unsigned char)reader.readC();
}
} else {
// read as binary
logI("reading binary...\n");
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;
}
} catch (EndOfFileException e) {
// read as binary
len=reader.size();
logI("reading binary for being too small...\n");
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;
}
}
} catch (EndOfFileException e) {
delete wave;
delete[] buf;
return false;
}
isBusy.lock(); isBusy.lock();
// TODO int waveCount=(int)song.wave.size();
song.wave.push_back(wave);
song.waveLen=waveCount+1;
isBusy.unlock(); isBusy.unlock();
return true; return true;
} }

View file

@ -11,7 +11,7 @@
#include <queue> #include <queue>
#define DIV_VERSION "0.4.6" #define DIV_VERSION "0.4.6"
#define DIV_ENGINE_VERSION 27 #define DIV_ENGINE_VERSION 28
enum DivStatusView { enum DivStatusView {
DIV_STATUS_NOTHING=0, DIV_STATUS_NOTHING=0,

View file

@ -16,6 +16,17 @@ void DivWavetable::putWaveData(SafeWriter* w) {
} }
} }
void DivWavetable::readWaveData(SafeReader& reader, short version) {
reader.readI();
reader.readI();
reader.readString(); // ignored for now
len=reader.readI();
min=reader.readI();
max=reader.readI();
reader.read(data,4*len);
}
bool DivWavetable::save(const char* path) { bool DivWavetable::save(const char* path) {
SafeWriter* w=new SafeWriter(); SafeWriter* w=new SafeWriter();
w->init(); w->init();

View file

@ -7,6 +7,7 @@ struct DivWavetable {
int data[256]; int data[256];
void putWaveData(SafeWriter* w); void putWaveData(SafeWriter* w);
void readWaveData(SafeReader& reader, short version);
bool save(const char* path); bool save(const char* path);
DivWavetable(): DivWavetable():
len(32), len(32),

View file

@ -3964,7 +3964,11 @@ bool FurnaceGUI::loop() {
exportAudio(copyOfName,DIV_EXPORT_MODE_MANY_CHAN); exportAudio(copyOfName,DIV_EXPORT_MODE_MANY_CHAN);
break; break;
case GUI_FILE_INS_OPEN: case GUI_FILE_INS_OPEN:
break;
case GUI_FILE_WAVE_OPEN: case GUI_FILE_WAVE_OPEN:
e->addWaveFromFile(copyOfName.c_str());
modified=true;
break;
case GUI_FILE_EXPORT_VGM: case GUI_FILE_EXPORT_VGM:
case GUI_FILE_EXPORT_ROM: case GUI_FILE_EXPORT_ROM:
showError("Coming soon!"); showError("Coming soon!");