#include <string.h>
#include <stdio.h>
#include <inttypes.h>
#include "debug.h"
#include "d00.h"
#define HIBYTE(val) (val >> 8)
#define LOBYTE(val) (val & 0xff)
static const unsigned short notetable[12] = {340,363,385,408,432,458,485,514,544,577,611,647};
static inline uint16_t LE_WORD(const uint16_t *val)
{
const uint8_t *b = (const uint8_t *)val;
return (b[1] << 8) + b[0];
}
CPlayer *Cd00Player::factory(Copl *newopl)
{
return new Cd00Player(newopl);
}
bool Cd00Player::load(const std::string &filename, const CFileProvider &fp)
{
binistream *f = fp.open(filename); if(!f) return false;
d00header *checkhead;
d00header1 *ch;
unsigned long filesize;
int i,ver1=0;
char *str;
checkhead = new d00header;
f->readString((char *)checkhead, sizeof(d00header));
if(strncmp(checkhead->id,"JCH\x26\x02\x66",6) || checkhead->type ||
!checkhead->subsongs || checkhead->soundcard) {
delete checkhead;
if(!fp.extension(filename, ".d00")) { fp.close(f); return false; }
ch = new d00header1;
f->seek(0); f->readString((char *)ch, sizeof(d00header1));
if(ch->version > 1 || !ch->subsongs)
{ delete ch; fp.close(f); return false; }
delete ch;
ver1 = 1;
} else
delete checkhead;
AdPlug_LogWrite("Cd00Player::load(f,\"%s\"): %s format D00 file detected!\n",
filename.c_str(), ver1 ? "Old" : "New");
filesize = fp.filesize(f); f->seek(0);
filedata = new char [filesize + 1]; f->readString((char *)filedata, filesize);
fp.close(f);
if(!ver1) { header = (struct d00header *)filedata;
version = header->version;
datainfo = (char *)filedata + LE_WORD(&header->infoptr);
inst = (struct Sinsts *)((char *)filedata + LE_WORD(&header->instptr));
seqptr = (unsigned short *)((char *)filedata + LE_WORD(&header->seqptr));
for(i=31;i>=0;i--) if(header->songname[i] == ' ')
header->songname[i] = '\0';
else
break;
for(i=31;i>=0;i--)
if(header->author[i] == ' ')
header->author[i] = '\0';
else
break;
} else { header1 = (struct d00header1 *)filedata;
version = header1->version;
datainfo = (char *)filedata + LE_WORD(&header1->infoptr);
inst = (struct Sinsts *)((char *)filedata + LE_WORD(&header1->instptr));
seqptr = (unsigned short *)((char *)filedata + LE_WORD(&header1->seqptr));
}
switch(version) {
case 0:
levpuls = 0;
spfx = 0;
header1->speed = 70; break;
case 1:
levpuls = (struct Slevpuls *)((char *)filedata + LE_WORD(&header1->lpulptr));
spfx = 0;
break;
case 2:
levpuls = (struct Slevpuls *)((char *)filedata + LE_WORD(&header->spfxptr));
spfx = 0;
break;
case 3:
spfx = 0;
levpuls = 0;
break;
case 4:
spfx = (struct Sspfx *)((char *)filedata + LE_WORD(&header->spfxptr));
levpuls = 0;
break;
}
if((str = strstr(datainfo,"\xff\xff")))
while((*str == '\xff' || *str == ' ') && str >= datainfo) {
*str = '\0'; str--;
}
else memset((char *)filedata+filesize,0,1);
rewind(0);
return true;
}
bool Cd00Player::update()
{
unsigned char c,cnt,trackend=0,fx,note;
unsigned short ord,*patt,buf,fxop,pattpos;
for(c=0;c<9;c++) {
channel[c].slideval += channel[c].slide; setfreq(c); vibrato(c);
if(channel[c].spfx != 0xffff) { if(channel[c].fxdel)
channel[c].fxdel--;
else {
channel[c].spfx = LE_WORD(&spfx[channel[c].spfx].ptr);
channel[c].fxdel = spfx[channel[c].spfx].duration;
channel[c].inst = LE_WORD(&spfx[channel[c].spfx].instnr) & 0xfff;
if(spfx[channel[c].spfx].modlev != 0xff)
channel[c].modvol = spfx[channel[c].spfx].modlev;
setinst(c);
if(LE_WORD(&spfx[channel[c].spfx].instnr) & 0x8000) note = spfx[channel[c].spfx].halfnote;
else note = spfx[channel[c].spfx].halfnote + channel[c].note;
channel[c].freq = notetable[note%12] + ((note/12) << 10);
setfreq(c);
}
channel[c].modvol += spfx[channel[c].spfx].modlevadd; channel[c].modvol &= 63;
setvolume(c);
}
if(channel[c].levpuls != 0xff) { if(channel[c].frameskip)
channel[c].frameskip--;
else {
channel[c].frameskip = inst[channel[c].inst].timer;
if(channel[c].fxdel)
channel[c].fxdel--;
else {
channel[c].levpuls = levpuls[channel[c].levpuls].ptr - 1;
channel[c].fxdel = levpuls[channel[c].levpuls].duration;
if(levpuls[channel[c].levpuls].level != 0xff)
channel[c].modvol = levpuls[channel[c].levpuls].level;
}
channel[c].modvol += levpuls[channel[c].levpuls].voladd; channel[c].modvol &= 63;
setvolume(c);
}
}
}
for(c=0;c<9;c++)
if(version < 3 ? channel[c].del : channel[c].del <= 0x7f) {
if(version == 4) if(channel[c].del == inst[channel[c].inst].timer)
if(channel[c].nextnote)
opl->write(0x83 + op_table[c], inst[channel[c].inst].sr);
if(version < 3)
channel[c].del--;
else
if(channel[c].speed)
channel[c].del += channel[c].speed;
else {
channel[c].seqend = 1;
continue;
}
} else {
if(channel[c].speed) {
if(version < 3)
channel[c].del = channel[c].speed;
else {
channel[c].del &= 0x7f;
channel[c].del += channel[c].speed;
}
} else {
channel[c].seqend = 1;
continue;
}
if(channel[c].rhcnt) { channel[c].rhcnt--;
continue;
}
readorder: ord = LE_WORD(&channel[c].order[channel[c].ordpos]);
switch(ord) {
case 0xfffe: channel[c].seqend = 1; continue; case 0xffff: channel[c].ordpos = LE_WORD(&channel[c].order[channel[c].ordpos + 1]);
channel[c].seqend = 1;
goto readorder;
default:
if(ord >= 0x9000) { channel[c].speed = ord & 0xff;
ord = LE_WORD(&channel[c].order[channel[c].ordpos - 1]);
channel[c].ordpos++;
} else
if(ord >= 0x8000) { channel[c].transpose = ord & 0xff;
if(ord & 0x100)
channel[c].transpose = -channel[c].transpose;
ord = LE_WORD(&channel[c].order[++channel[c].ordpos]);
}
patt = (unsigned short *)((char *)filedata + LE_WORD(&seqptr[ord]));
break;
}
channel[c].fxflag = 0;
readseq: if(!version) channel[c].rhcnt = channel[c].irhcnt;
pattpos = LE_WORD(&patt[channel[c].pattpos]);
if(pattpos == 0xffff) { channel[c].pattpos = 0;
channel[c].ordpos++;
goto readorder;
}
cnt = HIBYTE(pattpos);
note = LOBYTE(pattpos);
fx = pattpos >> 12;
fxop = pattpos & 0x0fff;
channel[c].pattpos++; pattpos = LE_WORD(&patt[channel[c].pattpos]);
channel[c].nextnote = LOBYTE(pattpos) & 0x7f;
if(version ? cnt < 0x40 : !fx) { switch(note) {
case 0: case 0x80:
if(!note || version) {
channel[c].key = 0;
setfreq(c);
}
case 0x7e: if(version)
channel[c].rhcnt = cnt;
channel[c].nextnote = 0;
break;
default: if(!(channel[c].fxflag & 1))
channel[c].vibdepth = 0;
if(!(channel[c].fxflag & 2))
channel[c].slideval = channel[c].slide = 0;
if(version) { if(note > 0x80) note -= 0x80;
else note += channel[c].transpose;
channel[c].note = note;
if(channel[c].ispfx != 0xffff && cnt < 0x20) { channel[c].spfx = channel[c].ispfx;
if(LE_WORD(&spfx[channel[c].spfx].instnr) & 0x8000) note = spfx[channel[c].spfx].halfnote;
else note += spfx[channel[c].spfx].halfnote;
channel[c].inst = LE_WORD(&spfx[channel[c].spfx].instnr) & 0xfff;
channel[c].fxdel = spfx[channel[c].spfx].duration;
if(spfx[channel[c].spfx].modlev != 0xff)
channel[c].modvol = spfx[channel[c].spfx].modlev;
else
channel[c].modvol = inst[channel[c].inst].data[7] & 63;
}
if(channel[c].ilevpuls != 0xff && cnt < 0x20) { channel[c].levpuls = channel[c].ilevpuls;
channel[c].fxdel = levpuls[channel[c].levpuls].duration;
channel[c].frameskip = inst[channel[c].inst].timer;
if(levpuls[channel[c].levpuls].level != 0xff)
channel[c].modvol = levpuls[channel[c].levpuls].level;
else
channel[c].modvol = inst[channel[c].inst].data[7] & 63;
}
channel[c].freq = notetable[note%12] + ((note/12) << 10);
if(cnt < 0x20) playnote(c);
else { setfreq(c);
cnt -= 0x20; }
channel[c].rhcnt = cnt;
} else { if(cnt < 2) note += channel[c].transpose;
channel[c].note = note;
channel[c].freq = notetable[note%12] + ((note/12) << 10);
if(cnt == 1) setfreq(c);
else playnote(c);
}
break;
}
continue; } else { switch(fx) {
case 6: buf = channel[c].inst;
channel[c].inst = 0;
playnote(c);
channel[c].inst = buf;
channel[c].rhcnt = fxop;
continue; case 7: channel[c].vibspeed = fxop & 0xff;
channel[c].vibdepth = fxop >> 8;
channel[c].trigger = fxop >> 9;
channel[c].fxflag |= 1;
break;
case 8: if(!version)
channel[c].irhcnt = fxop;
break;
case 9: channel[c].vol = fxop & 63;
if(channel[c].vol + channel[c].cvol < 63) channel[c].vol += channel[c].cvol;
else
channel[c].vol = 63;
setvolume(c);
break;
case 0xb: if(version == 4)
channel[c].ispfx = fxop;
break;
case 0xc: channel[c].ispfx = 0xffff;
channel[c].spfx = 0xffff;
channel[c].inst = fxop;
channel[c].modvol = inst[fxop].data[7] & 63;
if(version < 3 && version && inst[fxop].tunelev) channel[c].ilevpuls = inst[fxop].tunelev - 1;
else {
channel[c].ilevpuls = 0xff;
channel[c].levpuls = 0xff;
}
break;
case 0xd: channel[c].slide = fxop;
channel[c].fxflag |= 2;
break;
case 0xe: channel[c].slide = -fxop;
channel[c].fxflag |= 2;
break;
}
goto readseq; }
}
for(c=0;c<9;c++)
if(channel[c].seqend)
trackend++;
if(trackend == 9)
songend = 1;
return !songend;
}
void Cd00Player::rewind(int subsong)
{
struct Stpoin {
unsigned short ptr[9];
unsigned char volume[9],dummy[5];
} tpoin;
int i;
if(subsong == -1) subsong = cursubsong;
if(version > 1) { if(subsong >= header->subsongs)
return;
} else
if(subsong >= header1->subsongs)
return;
memset(channel,0,sizeof(channel));
const unsigned char* data = (unsigned char*)filedata + sizeof(Stpoin)*subsong;
if(version > 1)
data += LE_WORD(&header->tpoin);
else
data += LE_WORD(&header1->tpoin);
memcpy(&tpoin, data, sizeof(Stpoin));
for(i=0;i<9;i++) {
if(LE_WORD(&tpoin.ptr[i])) { channel[i].speed = LE_WORD((unsigned short *)
((char *)filedata + LE_WORD(&tpoin.ptr[i])));
channel[i].order = (unsigned short *)
((char *)filedata + LE_WORD(&tpoin.ptr[i]) + 2);
} else { channel[i].speed = 0;
channel[i].order = 0;
}
channel[i].ispfx = 0xffff; channel[i].spfx = 0xffff; channel[i].ilevpuls = 0xff; channel[i].levpuls = 0xff; channel[i].cvol = tpoin.volume[i] & 0x7f; channel[i].vol = channel[i].cvol; }
songend = 0;
opl->init(); opl->write(1,32); cursubsong = subsong;
}
std::string Cd00Player::gettype()
{
char tmpstr[40];
sprintf(tmpstr,"EdLib packed (version %d)",version > 1 ? header->version : header1->version);
return std::string(tmpstr);
}
float Cd00Player::getrefresh()
{
if(version > 1)
return header->speed;
else
return header1->speed;
}
unsigned int Cd00Player::getsubsongs()
{
if(version <= 1) return header1->subsongs;
else
return header->subsongs;
}
void Cd00Player::setvolume(unsigned char chan)
{
unsigned char op = op_table[chan];
unsigned short insnr = channel[chan].inst;
opl->write(0x43 + op,(int)(63-((63-(inst[insnr].data[2] & 63))/63.0)*(63-channel[chan].vol)) +
(inst[insnr].data[2] & 192));
if(inst[insnr].data[10] & 1)
opl->write(0x40 + op,(int)(63-((63-channel[chan].modvol)/63.0)*(63-channel[chan].vol)) +
(inst[insnr].data[7] & 192));
else
opl->write(0x40 + op,channel[chan].modvol + (inst[insnr].data[7] & 192));
}
void Cd00Player::setfreq(unsigned char chan)
{
unsigned short freq = channel[chan].freq;
if(version == 4) freq += inst[channel[chan].inst].tunelev;
freq += channel[chan].slideval;
opl->write(0xa0 + chan, freq & 255);
if(channel[chan].key)
opl->write(0xb0 + chan, ((freq >> 8) & 31) | 32);
else
opl->write(0xb0 + chan, (freq >> 8) & 31);
}
void Cd00Player::setinst(unsigned char chan)
{
unsigned char op = op_table[chan];
unsigned short insnr = channel[chan].inst;
opl->write(0x63 + op, inst[insnr].data[0]);
opl->write(0x83 + op, inst[insnr].data[1]);
opl->write(0x23 + op, inst[insnr].data[3]);
opl->write(0xe3 + op, inst[insnr].data[4]);
opl->write(0x60 + op, inst[insnr].data[5]);
opl->write(0x80 + op, inst[insnr].data[6]);
opl->write(0x20 + op, inst[insnr].data[8]);
opl->write(0xe0 + op, inst[insnr].data[9]);
if(version)
opl->write(0xc0 + chan, inst[insnr].data[10]);
else
opl->write(0xc0 + chan, (inst[insnr].data[10] << 1) + (inst[insnr].tunelev & 1));
}
void Cd00Player::playnote(unsigned char chan)
{
opl->write(0xb0 + chan, 0); setinst(chan);
channel[chan].key = 1;
setfreq(chan);
setvolume(chan);
}
void Cd00Player::vibrato(unsigned char chan)
{
if(!channel[chan].vibdepth)
return;
if(channel[chan].trigger)
channel[chan].trigger--;
else {
channel[chan].trigger = channel[chan].vibdepth;
channel[chan].vibspeed = -channel[chan].vibspeed;
}
channel[chan].freq += channel[chan].vibspeed;
setfreq(chan);
}