123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570 |
- /*
- * Adplug - Replayer for many OPL2/OPL3 audio file formats.
- * Copyright (C) 1999 - 2007 Simon Peter, <dn.tlp@gmx.net>, et al.
- *
- * This library is free software; you can redistribute it and/or
- * modify it under the terms of the GNU Lesser General Public
- * License as published by the Free Software Foundation; either
- * version 2.1 of the License, or (at your option) any later version.
- *
- * This library 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
- * Lesser General Public License for more details.
- *
- * You should have received a copy of the GNU Lesser General Public
- * License along with this library; if not, write to the Free Software
- * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
- *
- * rix.cpp - Softstar RIX OPL Format Player by palxex <palxex.ys168.com>
- * BSPAL <BSPAL.ys168.com>
- */
- #include <cstring>
- #include <cstdlib>
- #include "rix.h"
- using namespace std;
- #if !defined(_WIN32) || defined(__SYMBIAN32__)
- #define stricmp strcasecmp
- #endif
- #if defined(__hppa__) || \
- defined(__m68k__) || defined(mc68000) || defined(_M_M68K) || \
- (defined(__MIPS__) && defined(__MISPEB__)) || \
- defined(__ppc__) || defined(__POWERPC__) || defined(_M_PPC) || \
- defined(__sparc__)
- // big endian
- #define RIX_SWAP32(a) (((a) << 24) | (((a) << 8) & 0x00FF0000) | (((a) >> 8) & 0x0000FF00) | ((a) >> 24))
- #else
- // little endian
- #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;
- /*** public methods *************************************/
- CPlayer *CrixPlayer::factory(Copl *newopl)
- {
- return new CrixPlayer(newopl);
- }
- CrixPlayer::CrixPlayer(Copl *newopl)
- : CPlayer(newopl), flag_mkf(0), file_buffer(0), rix_buf(0)
- #if USE_RIX_EXTRA_INIT
- , extra_regs(NULL), extra_vals(NULL), extra_length(0)
- #endif
- {
- }
- CrixPlayer::~CrixPlayer()
- {
- if(file_buffer)
- delete [] file_buffer;
- #if USE_RIX_EXTRA_INIT
- if (extra_regs) delete[] extra_regs;
- if (extra_vals) delete[] extra_vals;
- #endif
- }
- #if USE_RIX_EXTRA_INIT
- void CrixPlayer::set_extra_init(uint32_t* regs, uint8_t* datas, int n)
- {
- extra_length = n;
- if (extra_regs) delete[] extra_regs;
- if (extra_vals) delete[] extra_vals;
- if (n > 0)
- {
- extra_regs = new uint32_t[n];
- extra_vals = new uint8_t[n];
- if (extra_regs) memcpy(extra_regs, regs, n * sizeof(uint32_t));
- if (extra_vals) memcpy(extra_vals, datas, n * sizeof(uint8_t));
- }
- else
- {
- extra_regs = NULL;
- extra_vals = NULL;
- }
- }
- #endif
- 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, bool reinit)
- {
- play_end = 0;
- pos = 0;
- if (reinit)
- {
- 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;
- 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;
- }
- if (reinit)
- {
- opl->init();
- opl->write(1, 32); // go to OPL2 mode
- set_new_int();
- data_initial();
- }
- }
- unsigned int 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;
- }
- /*------------------Implemention----------------------------*/
- inline void CrixPlayer::set_new_int()
- {
- // if(!ad_initial()) exit(1);
- 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)
- {
- // uint16_t i = p2+a0b0_data2[index];
- 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_a0b0_reg(6);
- ad_a0b0_reg(7);
- ad_a0b0_reg(8);
- ad_a0b0l_reg_(8,0x18,0);
- ad_a0b0l_reg_(7,0x1F,0);
- // This is required for correct attack effect, by louyihua
- #if USE_RIX_EXTRA_INIT
- if (extra_regs && extra_vals && extra_length > 0)
- {
- for (uint32_t i = 0; i < extra_length; i++)
- opl->write(extra_regs[i], extra_vals[i]);
- }
- #else
- opl->write(0xa8, 87);
- opl->write(0xb8, 9);
- opl->write(0xa7, 3);
- opl->write(0xb7, 15/*10*/); // Changed from 10 (original value) to 15 for better quality
- #endif
- }
- bd_modify = 0;
- ad_bd_reg();
- 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++;
- }
- ad_bd_reg();
- ad_08_reg();
- for(i=0;i<9;i++) ad_a0b0_reg(i);
- e0_reg_flag = 0x20;
- for(i=0;i<18;i++) ad_bop(0xE0+reg_data[i],0);
- ad_bop(1,e0_reg_flag);
- return 1;//ad_test();
- }
- /*----------------------------------------------------------*/
- 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; /* aging */
- 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) /* important !*/
- {
- 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);
- }
|