Mercurial > audlegacy-plugins
view src/adplug/core/fmc.cxx @ 3072:8f56795e348d
crossfade-ng: correctly workaround possible buffer starvation issues with streams
author | William Pitcock <nenolod@atheme.org> |
---|---|
date | Sun, 26 Apr 2009 00:01:59 -0500 |
parents | a31410f560cf |
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 fmc.cpp - FMC Loader by Riven the Mage <riven@ok.ru> */ #include "fmc.h" /* -------- Public Methods -------------------------------- */ CPlayer * CfmcLoader::factory (Copl * newopl) { return new CfmcLoader (newopl); } bool CfmcLoader::load (VFSFile * fd, const CFileProvider & fp) { binistream *f = fp.open (fd); if (!f) return false; const unsigned char conv_fx[16] = { 0, 1, 2, 3, 4, 8, 255, 255, 255, 255, 26, 11, 12, 13, 14, 15 }; int i, j, k, t = 0; // read header f->readString (header.id, 4); f->readString (header.title, 21); header.numchan = f->readInt (1); // 'FMC!' - signed ? if (strncmp (header.id, "FMC!", 4)) { fp.close (f); return false; } // init CmodPlayer realloc_instruments (32); realloc_order (256); realloc_patterns (64, 64, header.numchan); init_trackord (); // load order for (i = 0; i < 256; i++) order[i] = f->readInt (1); f->ignore (2); // load instruments for (i = 0; i < 32; i++) { instruments[i].synthesis = f->readInt (1); instruments[i].feedback = f->readInt (1); instruments[i].mod_attack = f->readInt (1); instruments[i].mod_decay = f->readInt (1); instruments[i].mod_sustain = f->readInt (1); instruments[i].mod_release = f->readInt (1); instruments[i].mod_volume = f->readInt (1); instruments[i].mod_ksl = f->readInt (1); instruments[i].mod_freq_multi = f->readInt (1); instruments[i].mod_waveform = f->readInt (1); instruments[i].mod_sustain_sound = f->readInt (1); instruments[i].mod_ksr = f->readInt (1); instruments[i].mod_vibrato = f->readInt (1); instruments[i].mod_tremolo = f->readInt (1); instruments[i].car_attack = f->readInt (1); instruments[i].car_decay = f->readInt (1); instruments[i].car_sustain = f->readInt (1); instruments[i].car_release = f->readInt (1); instruments[i].car_volume = f->readInt (1); instruments[i].car_ksl = f->readInt (1); instruments[i].car_freq_multi = f->readInt (1); instruments[i].car_waveform = f->readInt (1); instruments[i].car_sustain_sound = f->readInt (1); instruments[i].car_ksr = f->readInt (1); instruments[i].car_vibrato = f->readInt (1); instruments[i].car_tremolo = f->readInt (1); instruments[i].pitch_shift = f->readInt (1); f->readString (instruments[i].name, 21); } // load tracks for (i = 0; i < 64; i++) { if (f->ateof ()) break; for (j = 0; j < header.numchan; j++) { for (k = 0; k < 64; k++) { fmc_event event; // read event event.byte0 = f->readInt (1); event.byte1 = f->readInt (1); event.byte2 = f->readInt (1); // convert event tracks[t][k].note = event.byte0 & 0x7F; tracks[t][k].inst = ((event.byte0 & 0x80) >> 3) + (event.byte1 >> 4) + 1; tracks[t][k].command = conv_fx[event.byte1 & 0x0F]; tracks[t][k].param1 = event.byte2 >> 4; tracks[t][k].param2 = event.byte2 & 0x0F; // fix effects if (tracks[t][k].command == 0x0E) // 0x0E (14): Retrig tracks[t][k].param1 = 3; if (tracks[t][k].command == 0x1A) // 0x1A (26): Volume Slide { if (tracks[t][k].param1 > tracks[t][k].param2) { tracks[t][k].param1 -= tracks[t][k].param2; tracks[t][k].param2 = 0; } else { tracks[t][k].param2 -= tracks[t][k].param1; tracks[t][k].param1 = 0; } } } t++; } } fp.close (f); // convert instruments for (i = 0; i < 31; i++) buildinst (i); // order length for (i = 0; i < 256; i++) { if (order[i] >= 0xFE) { length = i; break; } } // data for Protracker activechan = (0xffffffff >> (32 - header.numchan)) << (32 - header.numchan); nop = t / header.numchan; restartpos = 0; // flags flags = Faust; rewind (0); return true; } float CfmcLoader::getrefresh () { return 50.0f; } std::string CfmcLoader::gettype () { return std::string ("Faust Music Creator"); } std::string CfmcLoader::gettitle () { return std::string (header.title); } std::string CfmcLoader::getinstrument (unsigned int n) { return std::string (instruments[n].name); } unsigned int CfmcLoader::getinstruments () { return 32; } /* -------- Private Methods ------------------------------- */ void CfmcLoader::buildinst (unsigned char i) { inst[i].data[0] = ((instruments[i].synthesis & 1) ^ 1); inst[i].data[0] |= ((instruments[i].feedback & 7) << 1); inst[i].data[3] = ((instruments[i].mod_attack & 15) << 4); inst[i].data[3] |= (instruments[i].mod_decay & 15); inst[i].data[5] = ((15 - (instruments[i].mod_sustain & 15)) << 4); inst[i].data[5] |= (instruments[i].mod_release & 15); inst[i].data[9] = (63 - (instruments[i].mod_volume & 63)); inst[i].data[9] |= ((instruments[i].mod_ksl & 3) << 6); inst[i].data[1] = (instruments[i].mod_freq_multi & 15); inst[i].data[7] = (instruments[i].mod_waveform & 3); inst[i].data[1] |= ((instruments[i].mod_sustain_sound & 1) << 5); inst[i].data[1] |= ((instruments[i].mod_ksr & 1) << 4); inst[i].data[1] |= ((instruments[i].mod_vibrato & 1) << 6); inst[i].data[1] |= ((instruments[i].mod_tremolo & 1) << 7); inst[i].data[4] = ((instruments[i].car_attack & 15) << 4); inst[i].data[4] |= (instruments[i].car_decay & 15); inst[i].data[6] = ((15 - (instruments[i].car_sustain & 15)) << 4); inst[i].data[6] |= (instruments[i].car_release & 15); inst[i].data[10] = (63 - (instruments[i].car_volume & 63)); inst[i].data[10] |= ((instruments[i].car_ksl & 3) << 6); inst[i].data[2] = (instruments[i].car_freq_multi & 15); inst[i].data[8] = (instruments[i].car_waveform & 3); inst[i].data[2] |= ((instruments[i].car_sustain_sound & 1) << 5); inst[i].data[2] |= ((instruments[i].car_ksr & 1) << 4); inst[i].data[2] |= ((instruments[i].car_vibrato & 1) << 6); inst[i].data[2] |= ((instruments[i].car_tremolo & 1) << 7); inst[i].slide = instruments[i].pitch_shift; }