1
0
Fork 0
forked from len0rd/rockbox
foxbox/apps/plugins/midi/synth.c
Stepan Moskovchenko 4b773c0c4c Added xxx2wav support, an icon, and some optimizations
git-svn-id: svn://svn.rockbox.org/rockbox/trunk@6306 a1c6a512-1295-4272-9138-f99709370657
2005-04-16 03:35:20 +00:00

440 lines
9.1 KiB
C

/***************************************************************************
* __________ __ ___.
* Open \______ \ ____ ____ | | _\_ |__ _______ ___
* Source | _// _ \_/ ___\| |/ /| __ \ / _ \ \/ /
* Jukebox | | ( <_> ) \___| < | \_\ ( <_> > < <
* Firmware |____|_ /\____/ \___ >__|_ \|___ /\____/__/\_ \
* \/ \/ \/ \/ \/
*
* Copyright (C) 2005 Stepan Moskovchenko
*
* All files in this archive are subject to the GNU General Public License.
* See the file COPYING in the source tree root for full license agreement.
*
* This software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY
* KIND, either express or implied.
*
****************************************************************************/
extern struct plugin_api * rb;
struct Event * getEvent(struct Track * tr, int evNum)
{
return tr->dataBlock + (evNum*sizeof(struct Event));
}
void readTextBlock(int file, char * buf)
{
char c = 0;
do
{
c = readChar(file);
} while(c == '\n' || c == ' ' || c=='\t');
rb->lseek(file, -1, SEEK_CUR);
int cp = 0;
do
{
c = readChar(file);
buf[cp] = c;
cp++;
} while (c != '\n' && c != ' ' && c != '\t' && !eof(file));
buf[cp-1]=0;
rb->lseek(file, -1, SEEK_CUR);
}
//Filename is the name of the config file
//The MIDI file should have been loaded at this point
int initSynth(struct MIDIfile * mf, char * filename, char * drumConfig)
{
char patchUsed[128];
char drumUsed[128];
int a=0;
for(a=0; a<MAX_VOICES; a++)
{
voices[a].cp=0;
voices[a].vol=0;
voices[a].ch=0;
voices[a].isUsed=0;
voices[a].note=0;
}
for(a=0; a<16; a++)
{
chVol[a]=100; //Default, not quite full blast..
chPanLeft[a]=64; //Center
chPanRight[a]=64; //Center
chPat[a]=0; //Ac Gr Piano
chPW[a]=64; // .. not .. bent ?
}
for(a=0; a<128; a++)
{
patchSet[a]=NULL;
drumSet[a]=NULL;
patchUsed[a]=0;
drumUsed[a]=0;
}
//Always load the piano.
//Some files will assume its loaded without specifically
//issuing a Patch command... then we wonder why we can't hear anything
patchUsed[0]=1;
//Scan the file to see what needs to be loaded
for(a=0; a<mf->numTracks; a++)
{
int ts=0;
if(mf->tracks[a] == NULL)
{
printf("\nNULL TRACK !!!");
rb->splash(HZ*2, true, "Null Track in loader.");
return -1;
}
for(ts=0; ts<mf->tracks[a]->numEvents; ts++)
{
if((getEvent(mf->tracks[a], ts)->status) == (MIDI_NOTE_ON+9))
drumUsed[getEvent(mf->tracks[a], ts)->d1]=1;
if( (getEvent(mf->tracks[a], ts)->status & 0xF0) == MIDI_PRGM)
{
if(patchUsed[getEvent(mf->tracks[a], ts)->d1]==0)
printf("\nI need to load patch %d.", getEvent(mf->tracks[a], ts)->d1);
patchUsed[getEvent(mf->tracks[a], ts)->d1]=1;
}
}
}
int file = rb->open(filename, O_RDONLY);
if(file == -1)
{
rb->splash(HZ*2, true, "Bad patch config.\nDid you install the patchset?");
return -1;
}
char name[40];
char fn[40];
//Scan our config file and load the right patches as needed
int c = 0;
rb->snprintf(name, 40, "");
for(a=0; a<128; a++)
{
while(readChar(file)!=' ' && !eof(file));
readTextBlock(file, name);
rb->snprintf(fn, 40, "/.rockbox/patchset/%s.pat", name);
printf("\nLOADING: <%s> ", fn);
if(patchUsed[a]==1)
patchSet[a]=gusload(fn);
// if(patchSet[a] == NULL)
// return -1;
while((c != '\n'))
c = readChar(file);
}
rb->close(file);
file = rb->open(drumConfig, O_RDONLY);
if(file == -1)
{
rb->splash(HZ*2, true, "Bad drum config.\nDid you install the patchset?");
return -1;
}
//Scan our config file and load the drum data
int idx=0;
char number[30];
while(!eof(file))
{
readTextBlock(file, number);
readTextBlock(file, name);
rb->snprintf(fn, 40, "/.rockbox/patchset/%s.pat", name);
idx = rb->atoi(number);
if(idx == 0)
break;
if(drumUsed[idx]==1)
drumSet[idx]=gusload(fn);
// if(drumSet[idx] == NULL)
// return -1;
while((c != '\n') && (c != 255) && (!eof(file)))
c = readChar(file);
}
rb->close(file);
return 0;
}
inline signed short int getSample(struct GWaveform * wf, unsigned int s)
{
//16 bit samples
if(wf->mode&1)
{
if(s<<1 >= wf->wavSize)
return 0;
/*
* Probably put the signed/unsigned and and 8-16 bit conversion
* into the patch loader and have it run there, once.
*/
//Sign conversion moved into guspat.c
unsigned char b1=wf->data[s<<1]; //+((wf->mode & 2) << 6);
unsigned char b2=wf->data[(s<<1)|1]; //+((wf->mode & 2) << 6);
return (b1 | (b2<<8)) ;
}
else
{ //8-bit samples
//Do we even have anything 8-bit in our set?
unsigned char b1=wf->data[s]+((wf->mode & 2) << 6);
return b1<<8;
}
}
inline void setPoint(struct SynthObject * so, int pt)
{
if(so->ch==9) //Drums, no ADSR
{
so->curOffset = 1<<27;
so->curRate = 1;
return;
}
if(so->wf==NULL)
{
printf("\nCrap... null waveform...");
exit(1);
}
if(so->wf->envRate==NULL)
{
printf("\nWaveform has no envelope set");
exit(1);
}
so->curPoint = pt;
int r=0;
int rate = so->wf->envRate[pt];
r=3-((rate>>6) & 0x3); // Some blatant Timidity code for rate conversion...
r*=3;
r = (rate & 0x3f) << r;
/*
Okay. This is the rate shift. Timidity defaults to 9, and sets
it to 10 if you use the fast decay option. Slow decay sounds better
on some files, except on some other files... you get chords that aren't
done decaying yet.. and they dont harmonize with the next chord and it
sounds like utter crap. Yes, even Timitidy does that. So I'm going to
default this to 10, and maybe later have an option to set it to 9
for longer decays.
*/
so->curRate = r<<10;
so->targetOffset = so->wf->envOffset[pt]<<(20);
if(pt==0)
so->curOffset = 0;
}
long msi=0;
inline void stopVoice(struct SynthObject * so)
{
if(so->state == STATE_RAMPDOWN)
return;
so->state = STATE_RAMPDOWN;
so->decay = 255;
}
int rampDown = 0;
inline signed short int synthVoice(int v)
{
//Probably can combine these 2 lines into one..
//But for now, this looks more readable
// struct GPatch * pat = patchSet[chPat[voices[v].ch]];
// struct GWaveform * wf = pat->waveforms[pat->noteTable[voices[v].note]];
struct SynthObject * so = &voices[v];
struct GWaveform * wf = so->wf;
signed int s;
if(so->state != STATE_RAMPDOWN)
{
if(so->loopDir==LOOPDIR_FORWARD)
{
so->cp += so->delta;
}
else
{
so->cp -= so->delta;
}
}
if( (so->cp>>9 >= (wf->wavSize)) && (so->state != STATE_RAMPDOWN))
stopVoice(so);
/*
//Original, working, no interpolation
s=getSample(wf, (so->cp>>10));
*/
int s2=getSample(wf, (so->cp>>10)+1);
if((wf->mode & (LOOP_REVERSE|LOOP_PINGPONG)) && so->loopState == STATE_LOOPING && (so->cp>>10 <= (wf->startLoop>>1)))
{
if(wf->mode & LOOP_REVERSE)
{
so->cp = (wf->endLoop)<<9;
s2=getSample(wf, (so->cp>>10));
} else
so->loopDir = LOOPDIR_FORWARD;
}
if((wf->mode & 28) && (so->cp>>10 >= wf->endLoop>>1))
{
so->loopState = STATE_LOOPING;
if((wf->mode & (24)) == 0)
{
so->cp = (wf->startLoop)<<9;
s2=getSample(wf, (so->cp>>10));
} else
so->loopDir = LOOPDIR_REVERSE;
}
//Better, working, linear interpolation
int s1=getSample(wf, (so->cp>>10));
s = s1 + ((long)((s2 - s1) * (so->cp & 1023))>>10);
if(so->curRate == 0)
stopVoice(so);
if(so->ch != 9) //Stupid ADSR code... and don't do ADSR for drums
{
if(so->curOffset < so->targetOffset)
{
so->curOffset += (so->curRate);
if(so -> curOffset > so->targetOffset && so->curPoint != 2)
{
if(so->curPoint != 5)
setPoint(so, so->curPoint+1);
else
stopVoice(so);
}
} else
{
so->curOffset -= (so->curRate);
if(so -> curOffset < so->targetOffset && so->curPoint != 2)
{
if(so->curPoint != 5)
setPoint(so, so->curPoint+1);
else
stopVoice(so);
}
}
}
if(so->curOffset < 0)
so->isUsed=0; //This is OK
s = s * (so->curOffset >> 22);
s = s>>6;
if(so->state == STATE_RAMPDOWN)
{
so->decay--;
if(so->decay == 0)
so->isUsed=0;
}
s = s * so->decay; s = s >> 9;
return s*((signed short int)so->vol*(signed short int)chVol[so->ch])>>14;
}
int mhL[16];
int mhR[16];
int mp=0; //Mix position, for circular array
// Was stuff for Ghetto Lowpass Filter, now deprecated.
inline void synthSample(int * mixL, int * mixR)
{
int a=0;
signed long int leftMix=0, rightMix=0, sample=0;
for(a=0; a<MAX_VOICES; a++)
{
if(voices[a].isUsed==1)
{
sample = synthVoice(a);
leftMix += (sample*chPanLeft[voices[a].ch])>>7;
rightMix += (sample*chPanRight[voices[a].ch])>>7;
}
}
//TODO: Automatic Gain Control, anyone?
//Or, should this be implemented on the DSP's output volume instead?
*mixL = leftMix;
*mixR = rightMix;
return; //No more ghetto lowpass filter.. linear intrpolation works well.
// HACK HACK HACK
// This is the infamous Ghetto Lowpass Filter
// Now that I have linear interpolation, it should not be needed anymore.
/*
mp++;
if(mp==4)
mp=0;
mhL[mp]=leftMix;
mhR[mp]=rightMix;
*mixL = 0;
*mixR = 0;
for(a=0; a<4; a++)
{
*mixL += mhL[a];
*mixR += mhR[a];
}
*mixL = *mixL>>4;
*mixR = *mixR>>4;
*/
// END HACK END HACK END HACK
}