Mercurial > audlegacy-plugins
view src/adplug/core/rix.cxx @ 3168:7e58928783b3
alsa-ng: Set thread handles to NULL when threads exit.
author | William Pitcock <nenolod@atheme.org> |
---|---|
date | Fri, 15 May 2009 00:05:48 -0500 |
parents | be86c72a06c9 |
children |
line wrap: on
line source
/* * 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., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA * * rix.cpp - Softstar RIX OPL Format Player by palxex <palxex.ys168.com> * BSPAL <BSPAL.ys168.com> */ #include "rix.h" #include "debug.h" #include <cstdlib> const unsigned char CrixPlayer::adflag[] = { 0, 0, 0, 1, 1, 1, 0, 0, 0, 1, 1, 1, 0, 0, 0, 1, 1, 1 }; const unsigned char CrixPlayer::reg_data[] = { 0, 1, 2, 3, 4, 5, 8, 9, 10, 11, 12, 13, 16, 17, 18, 19, 20, 21 }; const unsigned char CrixPlayer::ad_C0_offs[] = { 0, 1, 2, 0, 1, 2, 3, 4, 5, 3, 4, 5, 6, 7, 8, 6, 7, 8 }; const unsigned char 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 unsigned char 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 }; unsigned char CrixPlayer::for40reg[] = { 0x7F, 0x7F, 0x7F, 0x7F, 0x7F, 0x7F, 0x7F, 0x7F, 0x7F, 0x7F, 0x7F, 0x7F, 0x7F, 0x7F, 0x7F, 0x7F, 0x7F, 0x7F }; unsigned short 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), buf_addr (0) { } CrixPlayer::~CrixPlayer () { if (file_buffer) delete[]file_buffer; } bool CrixPlayer::load (VFSFile * fd, const CFileProvider & fp) { binistream *f = fp.open (fd); if (!f) return false; unsigned long i = 0; std::string filename (fd->uri); 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 unsigned char[fp.filesize (f) + 1]; f->seek (0); while (!f->eof ()) file_buffer[i++] = f->readInt (1); length = i; fp.close (f); if (!flag_mkf) buf_addr = 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 (unsigned short) * 300); memset (a0b0_data2, 0, sizeof (unsigned short) * 11); memset (a0b0_data3, 0, 18); memset (a0b0_data4, 0, 18); memset (a0b0_data5, 0, 96); memset (addrs_head, 0, 96); memset (insbuf, 0, 28 * sizeof (unsigned short)); memset (displace, 0, 11 * sizeof (unsigned short)); memset (reg_bufs, 0, 18 * sizeof (ADDT)); if (flag_mkf) { unsigned int *buf_index = (unsigned int *) file_buffer; int offset1 = buf_index[subsong], offset2; while ((offset2 = buf_index[++subsong]) == offset1); length = offset2 - offset1 + 1; buf_addr = file_buffer + offset1; } opl->init (); opl->write (1, 32); // go to OPL2 mode set_new_int (); data_initial (); } unsigned int CrixPlayer::getsubsongs () { if (flag_mkf) { unsigned int *buf_index = (unsigned int *) file_buffer; int songs = 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 unsigned short i; pause_flag = 1; for (i = 0; i < 11; i++) switch_ad_bd (i); } /*----------------------------------------------------------*/ inline void CrixPlayer::ad_a0b0l_reg_ (unsigned short index, unsigned short p2, unsigned short p3) { // unsigned short i = p2+a0b0_data2[index]; a0b0_data4[index] = p3; a0b0_data3[index] = p2; } inline void CrixPlayer::data_initial () { rhythm = buf_addr[2]; mus_block = (buf_addr[0x0D] << 8) + buf_addr[0x0C]; ins_block = (buf_addr[0x09] << 8) + buf_addr[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); } bd_modify = 0; // ad_bd_reg(); band = 0; music_on = 1; } /*----------------------------------------------------------*/ inline unsigned short CrixPlayer::ad_initial () { register unsigned short i, j, k = 0; for (i = 0; i < 25; i++) { f_buffer[i * 12] = (unsigned int) ((i * 24 + 10000) * 0.27461678223 + 4) >> 3; for (int t = 1; t < 12; t++) f_buffer[i * 12 + t] = (unsigned int) ((double) f_buffer[i * 12 + t - 1] * 1.06); } 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 (unsigned short reg, unsigned short 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 () { unsigned short 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 unsigned short CrixPlayer::rix_proc () { unsigned char ctrl = 0; if (music_on == 0 || pause_flag == 1) return 0; band = 0; while (buf_addr[I] != 0x80 && I < length - 1) { band_low = buf_addr[I - 1]; ctrl = buf_addr[I]; I += 2; switch (ctrl & 0xF0) { case 0x90: rix_get_ins (); rix_90_pro (ctrl & 0x0F); break; case 0xA0: rix_A0_pro (ctrl & 0x0F, ((unsigned short) 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; unsigned char *baddr = (&buf_addr[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 (unsigned short 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 (unsigned short ctrl_l, unsigned short 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 (unsigned short index, unsigned short v) /* important ! */ { short high = 0, low = 0; unsigned int 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 (unsigned short index, unsigned short p2, unsigned short p3) { unsigned short data; unsigned short 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 (unsigned short ctrl_l, unsigned short 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 (unsigned short ctrl_l, unsigned short index) { register unsigned short 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 (unsigned short 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 (unsigned short index, unsigned short *insb, unsigned short value) { register unsigned short 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 (unsigned short index) { unsigned short 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 (unsigned short index) { unsigned short 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 (unsigned short index) { unsigned short 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 (unsigned short index) { unsigned short 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 (unsigned short index) { unsigned short 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 (unsigned short index) { unsigned int res = 0; unsigned short 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 () { unsigned short data = rhythm < 1 ? 0 : 0x20; data |= bd_modify; ad_bop (0xBD, data); } /*--------------------------------------------------------------*/ inline void CrixPlayer::ad_a0b0_reg (unsigned short 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); }