#include <cstring>
#include <stdint.h>
#include "rix.h"
#include "debug.h"
#if defined(__hppa__) || \
defined(__m68k__) || defined(mc68000) || defined(_M_M68K) || \
(defined(__MIPS__) && defined(__MISPEB__)) || \
defined(__ppc__) || defined(__POWERPC__) || defined(_M_PPC) || \
defined(__sparc__)
#define RIX_SWAP32(a) (((a) << 24) | (((a) << 8) & 0x00FF0000) | (((a) >> 8) & 0x0000FF00) | ((a) >> 24))
#else
#define RIX_SWAP32(a) (a)
#endif
const uint8_t CrixPlayer::adflag[] = {0,0,0,1,1,1,0,0,0,1,1,1,0,0,0,1,1,1};
const uint8_t CrixPlayer::reg_data[] = {0,1,2,3,4,5,8,9,10,11,12,13,16,17,18,19,20,21};
const uint8_t CrixPlayer::ad_C0_offs[] = {0,1,2,0,1,2,3,4,5,3,4,5,6,7,8,6,7,8};
const uint8_t CrixPlayer::modify[] = {0,3,1,4,2,5,6,9,7,10,8,11,12,15,13,16,14,17,12,\
15,16,0,14,0,17,0,13,0};
const uint8_t CrixPlayer::bd_reg_data[] = {
0x00,0x00,0x00,0x00,0x00,0x00,0x10,0x08,0x04,0x02,0x01,
0x00,0x01,0x01,0x03,0x0F,0x05,0x00,0x01,0x03,0x0F,0x00,
0x00,0x00,0x01,0x00,0x00,0x01,0x01,0x0F,0x07,0x00,0x02,
0x04,0x00,0x00,0x00,0x01,0x00,0x00,0x00,0x00,0x00,0x0A,
0x04,0x00,0x08,0x0C,0x0B,0x00,0x00,0x00,0x01,0x00,0x00,
0x00,0x00,0x0D,0x04,0x00,0x06,0x0F,0x00,0x00,0x00,0x00,
0x01,0x00,0x00,0x0C,0x00,0x0F,0x0B,0x00,0x08,0x05,0x00,
0x00,0x00,0x00,0x00,0x00,0x00,0x04,0x00,0x0F,0x0B,0x00,
0x07,0x05,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x01,0x00,
0x0F,0x0B,0x00,0x05,0x05,0x00,0x00,0x00,0x00,0x00,0x00,
0x00,0x01,0x00,0x0F,0x0B,0x00,0x07,0x05,0x00,0x00,0x00,
0x00,0x00,0x00};
uint8_t CrixPlayer::for40reg[] = {0x7F,0x7F,0x7F,0x7F,0x7F,0x7F,0x7F,0x7F,0x7F,
0x7F,0x7F,0x7F,0x7F,0x7F,0x7F,0x7F,0x7F,0x7F};
const uint16_t CrixPlayer::mus_time = 0x4268;
CPlayer *CrixPlayer::factory(Copl *newopl)
{
return new CrixPlayer(newopl);
}
CrixPlayer::CrixPlayer(Copl *newopl)
: CPlayer(newopl), flag_mkf(0), file_buffer(0), rix_buf(0)
{
}
CrixPlayer::~CrixPlayer()
{
if(file_buffer)
delete [] file_buffer;
}
bool CrixPlayer::load(const std::string &filename, const CFileProvider &fp)
{
binistream *f = fp.open(filename); if(!f) return false;
uint32_t i=0;
if(stricmp(filename.substr(filename.length()-4,4).c_str(),".mkf")==0)
{
flag_mkf=1;
f->seek(0);
int offset=f->readInt(4);
f->seek(offset);
}
if(f->readInt(2)!=0x55aa){ fp.close(f);return false; }
file_buffer = new uint8_t [fp.filesize(f) + 1];
f->seek(0);
while(!f->eof())
file_buffer[i++]=f->readInt(1);
length=i;
fp.close(f);
if(!flag_mkf)
rix_buf=file_buffer;
rewind(0);
return true;
}
bool CrixPlayer::update()
{
int_08h_entry();
return !play_end;
}
void CrixPlayer::rewind(int subsong)
{
I = 0; T = 0;
mus_block = 0;
ins_block = 0;
rhythm = 0;
music_on = 0;
pause_flag = 0;
band = 0;
band_low = 0;
e0_reg_flag = 0;
bd_modify = 0;
sustain = 0;
play_end = 0;
pos = index = 0;
memset(f_buffer, 0, sizeof(f_buffer));
memset(a0b0_data2, 0, sizeof(a0b0_data2));
memset(a0b0_data3, 0, sizeof(a0b0_data3));
memset(a0b0_data4, 0, sizeof(a0b0_data4));
memset(a0b0_data5, 0, sizeof(a0b0_data5));
memset(addrs_head, 0, sizeof(addrs_head));
memset(insbuf, 0, sizeof(insbuf));
memset(displace, 0, sizeof(displace));
memset(reg_bufs, 0, sizeof(reg_bufs));
memset(for40reg, 0x7F, sizeof(for40reg));
if(flag_mkf)
{
uint32_t *buf_index=(uint32_t *)file_buffer;
int offset1=RIX_SWAP32(buf_index[subsong]),offset2;
while((offset2=RIX_SWAP32(buf_index[++subsong]))==offset1);
length=offset2-offset1+1;
rix_buf=file_buffer+offset1;
}
opl->init();
opl->write(1,32); set_new_int();
data_initial();
}
uint32_t CrixPlayer::getsubsongs()
{
if(flag_mkf)
{
uint32_t *buf_index=(uint32_t *)file_buffer;
int songs=RIX_SWAP32(buf_index[0])/4,i=0;
for(i=0;i<songs;i++)
if(buf_index[i+1]==buf_index[i])
songs--;
return songs;
}
else
return 1;
}
float CrixPlayer::getrefresh()
{
return 70.0f;
}
inline void CrixPlayer::set_new_int()
{
ad_initial();
}
inline void CrixPlayer::Pause()
{
register uint16_t i;
pause_flag = 1;
for(i=0;i<11;i++)
switch_ad_bd(i);
}
inline void CrixPlayer::ad_a0b0l_reg_(uint16_t index,uint16_t p2,uint16_t p3)
{
a0b0_data4[index] = p3;
a0b0_data3[index] = p2;
}
inline void CrixPlayer::data_initial()
{
rhythm = rix_buf[2];
mus_block = (rix_buf[0x0D]<<8)+rix_buf[0x0C];
ins_block = (rix_buf[0x09]<<8)+rix_buf[0x08];
I = mus_block+1;
if(rhythm != 0)
{
ad_a0b0l_reg_(8,0x18,0);
ad_a0b0l_reg_(7,0x1F,0);
}
bd_modify = 0;
band = 0; music_on = 1;
}
inline uint16_t CrixPlayer::ad_initial()
{
uint16_t i,j,k = 0;
for(i=0;i<25;i++)
{
uint32_t res = ((uint32_t)i*24+10000)*52088/250000*0x24000/0x1B503;
f_buffer[i*12]=((uint16_t)res+4)>>3;
for(int t=1;t<12;t++)
{
res*=1.06;
f_buffer[i*12+t]=((uint16_t)res+4)>>3;
}
}
for(i=0;i<8;i++)
for(j=0;j<12;j++)
{
a0b0_data5[k] = i;
addrs_head[k] = j;
k++;
}
e0_reg_flag = 0x20;
return 1;}
inline void CrixPlayer::ad_bop(uint16_t reg,uint16_t value)
{
if(reg == 2 || reg == 3)
AdPlug_LogWrite("switch OPL2/3 mode!\n");
opl->write(reg & 0xff, value & 0xff);
}
inline void CrixPlayer::int_08h_entry()
{
uint16_t band_sus = 1;
while(band_sus)
{
if(sustain <= 0)
{
band_sus = rix_proc();
if(band_sus) sustain += band_sus;
else
{
play_end=1;
break;
}
}
else
{
if(band_sus) sustain -= 14;
break;
}
}
}
inline uint16_t CrixPlayer::rix_proc()
{
uint8_t ctrl = 0;
if(music_on == 0||pause_flag == 1) return 0;
band = 0;
while(rix_buf[I] != 0x80 && I<length-1)
{
band_low = rix_buf[I-1];
ctrl = rix_buf[I]; I+=2;
switch(ctrl&0xF0)
{
case 0x90: rix_get_ins(); rix_90_pro(ctrl&0x0F); break;
case 0xA0: rix_A0_pro(ctrl&0x0F,((uint16_t)band_low)<<6); break;
case 0xB0: rix_B0_pro(ctrl&0x0F,band_low); break;
case 0xC0: switch_ad_bd(ctrl&0x0F);
if(band_low != 0) rix_C0_pro(ctrl&0x0F,band_low);
break;
default: band = (ctrl<<8)+band_low; break;
}
if(band != 0) return band;
}
music_ctrl();
I = mus_block+1;
band = 0; music_on = 1;
return 0;
}
inline void CrixPlayer::rix_get_ins()
{
int i;
uint8_t *baddr = (&rix_buf[ins_block])+(band_low<<6);
for(i = 0; i < 28; i++)
insbuf[i] = (baddr[i * 2 + 1] << 8) + baddr[i * 2];
}
inline void CrixPlayer::rix_90_pro(uint16_t ctrl_l)
{
if(rhythm == 0 || ctrl_l < 6)
{
ins_to_reg(modify[ctrl_l*2],insbuf,insbuf[26]);
ins_to_reg(modify[ctrl_l*2+1],insbuf+13,insbuf[27]);
return;
}
else if(ctrl_l > 6)
{
ins_to_reg(modify[ctrl_l*2+6],insbuf,insbuf[26]);
return;
}
else
{
ins_to_reg(12,insbuf,insbuf[26]);
ins_to_reg(15,insbuf+13,insbuf[27]);
return;
}
}
inline void CrixPlayer::rix_A0_pro(uint16_t ctrl_l,uint16_t index)
{
if(rhythm == 0 || ctrl_l <= 6)
{
prepare_a0b0(ctrl_l,index>0x3FFF?0x3FFF:index);
ad_a0b0l_reg(ctrl_l,a0b0_data3[ctrl_l],a0b0_data4[ctrl_l]);
}
else return;
}
inline void CrixPlayer::prepare_a0b0(uint16_t index,uint16_t v)
{
short high = 0,low = 0; uint32_t res;
int res1 = (v-0x2000)*0x19;
if(res1 == (int)0xff) return;
low = res1/0x2000;
if(low < 0)
{
low = 0x18-low; high = (signed short)low<0?0xFFFF:0;
res = high; res<<=16; res+=low;
low = ((signed short)res)/(signed short)0xFFE7;
a0b0_data2[index] = low;
low = res;
res = low - 0x18;
high = (signed short)res%0x19;
low = (signed short)res/0x19;
if(high != 0) {low = 0x19; low = low-high;}
}
else
{
res = high = low;
low = (signed short)res/(signed short)0x19;
a0b0_data2[index] = low;
res = high;
low = (signed short)res%(signed short)0x19;
}
low = (signed short)low*(signed short)0x18;
displace[index] = low;
}
inline void CrixPlayer::ad_a0b0l_reg(uint16_t index,uint16_t p2,uint16_t p3)
{
uint16_t data; uint16_t i = p2+a0b0_data2[index];
a0b0_data4[index] = p3;
a0b0_data3[index] = p2;
i = ((signed short)i<=0x5F?i:0x5F);
i = ((signed short)i>=0?i:0);
data = f_buffer[addrs_head[i]+displace[index]/2];
ad_bop(0xA0+index,data);
data = a0b0_data5[i]*4+(p3<1?0:0x20)+((data>>8)&3);
ad_bop(0xB0+index,data);
}
inline void CrixPlayer::rix_B0_pro(uint16_t ctrl_l,uint16_t index)
{
register int temp = 0;
if(rhythm == 0 || ctrl_l < 6) temp = modify[ctrl_l*2+1];
else
{
temp = ctrl_l > 6?ctrl_l*2:ctrl_l*2+1;
temp = modify[temp+6];
}
for40reg[temp] = index>0x7F?0x7F:index;
ad_40_reg(temp);
}
inline void CrixPlayer::rix_C0_pro(uint16_t ctrl_l,uint16_t index)
{
register uint16_t i = index>=12?index-12:0;
if(ctrl_l < 6 || rhythm == 0)
{
ad_a0b0l_reg(ctrl_l,i,1);
return;
}
else
{
if(ctrl_l != 6)
{
if(ctrl_l == 8)
{
ad_a0b0l_reg(ctrl_l,i,0);
ad_a0b0l_reg(7,i+7,0);
}
}
else ad_a0b0l_reg(ctrl_l,i,0);
bd_modify |= bd_reg_data[ctrl_l];
ad_bd_reg();
return;
}
}
inline void CrixPlayer::switch_ad_bd(uint16_t index)
{
if(rhythm == 0 || index < 6) ad_a0b0l_reg(index,a0b0_data3[index],0);
else
{
bd_modify &= (~bd_reg_data[index]),
ad_bd_reg();
}
}
inline void CrixPlayer::ins_to_reg(uint16_t index,uint16_t* insb,uint16_t value)
{
register uint16_t i;
for(i=0;i<13;i++) reg_bufs[index].v[i] = insb[i];
reg_bufs[index].v[13] = value&3;
ad_bd_reg(),ad_08_reg(),
ad_40_reg(index),ad_C0_reg(index),ad_60_reg(index),
ad_80_reg(index),ad_20_reg(index),ad_E0_reg(index);
}
inline void CrixPlayer::ad_E0_reg(uint16_t index)
{
uint16_t data = e0_reg_flag == 0?0:(reg_bufs[index].v[13]&3);
ad_bop(0xE0+reg_data[index],data);
}
inline void CrixPlayer::ad_20_reg(uint16_t index)
{
uint16_t data = (reg_bufs[index].v[9] < 1?0:0x80);
data += (reg_bufs[index].v[10] < 1?0:0x40);
data += (reg_bufs[index].v[5] < 1?0:0x20);
data += (reg_bufs[index].v[11] < 1?0:0x10);
data += (reg_bufs[index].v[1]&0x0F);
ad_bop(0x20+reg_data[index],data);
}
inline void CrixPlayer::ad_80_reg(uint16_t index)
{
uint16_t data = (reg_bufs[index].v[7]&0x0F),temp = reg_bufs[index].v[4];
data |= (temp << 4);
ad_bop(0x80+reg_data[index],data);
}
inline void CrixPlayer::ad_60_reg(uint16_t index)
{
uint16_t data = reg_bufs[index].v[6]&0x0F,temp = reg_bufs[index].v[3];
data |= (temp << 4);
ad_bop(0x60+reg_data[index],data);
}
inline void CrixPlayer::ad_C0_reg(uint16_t index)
{
uint16_t data = reg_bufs[index].v[2];
if(adflag[index] == 1) return;
data *= 2,
data |= (reg_bufs[index].v[12] < 1?1:0);
ad_bop(0xC0+ad_C0_offs[index],data);
}
inline void CrixPlayer::ad_40_reg(uint16_t index)
{
uint32_t res = 0;
uint16_t data = 0,temp = reg_bufs[index].v[0];
data = 0x3F - (0x3F & reg_bufs[index].v[8]),
data *= for40reg[index],
data *= 2,
data += 0x7F,
res = data;
data = res/0xFE,
data -= 0x3F,
data = -data,
data |= temp<<6;
ad_bop(0x40+reg_data[index],data);
}
inline void CrixPlayer::ad_bd_reg()
{
uint16_t data = rhythm < 1? 0:0x20;
data |= bd_modify;
ad_bop(0xBD,data);
}
inline void CrixPlayer::ad_a0b0_reg(uint16_t index)
{
ad_bop(0xA0+index,0);
ad_bop(0xB0+index,0);
}
inline void CrixPlayer::music_ctrl()
{
register int i;
for(i=0;i<11;i++)
switch_ad_bd(i);
}